ThinkPhp架構研究之二 魔術函數 __call

來源:互聯網
上載者:User

 

在看 TP架構的協助文檔的時候 看到一個很有意思的功能---欄位的動態查詢

 

官方是這麼描述的 

 

 

ThinkPHP提供了資料癿勱態查詢方法,可以簡化你癿查詢代碼,例如:<br />$User->where('name="ThinkPHP"')->find();<br />可以簡化為:<br />$User->getByName('ThinkPHP'); </p><p>$User->where('email="thinkphp@qq.com"')->find();  

 

第一感覺 是 getBy後面的內容是欄位名字 於是乎 就翻開Model類源碼去看了看

果不其然..發現了一個 魔術函數 __CALL;

__call 函數是 在用對象訪問一個該類不存在的方法時調用的函數

系統會自動把方法名和 參數都傳到 __call方法中

 

TP是這麼定義的

/**<br /> +----------------------------------------------------------<br /> * 利用__call方法實現一些特殊的Model方法 (魔術方法)<br /> +----------------------------------------------------------<br /> * @access public<br /> +----------------------------------------------------------<br /> * @param string $method 方法名稱<br /> * @param array $args 調用參數<br /> +----------------------------------------------------------<br /> * @return mixed<br /> +----------------------------------------------------------<br /> */<br /> public function __call($method,$args) {<br /> if(in_array(strtolower($method),array('field','table','where','order','limit','page','having','group','lock','distinct'),true)) {<br /> // 連貫操作的實現<br /> $this->options[strtolower($method)] = $args[0];<br /> return $this;<br /> }elseif(in_array(strtolower($method),array('count','sum','min','max','avg'),true)){<br /> // 統計查詢的實現<br /> $field = isset($args[0])?$args[0]:'*';<br /> return $this->getField(strtoupper($method).'('.$field.') AS tp_'.$method);<br /> }elseif(strtolower(substr($method,0,5))=='getby') {<br /> // 根據某個欄位擷取記錄<br /> $field = parse_name(substr($method,5));<br /> $options['where'] = $field.'=/''.$args[0].'/'';<br /> return $this->find($options);<br /> }else{<br /> throw_exception(__CLASS__.':'.$method.L('_METHOD_NOT_EXIST_'));<br /> return;<br /> }<br /> } 

 

仔細閱讀代碼

它提供了三種方式 如果都不符合直接報錯提示

 

第一種

if(in_array(strtolower($method),array('field','table','where','order','limit','page','having','group','lock','distinct'),true)) {<br /> // 連貫操作的實現<br /> $this->options[strtolower($method)] = $args[0];<br /> return $this;<br /> } 

 

應該能看懂吧...

_call把捕獲的方法及其參數傳遞進來

判斷 $method是否在允許的數組中,如果存在 將參數賦給$this->option變數

然後返回 $this對象

這是什麼意思?

 

我測試了一下<br />代碼如下<br />$Form= M("Form");<br />$data=$Form->order('id desc')->limit(6)->select(); 

 

按照 __call函數 第一種解釋就是 

$Form->order('id desc') 執行之後

$this->option變數會變成

$this->option['order']='id desc',

然後 會返回$this對象 

變成

$this->option['limit']='6',

然後會執行SQL去搜尋資料

 

 

第二種就是

 elseif(in_array(strtolower($method),array('count','sum','min','max','avg'),true)){<br /> // 統計查詢的實現<br /> $field = isset($args[0])?$args[0]:'*';<br /> return $this->getField(strtoupper($method).'('.$field.') AS tp_'.$method);<br /> } 

 

這個的意思就是 

如果符合array('count','sum','min','max','avg'),true)

會將參數作為 sql語句的搜尋欄位 ,如果沒有 預設就是 星

也就是 SELECT count(*)|sum(*)|min(*)|max(*)|avg(*) ********返回給資料

第三種就是咱們前面說的

elseif(strtolower(substr($method,0,5))=='getby') {<br /> // 根據某個欄位擷取記錄<br /> $field = parse_name(substr($method,5));<br /> $options['where'] = $field.'=/''.$args[0].'/'';<br /> return $this->find($options);<br /> } 

 

這句代碼會將 $User->getby* 中的*作為where 語句的參數來執行

 

假如說 代碼為 

$User->getByUsername("siren");

那麼 通過__call執行之後形成的語句就是

select * from table where username="siren"

 

 

 

 

就是這樣..如果有錯的地方 歡迎提出來..

 

 

 

 

 

 

 

 

 

 

 


聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.