php define 修改_php define常量详解

看手册说define定义的常量只允许:

仅允许标量和 null。

标量的类型是 integer, float,string 或者 boolean。

也能够定义常量值的类型为 resource ,但并不推荐这么做,可能会导致未知状况的发生。

今天阅读php源码,发现define的第二个参数其实也可以是一个对象。

例子:

复制代码 代码示例:

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('>_

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值