本小节主要是介绍yaf_dispatcher_fix_default函数的做用以及实现细节。php
static inline void yaf_dispatcher_fix_default(yaf_dispatcher_t *dispatcher, yaf_request_t *request)html
该函数用于对module/controller/action三个值进行标准化处理。以便于后续对具体类文件进行检索定位。git
示例github
yaf_dispatcher_t *dispatcher数组
zval数据类型, Yaf_Dispatcher类实例。函数
yaf_request_t *request测试
zval数据类型, Yaf_Reqeust_Http实例spa
/** * 对module/controller/action作相应的大小写处理 * 处理逻辑以下 * 1) 从yaf_request_ce中读取出module, constorller, action * 2) 若是1)读取的参数为空或非字符类型, 则从yaf_dispatcher_ce中读取相应的default值 * 3) module的处理是先所有小写而后首字母大写 * 4) controller是先所有小写,而后依据'_'为分隔符作首字母大写处理, 例如 Index_sub => Index_Sub * 5) action的处理是所有小写 * 6) 以上处理完成后均将处理后的结果值回写yaf_request_ce对应的成员属性 */ static inline void yaf_dispatcher_fix_default(yaf_dispatcher_t *dispatcher, yaf_request_t *request) /* {{{ */ { zval *module, *controller, *action; module = zend_read_property(yaf_request_ce, request, ZEND_STRL(YAF_REQUEST_PROPERTY_NAME_MODULE), 1, NULL); action = zend_read_property(yaf_request_ce, request, ZEND_STRL(YAF_REQUEST_PROPERTY_NAME_ACTION), 1, NULL); controller = zend_read_property(yaf_request_ce, request, ZEND_STRL(YAF_REQUEST_PROPERTY_NAME_CONTROLLER), 1, NULL); // module先所有小写而后首字母大写 if (Z_TYPE_P(module) != IS_STRING || !Z_STRLEN_P(module)) { zval *default_module = zend_read_property(yaf_dispatcher_ce, dispatcher, ZEND_STRL(YAF_DISPATCHER_PROPERTY_NAME_MODULE), 1, NULL); zend_update_property(yaf_request_ce, request, ZEND_STRL(YAF_REQUEST_PROPERTY_NAME_MODULE), default_module); } else { char *p = zend_str_tolower_dup(Z_STRVAL_P(module), Z_STRLEN_P(module)); *p = toupper(*p); zend_update_property_stringl(yaf_request_ce, request, ZEND_STRL(YAF_REQUEST_PROPERTY_NAME_MODULE), p, Z_STRLEN_P(module)); efree(p); } // controller先所有小写而后以'_'或'\\'为界首字母大写, 例 index_sub => Index_Sub if (Z_TYPE_P(controller) != IS_STRING || !Z_STRLEN_P(controller)) { zval *default_controller = zend_read_property(yaf_dispatcher_ce, dispatcher, ZEND_STRL(YAF_DISPATCHER_PROPERTY_NAME_CONTROLLER), 1, NULL); zend_update_property(yaf_request_ce, request, ZEND_STRL(YAF_REQUEST_PROPERTY_NAME_CONTROLLER), default_controller); } else { char *q, *p = zend_str_tolower_dup(Z_STRVAL_P(controller), Z_STRLEN_P(controller)); /** * Upper controller name * eg: Index_sub -> Index_Sub */ q = p; *q = toupper(*q); while (*q != '\0') { if (*q == '_' || *q == '\\') { if (*(q+1) != '\0') { *(q+1) = toupper(*(q+1)); q++; } } q++; } zend_update_property_stringl(yaf_request_ce, request, ZEND_STRL(YAF_REQUEST_PROPERTY_NAME_CONTROLLER), p, Z_STRLEN_P(controller)); efree(p); } // action所有小写 if (Z_TYPE_P(action) != IS_STRING || !Z_STRLEN_P(action)) { zval *default_action = zend_read_property(yaf_dispatcher_ce, dispatcher, ZEND_STRL(YAF_DISPATCHER_PROPERTY_NAME_ACTION), 1, NULL); zend_update_property(yaf_request_ce, request, ZEND_STRL(YAF_REQUEST_PROPERTY_NAME_ACTION), default_action); } else { char *p = zend_str_tolower_dup(Z_STRVAL_P(action), Z_STRLEN_P(action)); zend_update_property_stringl(yaf_request_ce, request, ZEND_STRL(YAF_REQUEST_PROPERTY_NAME_ACTION), p, Z_STRLEN_P(action)); efree(p); } } /* }}} */
在这里,须要注意一个细节, 就是controller的那段处理逻辑。在C语言中有两种定义字符串的方式.net
二者的区别体如今code
那么在上述代码里是怎么实现的字符修改呢?为了理解上面的知识点,写段测试代码用于验证。以下
#include<stdio.h> #include<ctype.h> #include<string.h> int main() { const char *controller = "index_sub"; char *q, *p = strdup(controller); q = p; *q = toupper(*q); while (*q != '\0') { if (*q == '_') { if (*(q+1) != '\0') { *(q+1) = toupper(*(q+1)); q++; } } q++; } printf("%s\n", p); return 0; }