亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > PHP > 正文

CI框架源碼閱讀筆記7 配置管理組件 Config.php

2020-03-22 20:28:35
字體:
來源:轉載
供稿:網友
  • 一個靈活可控的html' target='_blank'>應用程序中,必然會存在大量的可控參數(我們稱為配置),例如在CI的主配置文件中(這里指Application/Config/Config.php文件),有如下多項配置:

    $config['base_url']   = 'http://test.xq.com';$config['index_page'] = '';$config['uri_protocol']     = 'AUTO';$config['url_suffix'] = '.html';$config['language']  = 'english';$config['charset'] = 'UTF-8';$config['enable_hooks'] = FALSE;…………………………

    不僅如此,CI還允許你將配置參數放到主配置文件之外。例如,你可以定義自己的配置文件為Config_app.php, 然后在你的應用程序控制器中這樣加載你的配置文件:

    $this->config->load('config_app');

    如此紛繁多樣的配置項和配置文件,CI是如何進行管理的?這便是我們今天要跟蹤的內容:CI的配置管理組件-Config.php.

    先看該組件的類圖:

    其中:

    _config_paths:要搜索的配置文件的路徑,這里指APPPATH目錄,你的配置文件也應該位于APPPATH下。

    Config: 這個數組用于存放所有的配置項的item

    Is_loaded: 存放所有的已經加載的配置文件列表。

    _construct: 組件的構造函數,主要是配置base_url

    _assign_to_config: 允許index.php中的配置項覆蓋主配置文件中的設置

    _uri_string,site_url,base_url,system_url: URI, 項目路徑等相關處理。

    load: 加載配置文件。

    item:獲取配置項

    slash_item:同item,不同的是,在最后加了”/”分隔符,一般只有site_url,base_url等會需要slash_item

    下面我們去剖析各個方法的具體實現:

    1.  組件初始化 _construct

    之前我們在分析Common.php全局函數的時候提到過,在Config組件實例化之前,所有的組配置文件的獲取都是由get_config()函數來代理的。在Config組件實例化時,要將所有的配置存放到自己的私有變量$config中,便于之后的訪問和處理:

    $this->config =& get_config();

    由于我們應用程序很多時候需要獲取base_url的值,而這個值并不是必填項(config中base_url可以設置為空),但我們又不希望獲取到的base_url的值為空。因此,CI在Config組件初始化的時候,對base_url做了一定的處理。這主要出現在Config.php中base_url設置為空的情況:

    (1). 如果設置了$_SERVER[‘HTTP_HOST’],則base_url被設置為Protocal(http或者https) + $_SERVER['HTTP_HOST'] + SCIRPT_PATH的形式:

    $base_url = isset($_SERVER['HTTPS']) && strtolower($_SERVER['HTTPS']) !== 'off' ? 'https' : 'http';$base_url .= '://'. $_SERVER['HTTP_HOST'];$base_url .= str_replace(basename($_SERVER['SCRIPT_NAME']), '', $_SERVER['SCRIPT_NAME']);

    (2). 否者,直接被設置為http://localhost/:

    $base_url = 'http://localhost/';

    (3). 同時將base_url配置項映射到配置數組中,方便之后的訪問(set_item方法我們稍后會將,這里只需要知道,它是添加到配置項且會覆蓋舊值):

    $this->set_item('base_url', $base_url);

    之后我們會看到,base_url這個配置項對于很多組件都是必須的,因此,CI花費一定的精力來保證base_url的正確性,也是可以理解的。

    2.  加載配置文件 load

    這是Config組件中較核心的方法之一,該函數的簽名:

    function load($file = '', $use_sections = FALSE, $fail_gracefully = FALSE)

    所有的參數都是可選參數。

    我們這里簡單解釋一下各形參的含義:

      $file 需要加載的配置文件,可以包含后綴名也不可以不包含,如果未指定該參數,則默認加載Config.php文件

      $user_sections: 是否為加載的配置文件使用獨立的section,這么說可能還是不明白,試想,如果你定義了自己的配置文件,而你的配置文件中的配置項可能與Config.php文件中的配置項沖突,通過指定$section為true可以防止配置項的覆蓋。

      $fail_gracefully: 要load的配置文件不存在時的處理。Gracefully意為優雅的,如果該參數設置為true,則在文件不存在時只會返回false,而不會顯示錯誤。

    下面看該方法的具體實現:

    (1). 配置文件名預處理:

    $file = ($file == '') ? 'config' : str_replace('.php', '', $file);

    這個$file最后只包含文件名,而不包含擴展名。如果該參數為空,則默認加載Config.php配置文件。這同時也說明,我們加載自己的配置文件時:

    $this->config->load("");與

    $this->config->load("config")效果是一樣的,而:

    $this->config->load("config_app")與

    $this->config->load("config_app.php")的效果也是一樣的。

    如果啟用了$use_sections,這個$file會作為config的主鍵。

    (2). 查找和加載配置文件。

    在跟蹤實現之前,先解釋幾個查找和加載過程中比較重要的參數:

    $found 這個參數實際上是個flag,用于標識配置文件是否查找到,一旦查找到配置文件,則停止任何搜索。$loaded 同$found參數類似,這個$loaded也是一個flag,用于標識請求的配置文件是否被加載。一般情況下,被加載的配置文件會被CI_Config:: is_loaded變量追蹤$_config_path 要查找的配置路徑,這個變量由于是寫死在Config組件中的,且沒有提供添加或者更改的接口。因此我們可以認為_config_path就是APPPATH.也就是,配置文件的load一定是在APPPATH目錄下查找的。$check_locations 這個參數是要查找的位置(具體文件)。同樣,如果定了ENVIRONMENT且存在相應ENVIRONMENT下的配置文件,優先加載該文件。

    (3).具體的查找過程是一個雙重的foreach循環:

    /*  對于config_paths中的路徑循環查找 */foreach ($this->_config_paths as $path){	  /* 對每個location查找,也就是分別對ENVIRONMENT/config/ 和 config/ 目錄查找  */  foreach ($check_locations as $location)  {	/* 實際的配置文件名 */	$file_path = $path.'config/'.$location.'.php';	
    /* 如果已經加載,則跳至最外層循環,事實上,由于_config_paths的設定,會跳出整個循環 */ if (in_array($file_path, $this->is_loaded, TRUE)) {   $loaded = TRUE;   continue 2; } /* 若文件存在,跳出當前循環 */ if (file_exists($file_path)) {   $found = TRUE;   break; }  }  /* 如果沒有找到配置文件,繼續下一次循環。同樣,由于_config_path的設定,會跳出整個循環 */  if ($found === FALSE)  { continue;  }}

    (4).引入配置文件

    到這里,如果配置文件不存在,則$found和$loaded都為false,CI會根據fail_gracefully參數決定文件不存在的處理方式;如果文件存在,則需要對配置文件的格式檢查:

    /* 引入配置文件 */include($file_path);/* 配置文件的格式檢查,這同時也說明,配置文件中最起碼應該包含$config數組 */if ( ! isset($config) OR ! is_array($config)){  if ($fail_gracefully === TRUE)  {	return FALSE;  }  show_error('Your '.$file_path.' file does not appear to contain a valid configuration array.');}

    (5).對use_sections參數的處理

    前面說過,use_secitons參數如果為true,則CI_Config會對該配置文件啟用獨立的key存儲。例如,我們在controller中這樣加載配置文件:

    $this->config->load("config_app",true);

    則config數組是這樣的格式:

    [config] => Array(    [base_url] => http://test.xq.com    [index_page] =>    [uri_protocol] => AUTO    [url_suffix] => .html    [proxy_ips] =>    [web_akey] => yyyyyyyyyyyy    [config_app] => Array        (            [web_akey] => xxxxxxx            [web_skey] => xxxxxxxxxxxxxxxxxxx            [web_callback_url] => http://test.xq.com/            [sess_pre] => WEB_APP            [cart_min] => 1            [cart_max] => 999        ))

    相反,如果我們不指定use_sections,則數組是這樣存儲的:

    [config] => Array(    [base_url] => http://test.xq.com    [index_page] =>    [uri_protocol] => AUTO    [url_suffix] => .html    [web_akey] => xxxxxxx    [web_skey] => xxxxxxxxxxxxxxxxxxx    [web_callback_url] => http://test.xq.com/    [sess_pre] => WEB_APP    [cart_min] => 1    [cart_max] => 999)

    這也意味著,在不啟用user_secitons的情況下,如果你的配置文件中有與主配置文件Config.php相同的鍵,則會覆蓋主配置文件中的項:

    /* 啟用單獨的key存放加載的config */if ($use_sections === TRUE){  if (isset($this->config[$file]))  {	$this->config[$file] = array_merge($this->config[$file], $config);  }  else  {	$this->config[$file] = $config;  }}else{  /* 執行merge,更改CI_Config::config */  $this->config = array_merge($this->config, $config);}

    (6).錯誤處理

    雙層循環完成后,如果loaded為false,也就是未成功加載任何配置,則根據fail_gracefully做相應的錯誤處理:

    /* 未成功加載任何配置 */if ($loaded === FALSE){  if ($fail_gracefully === TRUE)  {	return FALSE;  }  show_error('The configuration file '.$file.'.php does not exist.');}
    3.  獲取配置項item,slash_item

    item方法用于在配置中獲取特定的配置項,改方法的簽名:

    function item($item, $index = '')

    注意,如果你在load配置文件的時候啟用了use-sections,則在使用item()獲取配置項的時候需要指定第二個參數,也就是加載的配置文件的文件名(不包含后綴)。為了更清楚這一點,我們假設現在Config/目錄下有配個配置文件:config.php和config_app.php,這兩個配置文件中含有一個相同的鍵web_akey, 在config.php中,該配置為:

    $config['web_akey']  = 'yyyyyyyyyyyy';

    而config_app.php中,該配置為:

    $config['web_akey'] = 'xxxxxxx';

    現在,通過use-sections的方法加載config_app配置文件(config.php會在Config組件初始化的時候被加載):

    $this->config->load("config_app",true);

    然后在控制器中獲取web_akey配置項:

    echo "config_app:web_akey => ",$this->config->item("web_akey","config_app"),"<br/>";echo "config    :web_akey => ",$this->config->item("web_akey");

    實際的獲取結果:

    config_app:web_akey => xxxxxxxconfig :web_akey => yyyyyyyyyyyy

    了解原理之后,該方法的實現就比較簡單了:

    function item($item, $index = ''){	  /* 沒有設置use_sections的情況,直接在config中尋找配置項 */  if ($index == '')  {	if ( ! isset($this->config[$item]))	{	  return FALSE;	}	$pref = $this->config[$item];  }  else  {	if ( ! isset($this->config[$index]))	{	  return FALSE;	}	if ( ! isset($this->config[$index][$item]))	{	  return FALSE;	}	$pref = $this->config[$index][$item];  }  /* 統一的return出口 */  return $pref;}

    slash_item實際上與item()方法類似,但他不會去用戶的配置中尋找,并且,他返回的是主配置文件中的配置項,并在配置項最后添加反斜杠.這個方法,通常用于base_url和index_page這兩個配置項的處理:

    該方法的實現源碼:

    function slash_item($item){	  /* 不存在配置項 */  if ( ! isset($this->config[$item]))  {	return FALSE;  }  /* 配置項為空 */  if( trim($this->config[$item]) == '')  {	return '';  }	  /* 去除最后的多余的"/",并在結尾添加一個"/" */  return rtrim($this->config[$item], '/').'/';}
    4.  獲取站點site_url, base_url,system_url

    這里先澄清這幾個含義的區別:

    echo "site_url  : ",$this->config->site_url("index/rain"),"</br>";echo "base_url  : ",$this->config->base_url("index/rain"),"<br/>";echo "system_url: ",$this->config->system_url();

    的結果分別是:

    site_url : http://test.xq.com/index/rain.htmlbase_url : http://test.xq.com/index/rainsystem_url: http://test.xq.com/system/

    可以看出,site_url是添加了suffix(在Config/config.php中配置)后的url地址(呵呵,如果你的uri中有query string,則Ci總是在最后添加suffix:http://test.xq.com/index/rain?w=ss.html 是不是很奇怪.)

    base_url則是沒有添加suffix的url地址。

    而system_url這個東西很奇怪,是獲取系統的url路徑。但實際上,由于system路徑并沒有直接執行的腳本,所以這個方法的實際用途是什么,暫時不知。有知道的童鞋麻煩告知。

    具體的方法實現,這里不贅述了。直接貼出源碼:

    function site_url($uri = ''){	/* 沒有設置uri,使用base_url + index_page */	if ($uri == '')	{		return $this->slash_item('base_url').$this->item('index_page');	}		/* enable_query_strings未啟用,可以添加suffix后綴 */	if ($this->item('enable_query_strings') == FALSE)	{		$suffix = ($this->item('url_suffix') == FALSE) ? '' : $this->item('url_suffix');		return $this->slash_item('base_url').$this->slash_item('index_page').$this->_uri_string($uri).$suffix;	}	/* 否者不添加suffix后綴 */	else	{		return $this->slash_item('base_url').$this->item('index_page').'?'.$this->_uri_string($uri);	}}/* 獲取base_url,注意與site_url的區別 */function base_url($uri = ''){	return $this->slash_item('base_url').ltrim($this->_uri_string($uri), '/');}/* 獲取system url */function system_url(){       /* 獲取系統目錄.   BASEPATH:/search/xx/phpCode/CI/system/ */    $x = explode("/", preg_replace("|/*(.+?)/*$|", "http://1", BASEPATH));    return $this->slash_item('base_url').end($x).'/';}
    5.  獲取URI String: _uri_string

    site_url和base_url都調用了_uri_string。這個函數是做什么用的呢?

    按理來說, _uri_string的功能應該由URI組件來完成,這里卻放在了Config組件中,似乎有些不妥(實際上,_uri_string是為base_url和site_url專屬服務的)。

    對于這樣的uri:

    array(    'p1' => 'param1',    'p2' => 'param2')

    如果enable_query_string為false,則_uri_string處理過后是這樣的形式:

    param1/param2

    而enable_query_string為true,則處理后的形式是這樣的:

    p1=param1&p2=param2

    這是我們常見(雖然很難看且SEO不好)的形式。改方法的實現源碼:

    protected function _uri_string($uri){		/* enable_query_strings 為false,直接implode */	if ($this->item('enable_query_strings') == FALSE)	{		if (is_array($uri))		{			$uri = implode('/', $uri);		}		$uri = trim($uri, '/');	}	/* 否者,拼接成類似param1=param1&param2=param2的形式 */	else	{		if (is_array($uri))		{			$i = 0;			$str = '';			foreach ($uri as $key => $val)			{					/* 第一個參數前面不需要加& */				$prefix = ($i == 0) ? '' : '&';				$str .= $prefix.$key.'='.$val;				$i++;			}			$uri = $str;		}	}    return $uri;}
    6.  設置配置項 set_item _assign_to_config

    與item()相反,set_item用于設置配置項。如果配置項已經存在,則會被覆蓋:

    $this->config[$item] = $value;

    _assign_to_config同set_item,該方法提供了數組的設置方式(調用set_item。我們之前在解釋CodeIgniter.php文件的時候提到過:改方法允許在index.php中設置獨立的配置項,且index.php中的配置具有更高的優先權(會覆蓋主配置文件中的配置):

    function _assign_to_config($items = array()){	if (is_array($items))	{		foreach ($items as $key => $val)		{			$this->set_item($key, $val);		}	}}

    到這里,Config組件的基本解析就算是完成了,我們再次回顧下該組件的基本功能:

    set_item和item是Config組件的基本對外接口。也就是常見的setter 和getter,_assign_to_config算是批量的setter,slash_item則是特殊處理的getterload方法是加載配置文件,如果你自定義了自己的配置文件,需要先load使得你的配置納入CI_Config的管理之下。system_url,base_url,site_url,用于獲取特定的配置項。_uri_string是CI_Config中唯一一個Protected的方法。這個方法主要是處理uri,提供給site_url和base_url使用

    最后感慨一下,一個好的Config組件,會省不少事啊。

    PHP編程

    鄭重聲明:本文版權歸原作者所有,轉載文章僅為傳播更多信息之目的,如作者信息標記有誤,請第一時間聯系我們修改或刪除,多謝。

  • 發表評論 共有條評論
    用戶名: 密碼:
    驗證碼: 匿名發表
    亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
    91九色视频在线| 日本精品一区二区三区在线播放视频| 国产精品爽爽爽爽爽爽在线观看| 成人精品一区二区三区电影免费| 亚洲一区二区久久久| 69av在线视频| 国产视频精品xxxx| 欧美极品在线播放| 97人洗澡人人免费公开视频碰碰碰| 亚洲精品国产精品自产a区红杏吧| 国产美女精彩久久| 国产97在线播放| 97人人做人人爱| 国产一区二区美女视频| 亚洲色无码播放| 亚洲人成网7777777国产| 欧美在线亚洲一区| 日韩精品在线播放| 久久影视电视剧免费网站清宫辞电视| 久久影视电视剧凤归四时歌| 国产福利精品av综合导导航| 成人激情视频在线观看| 亚州av一区二区| 欧美孕妇毛茸茸xxxx| 亚洲免费福利视频| 国产v综合ⅴ日韩v欧美大片| 国产午夜精品久久久| 国产精品手机播放| 亚洲福利视频专区| 亚洲性视频网站| 国外成人在线直播| 国产亚洲精品美女| 在线观看日韩www视频免费| 日韩av影片在线观看| 欧美激情奇米色| 久久人人爽亚洲精品天堂| 狠狠爱在线视频一区| 国模视频一区二区| 久久影院在线观看| 26uuu另类亚洲欧美日本老年| 日韩免费在线观看视频| 成人精品久久久| 亚洲国产精品va在线看黑人动漫| 精品国产一区二区三区久久久狼| 欧美日韩福利在线观看| 亚洲激情在线观看视频免费| 欧美孕妇与黑人孕交| 自拍偷拍亚洲一区| 国产精品 欧美在线| 国产成人拍精品视频午夜网站| 国语自产在线不卡| 精品高清一区二区三区| 亚洲免费视频一区二区| 精品美女永久免费视频| 久久精品视频在线观看| 亚洲黄页网在线观看| 国产精品欧美日韩一区二区| 午夜精品一区二区三区av| 欧美肥老妇视频| 欧美中在线观看| 亚洲国产欧美一区| 亚洲伊人一本大道中文字幕| 亚洲高清久久久久久| 亚洲欧美国产精品va在线观看| 久热99视频在线观看| 国产精品免费视频xxxx| 亚洲理论在线a中文字幕| 久久av红桃一区二区小说| 日韩在线观看免费网站| 韩国三级日本三级少妇99| 亚洲影视中文字幕| 91在线免费视频| 成人亚洲综合色就1024| 欧美激情a在线| 亚洲一区二区福利| 97国产在线观看| 国产视频亚洲视频| 精品美女国产在线| 26uuu另类亚洲欧美日本一| 亚洲国产精品系列| 久久国产精品久久国产精品| 国产午夜精品理论片a级探花| 91人成网站www| 欧洲一区二区视频| 538国产精品视频一区二区| 久久综合免费视频| 国产成人拍精品视频午夜网站| 欧美肥婆姓交大片| 日韩激情av在线免费观看| 久久人人爽人人| 中文字幕在线看视频国产欧美在线看完整| 国产精品白嫩初高中害羞小美女| 97碰在线观看| 久久久久中文字幕| 国产精品pans私拍| 亚洲小视频在线观看| 欧美日韩亚洲国产一区| 91亚洲一区精品| 国产欧美一区二区三区在线| 国产精品入口夜色视频大尺度| 精品久久久久久久久久久久| 亚洲午夜精品视频| 亚洲第一黄色网| 国产精品视频永久免费播放| 欧美区二区三区| 国产主播欧美精品| 高清一区二区三区四区五区| 亚洲精品www久久久久久广东| 日韩暖暖在线视频| 欧美日韩精品二区| 国产精品福利在线观看| 国产成一区二区| 日韩中文字幕视频在线观看| 精品国产依人香蕉在线精品| 欧美日韩国产精品专区| 美女少妇精品视频| 国内揄拍国内精品少妇国语| 久久91亚洲精品中文字幕奶水| 欧美激情极品视频| 国产精品视频久久久| 亚洲欧美中文日韩v在线观看| 欧美精品在线观看| 国产精品久久久久久久久| 97在线看免费观看视频在线观看| 92看片淫黄大片看国产片| 久久成人人人人精品欧| 69久久夜色精品国产69乱青草| 国产欧美一区二区三区久久| 国产精品福利在线| 亚洲一二三在线| 亚洲欧美日韩一区在线| www.亚洲成人| 久久在线观看视频| 欧美激情综合亚洲一二区| 日韩中文字幕在线精品| 日韩在线免费视频| 欧美巨大黑人极品精男| 国产精品∨欧美精品v日韩精品| 午夜精品久久久久久久白皮肤| 欧美电影免费观看| 国产成人福利网站| 亚洲第一页自拍| 欲色天天网综合久久| 亚洲视频电影图片偷拍一区| 日韩精品极品视频| 日韩毛片在线观看| 美女扒开尿口让男人操亚洲视频网站| 欧美电影免费观看电视剧大全| 成人激情视频在线观看| 一区二区三区国产视频| 91免费国产网站| 国产精品一区专区欧美日韩| 最近2019年中文视频免费在线观看| 久久6免费高清热精品| 亚洲精品不卡在线| 国产99久久精品一区二区 夜夜躁日日躁| 亚洲视频欧洲视频| 亚洲日本中文字幕| 亚洲成人久久久久| 国产999精品| 中国人与牲禽动交精品| 欧美激情国产日韩精品一区18| 欧美老女人www| 久久国产精彩视频|