程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> 網頁編程 >> PHP編程 >> 關於PHP編程 >> 巧妙的重載魔術方法__call(),重載魔術__call

巧妙的重載魔術方法__call(),重載魔術__call

編輯:關於PHP編程

巧妙的重載魔術方法__call(),重載魔術__call


工作半年了,感覺這半年學到的東西比大學四年學到的還要多,主要原因是心靜下來了,目標也明確了,不會去整天的和游戲糾纏在一起了。大學時候其實也意識到了玩游戲會影響自己的正常學習和工作的,但是一直控制不了自己,還是忍不住經常去玩,沒日沒夜的玩(本來就是悶騷男,還宅著玩游戲,這也是大學四年只有游戲、左右手,沒有女朋友的一個原因了)。現在工作了,每天都有任務,看到旁邊的牛人們在項目中如魚得水,就有了趕超他們的想法,於是每天都會給自己一個額外的小任務去學習新的知識,到現在工作有半年了,對以前不熟悉的linux現在也可應熟悉的使用了,對不熟悉的js也有了新的認識,可以說現在我對工作可以勝任(如果分為新手、高級新手、勝任者、精通者、專家)了,開發過活動、接口、後台,也優化完善過系統的框架,只要是產品運營提出的合理需求都可以快速的支持到位。當然還確確實實的感受到一點:程序員真是一個奇怪的群體,大多時候總是會覺得自己的點子是最好的。當然這個算是自信但有時候討論的時候你的咄咄逼人不一定是好事,所以還要多聽聽其他人的想法,不但可以發現自己的不足,還會建立良好的:”友誼“。跟大家瞎扯了這麼多這半年的一點點感受,謝謝你可以堅持看完^_^。

下面步入真題,說說如何巧妙的運用php的魔術方法,我相信這個在大多數項目中會用到。

先說明一下,這個小技巧我在項目中已經有很好的應用了,給我們項目帶來了很大的方便,在這裡先賣賣關子,您不妨繼續往下看。

在項目中,可配的配置信息一定大量存在,比如說一個游戲的機器人開放時間段、支付方式的開啟與否、商城顯示title的配置等等,這些配置信息一般有一個特點就是沒有特定的規則,而且產品運營可以隨時的給據實際情況去修改,這些信息怎麼保存呢,肯定不會每種類型都去建一張表,這樣做簡直就是費力不討好,你想下,也許一張表中就保存了一條信息,所以得想想其他的方法,雖然這些信息沒有規則,但是他們卻有一個特點就是不會有太多,而且一般情況下數組就可以保存所有需要配置的信息,因此用json字符串存儲信息是個不錯的選擇,當需要使用的時候直接取出json_decode這樣就可以直接使用了,下面看看具體怎麼巧妙的利用php的魔術方法實現的。

這裡你先要了解下php的一個魔術方法__call(),查下php官方的文檔,是這樣解釋這個函數的

public mixed __call ( string $name , array $arguments )

__call() is triggered when invoking inaccessible methods in an object context.

意思就是說當在一個對象中調用一個不可訪問的方法(沒有權限、不存在)時會觸發這個函數,函數的參數$name是調用的函名,$arguments是調用的函數參數數組。看看下面這個例子:

class Test
{
    public function __call($name, $arguments)
    {
        echo "你調用了一個不存在的方法:\r";
        echo "函數名:{$name}\r";
        echo "參數: \r";
        print_r($arguments);
    }
}

$T = new Test();
$T->setrobottime("12", "18");

這個函數會輸出下面的結果

你調用了一個不存在的方法:
函數名:setrobottime
參數: 
Array
(
    [0] => 12
    [1] => 18
)

這樣,我們就可以不去直接定義函數,而是用這個特性去做一些事情了。下面看看代碼的實現思路,主要是思路,其中有些我是假設的,就像數據庫連接,這裡不主要講這個。

 

class Config
{
    /**
     * 這裡假定下數據庫表名為
     * config.config,
     * 字段為:
     * config_key varchar(50),
     * config_value text,
     * primary key(config_key)
     *
     * 數據庫連接為$link
     * 插入方法封裝為query
     * 獲取一條信息方法封裝為getOne
     */
    /**
     * 要進行的操作
     */
    private static $keys = array(
        //'調用方法' => 'key',
        'roboottime'    => 'ROBOOTTIME',
        'dailysignin'   => 'DAILYSIGNIN',
    );

    /**
     * 設置方法
     * @param string $config_key 配置項key
     * @param string $config_value 配置型內容(一般為json格式)
     * @returne boolen true/false 插入是否成功
     */
    private function set($config_key, $config_value){
        $sql = "insert into config.config (config_key,config_value) values ('{$config_key}','{$config_value}') on duplicate key update config_value='{$config_value}'";
        return $link->query($sql);
    }

    /**
     * 獲取值的方法
     * @param $config_key 要獲取的配置的key
     * @returne string/false json字符串/失敗
     */
    private function get($config_key)
    {
        $sql = "select * from config.config where config_key='{$config_key}'";
        if($ret = $link->getOne($sql, MYSQL_ASSOC)){
            return $ret;
        }
        return false;
    }

    /**
     * 重載魔術方法
     * @param string $name 被調用的方法名
     * @param array $arguments 調用時傳遞的參數
     * @return mixed 返回結果
     */
    public function __call($name, $arguments)
    {
        $act    = substr($name, 0, 3);
        $func   = substr($name, 3);
        if(!in_array($func, self::$keys)){
            return false;
        }
        if($act == 'set')
        {
            return $this->set(self::[$func], $arguments[0]);
        }
        elseif($act == 'get')
        {
            return $this->get(self::[$func]);
        }
        return false;
    }
}

 

這樣,我們的就可以通過一張表存儲多個信息了,調用時也很方便,只需要擴展下Config::$keys數組中的信息就可以了,這樣做只是為了規范,為了可以清晰的知道哪些配置存放在了這張表中。

使用的時候可以像這樣去存儲和獲取

 

$config = new Config();

$info = array("12","20");

//設置
$config->setroboottime(json_encode($info));

//獲取
$config->getroboottime();

 

這裡再說一個要注意的點,這些配置信息一般會緩存到redis中,放在數據庫中只是為了防止redis掛掉之後從數據庫中去恢復,這裡的一般指的是那些經常去讀取的信息,為了減少和db的交互,直接放在緩存中。

  本文版權歸作者iforever([email protected])所有,未經作者本人同意禁止任何形式的轉載,轉載文章之後必須在文章頁面明顯位置給出作者和原文連接,否則保留追究法律責任的權利。

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved