因为工作需要使用ThinkPHP3.2, 所以请叫我考古学家!
A方法
A方法用于在内部实例化控制器,调用格式:A(‘[项目://][分组/]模块’,’控制器层名称’)
源码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24
|
function A($name, $layer = '', $level = 0) { static $_action = array(); $layer = $layer ?: C('DEFAULT_C_LAYER'); $level = $level ?: ($layer == C('DEFAULT_C_LAYER') ? C('CONTROLLER_LEVEL') : 1); if (isset($_action[$name.$layer])) return $_action[$name.$layer];
$class = parse_res_name($name, $layer, $level); if (class_exists($class)) { $action = new $class(); $_action[$name.$layer] = $action;
return $action; } else { return false; } }
|
B方法
这是随着行为应运而生的新生函数,可以执行某个行为,例如
就是在项目开始之前,执行这个行为定义的所有函数。支持2个参数,第二个参数支持需要接受一个数组,例如
1
| B('app_begin',array("name"=& gt;"tdweb","time"=>time()));
|
源码
1 2 3 4 5 6 7 8 9 10 11 12 13 14
|
function B($name, $tag = '', &$params = null) { if ('' == $tag) { $name .= 'Behavior'; }
return \Think\Hook::exec($name, $tag, $params); }
|
C方法
C方法是Think用于设置、获取,以及保存配置参数的方法,使用频率较高。
表示设置DB_NAME配置参数的值为think,由于配置参数不区分大小写,所以下面的写法也是一样:
1 2 3
| $config['user_id'] = 1; $config['user_type'] = 1; C($config);
|
源码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42
|
function C($name = null, $value = null, $default = null) { static $_config = array(); if (empty($name)) { return $_config; } if (is_string($name)) { if (!strpos($name, '.')) { $name = strtoupper($name); if (is_null($value)) return isset($_config[$name]) ? $_config[$name] : $default; $_config[$name] = $value;
return null; } $name = explode('.', $name); $name[0] = strtoupper($name[0]); if (is_null($value)) return isset($_config[$name[0]][$name[1]]) ? $_config[$name[0][$name[1]]] : $default; $_config[$name[0][$name[1]]] = $value;
return null; } if (is_array($name)) { $_config = array_merge($_config, array_change_key_case($name, CASE_UPPER));
return null; }
return null; }
|
D方法
D方法应该是用的比较多的方法了,用于实例化自定义模型类,
是Think框架对Model类实例化的一种封装,并实现了单例模式,支持跨项目和分组调用,调用格式如下:
D(‘[项目://][分组/]模型’,’模型层名称’)
1 2 3 4 5
| D('Admin://User')
D('Admin/User')
|
- 由于增加了分层模型的支持,所以D方法也可以实例化其他的模型,例如:
1 2 3 4 5
| $User = D('User','Service');
$User = D('User','Logic');
|
源码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31
|
function D($name = '', $layer = '') { if (empty($name)) return new Think\Model; static $_model = array(); $layer = $layer ?: C('DEFAULT_M_LAYER'); if (isset($_model[$name.$layer])) return $_model[$name.$layer]; $class = parse_res_name($name, $layer); if (class_exists($class)) { $model = new $class(basename($name)); } elseif (false === strpos($name, '/')) { if (! C('APP_USE_NAMESPACE')) { import('Common/' . $layer . '/' . $class); } else { $class = '\\Common\\' . $layer . '\\' . $name . $layer; } $model = class_exists($class) ? new $class($name) : new Think\Model($name); } else { Think\Log::record('D方法实例化没找到模型类' . $class, Think\Log::NOTICE); $model = new Think\Model(basename($name)); } $_model[$name.$layer] = $model;
return $model; }
|
E方法
抛出异常处理
源码
1 2 3 4 5 6 7 8 9 10
|
function E($msg, $code = 0) { throw new Think\Exception($msg, $code); }
|
F方法
F方法其实是S方法的一个子集功能,仅用于简单数据缓存,并且只能支持文件形式,不支持缓存有效期,
因为采用的是返回方式,所以其效率较S方法较高,因此我们也称之为快速缓存方法。
默认的保存起始路径是DATA_PATH
(该常量在默认配置位于RUNTIME_PATH.'Data/'
下面),
也就是说会生成文件名为DATA_PATH.'data.'
的缓存文件。
注意:确保你的缓存标识的唯一,避免数据覆盖和冲突。
1 2
| F('user/data',$data); F('user/data');
|
- 支持批量删除功能,尤其是针对子目录缓存的情况,假设我们要删除user子目录下面的所有缓存数据,可以使用:
源码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39
|
function F($name, $value = '', $path = DATA_PATH) { static $_cache = array(); $filename = $path . $name . '.php'; if ('' !== $value) { if (is_null($value)) { if (false !== strpos($name, '*')) { return false; } else { unset($_cache[$name]); return Think\Storage::unlink($filename, 'F'); } } else { Think\Storage::put($filename, serialize($value), 'F'); $_cache[$name] = $value; return null; } } if (isset($_cache[$name])) { return $_cache[$name]; } if (Think\Storage::has($filename, 'F')) { $value = unserialize(Think\Storage::read($filename, 'F')); $_cache[$name] = $value; } else { $value = false; }
return $value; }
|
G方法
G方法的作用包括标记位置和区间统计两个功能
标记位置
表示把当前位置标记为begin标签,并且记录当前位置的执行时间,如果环境支持的话,还能记录内存占用情况。可以在任何位置调用G方法标记。
运行时间统计
标记位置后,我们就可以再次调用G方法进行区间统计了,例如:
1 2 3 4 5 6
| G('begin');
G('end');
echo G('begin','end').'s';
|
G(‘begin’,’end’) 表示统计begin位置到end位置的执行时间(单位是秒),begin必须是一个已经标记过的位置,
如果这个时候end位置还没被标记过,则会自动把当前位置标记为end标签,输出的结果类似于:
默认的统计精度是小数点后4位,如果觉得这个统计精度不够,还可以设置例如:
可能的输出会变成:
内存开销统计
如果你的环境支持内存占用统计的话,还可以使用G方法进行区间内存开销统计(单位为kb)
1
| echo G('begin','end','m').'kb';
|
源码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43
| define('MEMORY_LIMIT_ON',function_exists('memory_get_usage'));
function G($start, $end = '', $dec = 4) { static $_info = array(); static $_mem = array(); if (is_float($end)) { $_info[$start] = $end; } elseif (!empty($end)) { if (!isset($_info[$end])) $_info[$end] = microtime(true); if (MEMORY_LIMIT_ON && $dec == 'm') { if (!isset($_mem[$end])) $_mem[$end] = memory_get_usage(); return number_format(($_mem[$end] - $_mem[$start]) / 1024); } else { return number_format(($_info[$end] - $_info[$start]), $dec); } } else { $_info[$start] = microtime(true); if (MEMORY_LIMIT_ON) $_mem[$start] = memory_get_usage(); }
return null; }
|
I方法
主要用于更加方便和安全的获取系统输入变量,可以用于任何地方,用法格式如下:
I(‘变量类型.变量名’,[‘默认值’],[‘过滤方法’])
1 2
| echo I('get.id'); echo I('get.name');
|
1 2
| echo I('get.id',0); echo I('get.name','');
|
1 2
| echo I('get.name','','htmlspecialchars');
|
源码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161
|
function I($name, $default = '', $filter = null, $datas = null) { static $_PUT = null; if (strpos($name, '/')) { list($name, $type) = explode('/', $name, 2); } elseif (C('VAR_AUTO_STRING')) { $type = 's'; } if (strpos($name, '.')) { list($method, $name) = explode('.', $name, 2); } else { $method = 'param'; } switch (strtolower($method)) { case 'get': $input = &$_GET; break; case 'post': $input = &$_POST; break; case 'put': if (is_null($_PUT)) { parse_str(file_get_contents('php://input'), $_PUT); } $input = &$_PUT; break; case 'param': switch ($_SERVER['REQUEST_METHOD']) { case 'POST': $input = $_POST; break; case 'PUT': if (is_null($_PUT)) { parse_str(file_get_contents('php://input'), $_PUT); } $input = $_PUT; break; default: $input = $_GET; } break; case 'path': $input = array(); if (!empty($_SERVER['PATH_INFO'])) { $depr = C('URL_PATHINFO_DEPR'); $input = explode($depr, trim($_SERVER['PATH_INFO'], $depr)); } break; case 'request': $input = &$_REQUEST; break; case 'session' : $input =& $_SESSION; break; case 'cookie' : $input =& $_COOKIE; break; case 'server' : $input =& $_SERVER; break; case 'globals' : $input =& $GLOBALS; break; case 'data' : $input =& $datas; break; default: return null; } if ('' == $name) { $data = $input; $filters = isset($filter) ? $filter : C('DEFAULT_FILTER'); if ($filters) { if (is_string($filters)) { $filters = explode(',', $filters); } foreach ($filters as $filter) { $data = array_map_recursive($filter, $data); } } } elseif (isset($input[$name])) { $data = $input[$name]; $filters = isset($filter) ? $filter : C('DEFAULT_FILTER'); if ($filter) { if (is_string($filters)) { if (0 === strpos($filters, '/')) { if (1 !== preg_match($filters, (string)$data)) { return isset($default) ? $default : null; } } else { $filters = explode(',', $filters); } } elseif (is_int($filters)) { $filters = array($filters); } if (is_array($filters)) { foreach ($filters as $filter) { if (function_exists($filter)) { $data = is_array($data) ? array_map_recursive($filter,$data) : $filter($data); } else { $data = filter_var($data, is_int($filter)) ? $filter : filter_id($filter); if (false === $data) { return isset($default) ? $default : null; } } } } } if (!empty($type)) { switch (strtolower($type)) { case 'a': $data = (array)$data; break; case 'd': $data = (int)$data; break; case 'f': $data = (float)$data; break; case 'b': $data = (boolean)$data; break; case 's': default: $data = (string)$data; } } } else { $data = isset($default) ? $default : null; } is_array($data) && array_walk_recursive($data, 'think_filter');
return $data; }
function array_map_recursive($filter, $data) { $result = array(); foreach ($data as $key => $val) { $result[$key] = is_array($val) ? array_map_recursive($filter, $val) : call_user_func($filter, $val); }
return $result; }
|
L方法
L方法用于启用多语言的情况下,设置和获取当前的语言定义。
L(‘语言变量’,[‘语言值’])
- 除了使用语言包定义语言变量之外,我们可以用L方法动态设置语言变量,例如:
1
| $langVar = L('LANG_VAR');
|
源码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35
|
function L($name = null, $value = null) { static $_lang = array(); if (empty($name)) return $_lang; if (is_string($name)) { $name = strtoupper($name); if (is_null($value)) { return isset($_lang[$name]) ? $_lang[$name] : $name; } elseif (is_array($value)) { $replace = array_keys($value); foreach ($replace as &$v) { $v = '{$'.$v.'}'; } return str_replace($replace, $value, isset($_lang[$name]) ? $_lang[$name] : $name); } $_lang[$name] = $value;
return null; } if (is_array($name)) $_lang = array_merge($_lang, array_change_key_case($name, CASE_UPPER)); return null; }
|
M方法
M方法用于实例化一个基础模型类,和D方法的区别在于:
- 不需要自定义模型类,减少IO加载,性能较好;
- 实例化后只能调用基础模型类(默认是Model类)中的方法;
- 可以在实例化的时候指定表前缀、数据库和数据库的连接信息;
M(‘[基础模型名:]模型名’,’数据表前缀’,’数据库连接信息’)
1 2
| $User = M('User'); $User = M('db2.User','think_');
|
- 第三个连接信息参数可以使用DSN配置或者数组配置,甚至可以支持配置参数。
例如,在项目配置文件中配置了:
1
| 'DB_CONFIG'=>'mysql://user_a:1234@localhost:3306/think';
|
1
| $User = M('User','think_','DB_CONFIG');
|
基础模型类和数据库可以一起使用,例如:
1
| $User = M('CommonModel:db2.User','think_');
|
源码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
|
function M($name = '', $tablePrefix = '', $connection = '') { static $_model = array(); if (strpos($name, ':')) { list($class, $name) = explode(':', $name); } else { $class = 'Think\\Model'; } $guid = (is_array($connection) ? implode('', $connection) : $connection) . $tablePrefix . $name . '_' . $class; if (!isset($_model[$guid])) $_model[$guid] = new $class($name, $tablePrefix, $connection);
return $_model[$guid]; }
|
N方法
用于核心的查询、缓存统计的计数和统计。也可用于其他计数引用。
格式:N(‘计数位置’[,’步进值’])
1 2 3
| N('read',1);
$count = N('read');
|
源码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31
|
function N($key, $step = 0, $save = false) { static $_num = array(); if (!isset($_num[$key])) { $_num[$key] = (false !== $save) ? S('N_' . $key) :0; } if (empty($step)) { return $_num[$key]; } else { $_num[$key] = $_num[$key] + (int)$step; } if (false !== $save) { S('N_' . $key, $_num[$key], $save); }
return null; }
|
R方法
R方法用于调用某个控制器的操作方法,是A方法的进一步增强和补充。
R(‘[项目://][分组/]模块/操作’,’参数’,’控制器层名称’)
1
| $data = R('User/detail',array('5'));
|
源码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
|
function R($url, $vars = array(), $layer = '') { $info = pathinfo($url); $action = $info['basename']; $module = $info['dirname']; $class = A($module, $layer); if ($class) { if (is_string($vars)) { parse_str($vars, $vars); }
return call_user_func_array(array(&$class, $action.C('ACTION_SUFFIX')), $vars); } else { return false; } }
|
S方法
S方法还支持对当前的缓存方式传入缓存参数,例如:
1
| S('data',$Data,3600,'File',array('length'=>10,'temp'=>RUNTIME_PATH.'temp/'));
|
经测试,这样使用 只有前三个参数有效,后面的均无效
1
| { 'File',array('length'=>10,'temp'=>RUNTIME_PATH.'temp/')}
|
最终这么用:
1
| S('data1',$list,array('prefix'=>aaa','expire'=>'3600','temp'=>RUNTIME_PATH.'temp/1236'));
|
获取的时候:
1
| $sdata = S('data1','',array('prefix'=>'aaa','temp'=>RUNTIME_PATH.'temp/1236'));
|
源码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37
|
function S($name, $value = '', $options = null) { static $cache = ''; if (is_array($options)) { $type = isset($options['type']) ? $options['type'] : ''; $cache = Think\Cache::getInstance($type, $options); } elseif (is_array($name)) { $type = isset($name['type']) ? $name['type'] : ''; $cache = Think\Cache::getInstance($type, $name); } elseif (empty($cache)) { $cache = Think\Cache::getInstance(); }
if ('' === $value) { return $cache->get($name); } elseif (is_null($value)) { return $cache->rm($name); } else { if (is_array($options)) { $expire = isset($options['expire']) ? $options['expire'] : null; } else { $expire = is_numeric($options) ? $options : null; }
return $cache->set($name, $value, $expire); } }
|
T方法
为了更方便的输出模板文件,新版封装了一个T函数用于生成模板文件名。
T([资源://][模块@][主题/][控制器/]操作,[视图分层])
T函数的返回值是一个完整的模板文件名,可以直接用于display和fetch方法进行渲染输出。
1 2 3 4 5 6 7 8 9 10 11 12 13 14
| T('Public/menu');
T('blue/Public/menu');
T('Public/menu','Tpl');
T('Public/menu');
T('Public/menu');
T('Admin@Public/menu');
T('Extend://Admin@Public/menu');
|
在display方法中直接使用T函数:
1 2
| $this->display(T('Admin@Public/menu'));
|
源码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56
|
function T($template = '', $layer = '') { if (false === strpos($template, '://')) { $template = 'http://' . str_replace(':', '/', $template); } $info = parse_url($template); $file = $info['host'] . (isset($info['path']) ? $info['path'] : ''); $module = isset($info['user']) ? $info['user'] . '/' : MODULE_NAME . '/'; $extend = $info['scheme']; $layer = $layer ? $layer : C('DEFAULT_V_LAYER');
$auto = C('AUTOLOAD_NAMESPACE'); if ($auto && isset($auto[$extend])) { $baseUrl = $auto[$extend] . $module . $layer . '/'; } elseif (C('VIEW_PATH')) { $baseUrl = C('VIEW_PATH'); } elseif (defined('TMPL_PATH')) { $baseUrl = TMPL_PATH . $module; } else { $baseUrl = APP_PATH . $module . $layer . '/'; }
$theme = substr_count($file, '/') < 2 ? C('DEFAULT_THEME') : '';
$depr = C('TMPL_FILE_DEPR'); if ('' == $file) { $file = CONTROLLER_NAME . $depr . ACTION_NAME; } elseif (false === strpos($file, '/')) { $file = CONTROLLER_NAME . $depr . $file; } elseif ('/' != $depr) { $file = substr_count($file, '/') > 1 ? substr_replace($file, $depr, strpos($file, '/'), 1) : str_replace('/', $depr, $file); }
return $baseUrl . ($theme ? $theme . '/' : '') . $file . C('TMPL_TEMPLATE_SUFFIX'); }
|
U方法
U方法用于完成对URL地址的组装,特点在于可以自动根据当前的URL模式和设置生成对应的URL地址,格式为:
U(‘地址’,’参数’,’伪静态’,’是否跳转’,’显示域名’);
1 2 3 4
| U('User/add') U('Home/User/add') U('add') U('add')
|
源码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158
|
function U($url = '', $vars = '', $suffix = true, $domain = false) { $info = parse_url($url); $url = !empty($info['path'])?$info['path']:ACTION_NAME; if (isset($info['fragment'])) { $anchor = $info['fragment']; if (false !== strpos($anchor,'?')) { list($anchor,$info['query']) = explode('?',$anchor,2); } if (false !== strpos($anchor,'@')) { list($anchor,$host) = explode('@',$anchor, 2); } } elseif (false !== strpos($url,'@')) { list($url,$host) = explode('@',$info['path'], 2); } if (isset($host)) { $domain = $host.(strpos($host,'.')?'':strstr($_SERVER['HTTP_HOST'],'.')); } elseif ($domain === true){ $domain = $_SERVER['HTTP_HOST']; if (C('APP_SUB_DOMAIN_DEPLOY')) { $domain = $domain == 'localhost' ? 'localhost' : 'www' . strstr($_SERVER['HTTP_HOST'],'.'); foreach (C('APP_SUB_DOMAIN_RULES') as $key => $rule) { $rule = is_array($rule) ? $rule[0] : $rule; if(false === strpos($key,'*') && 0=== strpos($url,$rule)) { $domain = $key.strstr($domain,'.'); $url = substr_replace($url,'',0,strlen($rule)); break; } } } }
if (is_string($vars)) { parse_str($vars,$vars); } elseif (!is_array($vars)) { $vars = array(); } if (isset($info['query'])) { parse_str($info['query'],$params); $vars = array_merge($params,$vars); }
$depr = C('URL_PATHINFO_DEPR'); $urlCase = C('URL_CASE_INSENSITIVE'); if ($url) { if (0=== strpos($url,'/')) { $route = true; $url = substr($url,1); if('/' != $depr) { $url = str_replace('/',$depr,$url); } } else { if('/' != $depr) { $url = str_replace('/',$depr,$url); } $url = trim($url,$depr); $path = explode($depr,$url); $var = array(); $varModule = C('VAR_MODULE'); $varController = C('VAR_CONTROLLER'); $varAction = C('VAR_ACTION'); $var[$varAction] = !empty($path) ? array_pop($path) : ACTION_NAME; $var[$varController] = !empty($path) ? array_pop($path) : CONTROLLER_NAME; if($maps = C('URL_ACTION_MAP')) { if (isset($maps[strtolower($var[$varController])])) { $maps = $maps[strtolower($var[$varController])]; if ($action = array_search(strtolower($var[$varAction]), $maps)) { $var[$varAction] = $action; } } } if ($maps = C('URL_CONTROLLER_MAP')) { if ($controller = array_search(strtolower($var[$varController]),$maps)) { $var[$varController] = $controller; } } if ($urlCase) { $var[$varController] = parse_name($var[$varController]); } $module = '';
if (!empty($path)) { $var[$varModule] = implode($depr,$path); } else { if (C('MULTI_MODULE')) { if(MODULE_NAME != C('DEFAULT_MODULE') || !C('MODULE_ALLOW_LIST')) { $var[$varModule] = MODULE_NAME; } } } if ($maps = C('URL_MODULE_MAP')) { if ($_module = array_search(strtolower($var[$varModule]), $maps)) { $var[$varModule] = $_module; } } if (isset($var[$varModule])) { $module = $var[$varModule]; unset($var[$varModule]); } } }
if (C('URL_MODEL') == 0) { $url = __APP__.'?'.C('VAR_MODULE')."={$module}&".http_build_query(array_reverse($var)); if ($urlCase){ $url = strtolower($url); } if(!empty($vars)) { $vars = http_build_query($vars); $url .= '&'.$vars; } } else { if(isset($route)) { $url = __APP__.'/'.rtrim($url,$depr); }else{ $module = (defined('BIND_MODULE') && BIND_MODULE==$module )? '' : $module; $url = __APP__.'/'.($module?$module.MODULE_PATHINFO_DEPR:'').implode($depr,array_reverse($var)); } if($urlCase){ $url = strtolower($url); } if(!empty($vars)) { foreach ($vars as $var => $val){ if('' !== trim($val)) $url .= $depr . $var . $depr . urlencode($val); } } if($suffix) { $suffix = $suffix===true?C('URL_HTML_SUFFIX'):$suffix; if($pos = strpos($suffix, '|')){ $suffix = substr($suffix, 0, $pos); } if($suffix && '/' != substr($url,-1)){ $url .= '.'.ltrim($suffix,'.'); } } } if (isset($anchor)) { $url .= '#'.$anchor; } if ($domain) { $url = (is_ssl() ? 'https://' : 'http://') . $domain.$url; }
return $url; }
|
其他
M和D方法的区别:
- 都用于实例化一个模型类,M方法用于高效实例化一个基础模型类,D方法用于实例化一个用于自定义模型类。
- 不需要自定义模型类,减少IO加载,性能较好;
- 实例化后只能调用基础模型类(默认是Model类)中的方法;
- 可以在实例化的时候指定表前缀、数据库和数据库的连接信息;
使用M方法
如果是如下情况,请考虑使用 M方法:
- 对数据表进行简单的 CURD 操作而无复杂的业务逻辑时
- 只有个别的表有较为复杂的业务逻辑时,将 M方法 与实例化 CommonModel 类进行结合使用
M方法 甚至可以简单看着就是对参数表名对应的数据表的操作:
使用D方法
如果是如下情况,请考虑使用 D方法:
- 需要使用 ThinkPHP 模型中一些高级功能如自动验证功能(create()方法中实现)、关联模型等
- 业务逻辑比较复杂,且涉及的表众多
- 将业务逻辑定义在了自定义的模型类里面(Lib/Model目录下),而想在操作中实现这些业务逻辑