thinkphp3.2.0 setInc方法 源碼全面解析php執行個體

來源:互聯網
上載者:User
下面小編就為大家分享一篇thinkphp3.2.0 setInc方法 源碼全面解析,具有很好的參考價值,希望對大家有所協助。一起跟隨小編過來看看吧

我們先來看一下setInc的官方樣本:

需要一個欄位和一個自增的值(預設為1)

我們通過下面這個例子來一步步分析他的底層是怎麼實現的:

<?phpnamespace Home\Controller;use Think\Controller;class TestController extends Controller {  public function test() {    $tb_test = M('test');    $tb_test->where(['id'=>1])->setInc('test_number',2); //每次添加2    dump($tb_test->getLastSql());    //string(67) "UPDATE `tb_test` SET `test_number`=test_number+2 WHERE ( `id` = 1 )"  }}

第一步肯定是要找到setInc方法的源碼:

這裡我用到了phpstrom全域搜尋的方法,找到了setInc是在proj\ThinkPHP\Library\Think\Model.class.php下

/**   * 欄位值增長   * @access public   * @param string $field 欄位名   * @param integer $step 增長值   * @return boolean   */  public function setInc($field,$step=1) {    return $this->setField($field,array('exp',$field.'+'.$step));  }

可以看到這裡用到了setField這個方法,然後用exp自訂運算式設定 $field = $field + $step 到這裡,我們稍微瞭解了一點原理。

可是問題又來了setField又是怎麼實現的呢?在同個檔案下,找到setField方法:

/**   * 設定記錄的某個欄位值   * 支援使用資料庫欄位和方法   * @access public   * @param string|array $field 欄位名   * @param string $value 欄位值   * @return boolean   */  public function setField($field,$value='') {    if(is_array($field)) {      $data      =  $field;    }else{      $data[$field]  =  $value;    }    return $this->save($data);  }

這裡我們看到了常用到的save方法,這裡的 $data[$field] = $value; 其實就是 $data['test_number'] = array("exp","test_number+2")

接著來看最常用的save方法:

/**   * 儲存資料   * @access public   * @param mixed $data 資料   * @param array $options 運算式   * @return boolean   */  public function save($data='',$options=array()) {    if(empty($data)) {      // 沒有傳遞資料,擷取當前資料對象的值      if(!empty($this->data)) {        $data      =  $this->data;        // 重設資料        $this->data   =  array();      }else{        $this->error  =  L('_DATA_TYPE_INVALID_');        return false;      }    }    // 資料處理    $data    =  $this->_facade($data);    // 分析運算式    $options  =  $this->_parseOptions($options);    $pk     =  $this->getPk();    if(!isset($options['where']) ) {      // 如果存在主鍵資料 則自動作為更新條件      if(isset($data[$pk])) {        $where[$pk]     =  $data[$pk];        $options['where']  =  $where;        unset($data[$pk]);      }else{        // 如果沒有任何更新條件則不執行        $this->error    =  L('_OPERATION_WRONG_');        return false;      }    }    if(is_array($options['where']) && isset($options['where'][$pk])){      $pkValue  =  $options['where'][$pk];    }        if(false === $this->_before_update($data,$options)) {      return false;    }        $result   =  $this->db->update($data,$options);    if(false !== $result) {      if(isset($pkValue)) $data[$pk]  = $pkValue;      $this->_after_update($data,$options);    }    return $result;  }

最主要是的$options = $this->_parseOptions($options);和$result = $this->db->update($data,$options); 前者把參數轉換成用於拼接sql的字串數組,後者調用了proj\tptest\ThinkPHP\Library\Think\Db.class.php下的update方法:

/**   * 更新記錄   * @access public   * @param mixed $data 資料   * @param array $options 運算式   * @return false | integer   */  public function update($data,$options) {    $this->model =  $options['model'];    $sql  = 'UPDATE '      .$this->parseTable($options['table'])      .$this->parseSet($data)      .$this->parseWhere(!empty($options['where'])?$options['where']:'')      .$this->parseOrder(!empty($options['order'])?$options['order']:'')      .$this->parseLimit(!empty($options['limit'])?$options['limit']:'')      .$this->parseLock(isset($options['lock'])?$options['lock']:false)      .$this->parseComment(!empty($options['comment'])?$options['comment']:'');    return $this->execute($sql,$this->parseBind(!empty($options['bind'])?$options['bind']:array()));  }

最後其實就是用到了proj\ThinkPHP\Library\Think\Db\Driver\Mysql.class.php這個驅動類的execute方法。

/**   * 執行語句   * @access public   * @param string $str sql指令   * @return integer|false   */  public function execute($str) {    $this->initConnect(true);    if ( !$this->_linkID ) return false;    $this->queryStr = $str;    //釋放前次的查詢結果    if ( $this->queryID ) {  $this->free();  }    N('db_write',1);    // 記錄開始執行時間    G('queryStartTime');    $result =  mysql_query($str, $this->_linkID) ;    $this->debug();    if ( false === $result) {      $this->error();      return false;    } else {      $this->numRows = mysql_affected_rows($this->_linkID);      $this->lastInsID = mysql_insert_id($this->_linkID);      return $this->numRows;    }  }

最後用最底層的mysql_query執行SQL語句。

到此為止,setInc的源碼已經大致過了一遍了。想必大家對setInc如何執行也更瞭解了一點。

以上這篇thinkphp3.2.0 setInc方法 源碼全面解析就是小編分享給大家的全部內容了,希望能給大家一個參考,也希望大家多多支援php中文網。

相關推薦:

PHP 使用二進位儲存使用者狀態的執行個體php技巧

PHP+Redis 訊息佇列 實現高並發下註冊人數統計的執行個體php執行個體

laravel ORM 只開啟created_at的幾種方法總結php執行個體

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在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.