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. 《数学之美》第17章 由电视剧《暗算》所想到的—谈谈密码学的数学原理
  2. buildroot管理uboot+kernel+rootfs
  3. 获取java异常堆栈信息_Java 实例 - 获取异常的堆栈信息
  4. 大规模分布式跟踪系统的理论
  5. 力扣1317.将整数转换为两个无零整数之和
  6. android如何使用BroadcastReceiver后台实现来电通话记录的监听并存取到sqllite数据库通过Contentprovilder实现接口...
  7. 随机梯度下降(SGD)和批量梯度下降(BGD)的区别
  8. 简化超长的函数指针类型
  9. linux使用命令修改端口映射,linux端口映射命令是什么?使用iptables做端口映射shell的方法...
  10. JavaScript怎么安装_WebStrom 2019安装教程
  11. java毕业设计——基于java+JSP+MyEclipse的网上订餐系统设计与实现(毕业论文+程序源码)——网上订餐系统
  12. 鸿蒙系统安装电视家,华为荣耀智慧屏系统更新,支持安装第三方APP,网友:华为妥协了...
  13. The car's gossip
  14. implement 和 extends 的区别
  15. php 新浪微博登陆,PHP使用新浪微博登入第三方网站实例代码
  16. 基于jq的别踩白块儿小游戏
  17. 实习僧的字体加密破解
  18. 数据库系统概论实验二——创建及管理数据库
  19. C#根据驱动名称获得USB串口的端口实例
  20. Flyway学习和使用

热门文章

  1. 大学溃败始于老实人吃亏
  2. 大猩猩部队优化算法(Gorilla Troops Optimizer,GTO)
  3. GitHub Copilot一直等待问题
  4. He-HTLC:重新审视HTLC中的激励措施
  5. C#调用存储过程的几个方法
  6. Java sleep 替换_Thread.Sleep在Java中的替代品
  7. java大根堆和小根堆
  8. 大学计算机案例教程旧照片修复,对损坏老照片进行修复的PS教程
  9. 珍藏的老照片损坏如何修复?今天分享PS老照片修复教程别错过!
  10. 工具 mosquitto