class A {

public function __toString() {

return 'bar';

}

}

$a = new A();

define('foo', $a);

echo foo;

// 输出bar

php中的define究竟是如何实现的:ZEND_FUNCTION(define)

{

char *name;

int name_len;

zval *val;

zval *val_free = NULL;

zend_bool non_cs = 0;

int case_sensitive = CONST_CS;

zend_constant c;

// 接收3个参数,string,zval,bool

if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "sz|b", &name, &name_len, &val, &non_cs) == FAILURE) {

return;

}

// 是否大小写敏感

if(non_cs) {

case_sensitive = 0;

}

// 如果define类常量,则报错

if (zend_memnstr(name, "::", sizeof("::") - 1, name + name_len)) {

zend_error(E_WARNING, "Class constants cannot be defined or redefined");

RETURN_FALSE;

}

// 获取真正的值,用val保存

repeat:

switch (Z_TYPE_P(val)) {

case IS_LONG:

case IS_DOUBLE:

case IS_STRING:

case IS_BOOL:

case IS_RESOURCE:

case IS_NULL:

break;

case IS_OBJECT:

if (!val_free) {

if (Z_OBJ_HT_P(val)->get) {

val_free = val = Z_OBJ_HT_P(val)->get(val TSRMLS_CC);

goto repeat;

} else if (Z_OBJ_HT_P(val)->cast_object) {

ALLOC_INIT_ZVAL(val_free);

if (Z_OBJ_HT_P(val)->cast_object(val, val_free, IS_STRING TSRMLS_CC) == SUCCESS) {

val = val_free;

break;

}

}

}

/* no break */

default:

zend_error(E_WARNING,"Constants may only evaluate to scalar values");

if (val_free) {

zval_ptr_dtor(&val_free);

}

RETURN_FALSE;

}

// 构建常量

c.value = *val;

zval_copy_ctor(&c.value);

if (val_free) {

zval_ptr_dtor(&val_free);

}

c.flags = case_sensitive; /* non persistent */ // 如果大小写不敏感,则为0,敏感则为1

c.name = zend_strndup(name, name_len);

c.name_len = name_len+1;

c.module_number = PHP_USER_CONSTANT; // 标注非内核常量,而是用户定义的常量

// 注册常量

if (zend_register_constant(&c TSRMLS_CC) == SUCCESS) {

RETURN_TRUE;

} else {

RETURN_FALSE;

}

}

注意以repeat开始的一段循环,还用到了goto语句T_T

这段代码的作用为:

对于int,float,string,bool,resource,null,则实际定义的常量时直接使用这些值

对于object,则需要将object转成上述6个类型之一(如果转型之后依然是object,则继续转型)

如何将object成6个类型之一呢?从代码上看有2种手段:if (Z_OBJ_HT_P(val)->get) {

val_free = val = Z_OBJ_HT_P(val)->get(val TSRMLS_CC);

goto repeat;

}

// __toString()方法会在cast_object中被调用

else if (Z_OBJ_HT_P(val)->cast_object) {

ALLOC_INIT_ZVAL(val_free);

if (Z_OBJ_HT_P(val)->cast_object(val, val_free, IS_STRING TSRMLS_CC) == SUCCESS)

{

val = val_free;

break;

}

}

1,Z_OBJ_HT_P(val)->get ,宏展开之后为(*val).value.obj.handlers->get

2,Z_OBJ_HT_P(val)->cast_object,宏展开之后为(*val).value.obj.handlers->cast_object

handlers是一个包含很多函数指针的结构体,具体定义参见_zend_object_handlers 。该结构体中的函数指针均用于操作object,比如读取/修改对象属性、获取/调用对象方法等等...get和cast_object也是其中之一。

对于一般的对象,php提供了标准的cast_object函数zend_std_cast_object_tostring,代码位于php-src/zend/zend-object-handlers.c中:ZEND_API int zend_std_cast_object_tostring(zval *readobj, zval *writeobj, int type TSRMLS_DC) /* {{{ */

{

zval *retval;

zend_class_entry *ce;

switch (type) {

case IS_STRING:

ce = Z_OBJCE_P(readobj);

// 如果用户的class中定义了__toString,则尝试调用

if (ce->__tostring &&

(zend_call_method_with_0_params(&readobj, ce, &ce->__tostring, "__tostring", &retval) || EG(exception))) {

……

}

return FAILURE;

……

}

return FAILURE;

}

从上述具体实现来看,默认的cast_object就是去寻找class中的__tostring方法然后调用...

回到刚开始的例子,define('foo', $a) ,由于$a是A的实例,并且class A中定义了__toString,因此实际上foo常量就等于toString的返回值bar。

ps:继续挖掘一点小细节.

1,define有返回值

通常我们定义常量直接写成:define('foo', 123); 不过从define的实现上来看,它是有返回值的。根据手册上的描述:

成功时返回 TRUE, 或者在失败时返回 FALSE。

什么情况下define会失败呢?

举个例子:define('PHP_INT_MAX', 1); // 返回FALSE

define('FOO', 1); // 返回TRUE

define('FOO', 2); // 返回FALSE

上面代码包含了两种情况,一是我们尝试重新定义php内核的预定义常量,比如PHP_INT_MAX,这显然会失败。第二种情况是我们曾经在代码的某个位置定义过了一个常量FOO,然后又在接下来的程序中再次定义它,这也会造成失败。因此,在编码时最好将所有需要定义的常量写在一起,以免造成name重复。

2,常量名没有限制

再次回顾一下define的实现,其中仅仅判断name是否为XXX::YYY这种形式。

换句话说,define几乎对其name不做任何要求,当然也不需要name是一个合法的php变量名。因此,我们可以让define的常量取一些稀奇古怪的名称。例如:define('>_echo >_

不过如果定义了这样的常量,是没法直接使用的,会报语法错误。正确的使用方法如下:define('>_echo constant('>_输出123

php define 常量,phpdefine常量详解相关推荐

  1. 转 常量指针和指针常量的区别详解

    传送门 常量指针和指针常量的区别详解 在C/C++中关键字const用来定义一个只读的变量或者对象,有如下优点     (1)便于类型检查,如函数的函数 fun(const int a) a的值不允许 ...

  2. C++ 常量类型 const 详解

    1.什么是const?  常类型是指使用类型修饰符const说明的类型,常类型的变量或对象的值是不能被更新的.(当然,我们可以偷梁换柱进行更新:)  2.为什么引入const?  const 推出的初 ...

  3. class字节码文件中的常量池结构详解

    文章目录 前言 方法区 常量池基本结构 JVM 所定义的11种常量 常量池元素的复合结构 常量池的结束位置 常量池元素总数量 第一个常量池元素 父类常量 变量型常量池元素 自己的学习笔记,部分节选自& ...

  4. C++ 常量引用用法详解

    "常量引用"其实是"对 const 的引用"的简称. 顾名思义,它把它所指向的对象看作是常量(不一定是常量),因此不可以通过该引用来修改它所指向的对象的值. 严 ...

  5. TensorFlow创建常量(tf.constant)详解

    在TensorFlow API中创建常量的函数原型如下所示: tf.constant(value,dtype=None,shape=None,name='Const',verify_shape=Fal ...

  6. #ifndef HeaderName_h #define HeaderName_h #endif 使用详解

    想必很多人都看到过头文件中写有:#ifndef HeaderName_h                                                #define HeaderNa ...

  7. JVM详解之:运行时常量池

    文章目录 简介 class文件中的常量池 运行时常量池 静态常量详解 String常量 数字常量 符号引用详解 String Pool字符串常量池 总结 简介 JVM在运行的时候会对class文件进行 ...

  8. 【php基础入门】运算符、流程控制语句及单双引号区别和模块化编程的使用详解

    目录 字符串的单引号和双引号区别 PHP 运算符 算数运算符 字符运算符 关系运算符 逻辑运算符 赋值运算符 错误抑制符 三元运算符 PHP 流程控制 分支语句 循环语句 模块化编程 今日相关函数 今 ...

  9. PHP常量详解:define和const的区别

    常量是一个简单的标识符.在脚本执行期间该值不能改变(除了所谓的魔术常量,他们其实不是常量).常量默认大小写敏感.通常常量标识符总是大写的. 可以用define()函数来定义常量.在php5.3.0以后 ...

最新文章

  1. Docker 用法总结之:管理工具 shipyard 的具体使用指南
  2. 微信商城小程序操作为产品增加颜色尺寸长度等多规格内容
  3. 【集训队作业2018】喂鸽子
  4. sensor曝光量和曝光行的区别_4个要点,告诉你拼多多新的产品怎么增加曝光量!...
  5. 让pt-slave-restart支持MariaDB
  6. oracle 安装 挂载磁盘组_ora-15077,ASM磁盘组不能挂载
  7. java spring流程_浅谈SpringMVC执行过程
  8. php中$_get和$_post如何使用,怎么使用超级全局变量$_POST与$_GET
  9. 这样去写你的 HTML
  10. hashmap 扩容是元素还是数组_HashMap 中的容量与扩容实现
  11. 微信二次“回应”iOS 13.2杀后台问题:已找到解决方案
  12. MySQL学习记录===待续
  13. 如何监视SQL Server索引的总大小
  14. 在线Excel文件解析转换成JSON格式
  15. Python删除文件第一行
  16. iPhoneiPod Touch定位编写
  17. 获取请求真实IP地址的工具类
  18. nas磁盘用什么软件测试,手把手教你从NAS拿点空间当电脑硬盘使用 iSCSI开启网络硬盘共享...
  19. Linux命令之md5sum的作用以及使用方法(md5是什么?)
  20. symbian学习笔记

热门文章

  1. matlab 去除高斯噪声
  2. SpringBoot接入支付宝支付API
  3. v-if和v-for不能一起使用的原因以及解决办法
  4. 2017百度前端技术学院习题-03
  5. 数据分析项目实战day2
  6. Openwrt-18.06.2从u盘启动
  7. 【Arduino 和 HC-05 蓝牙模块完整教程】
  8. Chrome不允许在页面关闭或导航跳转时发送同步请求
  9. ASP.NET Web Api 使用CacheCow和ETag缓存资源(转载)
  10. 环信SDK 头像、昵称、表情自定义和群聊设置的实现 一(附源码)