前言:
先交代下背景,在一個項目中,有一個資料表有水準分表的需求。當時想找到一種方法,把對資料庫的操作,寫到一個模型裡,通過去換模型屬性中的table來達到代碼不變操作的資料表變化的效果。
我們都知道,模型要想關聯資料表的話,有兩中方式,第一種就是将模型名和資料表一緻。這樣模型就會預設關聯到名字對應的資料表。第二種就是定義模型的 protected $table 來指定表明。我當時就想,有沒有什麼方法,能初始化模型對象的時候将table屬性指派呢,這個值存在資料庫裡。這樣就可以動态的來控制這個模型關聯的表名了。
模型初始化
基于以上的需求,文檔裡的模型初始化引起了我的注意
tp5文檔
我感覺這是我要找的東西。緊接着我着手開始測試
首次測試
我根據手冊的寫法,在調用父類初始化後面,寫上對table的初始化,那麼現在我們來列印出來執行個體化的order模型
列印執行個體化對象
從圖中大家可以發現,table屬性的确已經修改了。然後我就沒有再做更多的測試了,因為我試過手動将table資料改為别的表名,就可以修改模型所關聯的資料表。我想這個table屬性已經有了肯定就沒問題了。
直到我在這個模型裡寫了很多方法後,我想去回來換個表名來試試寫入資料。爆炸的事情出現了。
//我使用包含table屬性的對象去查資料庫。查詢出來的結果,居然任然是原來那個模型名對應的表
$order = new \app\API\model\Order();
return $order->select();
這就很爆炸了,我已經寫好的這麼多代碼難道都不能用了?
冷靜下來之後,我決定先試試,手動在模型中該表table屬性來試試。
直接修改模型的table屬性
那麼我們再來通路下呢?
image.png
報錯,報表不存在,這是正确的,因為我沒有建立order_1這個表。不過這也說明了,order模型的确已經和order_1表關聯起來了。這樣就讓我摸不着頭腦了。以前學習面向對象那些理論又浮現在我們腦海裡。我梳理了下思路
- order模型中定義table屬性,其實是對父類Model中的table的重寫。并且table屬性是一個protected的。那麼就是說,隻有在模型内部或則子類中可以使用和修改
- 在執行個體化模型的地方是控制器,也就是類的外部,理論上外部是對象隻能讀取和操作類中public的屬性的
- 可是initialize又是在執行個體化模型自動觸發的方法,觸發的地方又是在模型的内部。
-
但是我又是使用$this 又是指代我控制器中的被執行個體化出來的模型對象。那是不是還是不能通路被保護的屬性呢?
好了瞎分析完了之後,我決定還是去網上搜尋下,看看有沒有人和我一樣的應用場景。後來我發現,使用模型初始化的人,似乎很少。少數幾篇部落格講解了下。其中有個應用場景和我類似。他在代碼中是這樣寫的
protected function initialize()
{
parent::initialize(); // TODO: Change the autogenerated stub
$this->table('order_1');//假裝這裡名字是從資料庫裡取得
}
我通過IDE的智能感覺,進入套table方法中,看了下注釋
看樣子是我需要的方法,可這個方法不是模型基類裡的啊,是在query類當中的。我有些摸不清楚頭腦,但不管怎樣,還是要試一試
為了讓代碼不報錯,我去增加了一個order_1表。這一試,嘿嘿,搞定了!
列印出了我在新表中的一條資料。哈哈,看來這個思路是可行的
就在我認為這個思路是可行的時候,我在執行我寫好的一些模型方法時,我發現了一個大坑!!
我就不詳細說我是怎麼發現的了。直接看代碼
模型裡我還是這麼寫的
protected function initialize()
{
parent::initialize(); // TODO: Change the autogenerated stub
$this->table('order_1');//假裝這裡名字是從資料庫裡取得
}
//控制器裡執行個體化模型後,調用count方法
$order = new \app\API\model\Order();
var_dump($order->count());
var_dump($order->count());die;
結果讓人非常的震驚!
結果
同一個對象,調用同一個方法,結果居然不一樣!我反複試過都是這樣。我決定将他們的sql列印出來看看有什麼問題
結果同樣是讓人哭笑不得
結果模型初始化是一次性的?(黑人問号.jpg)
我當時就覺得是tp5的bug,我要向作者反應!
後來我冷靜下來,還是先把項目問題解決了來再說哦。飯碗要緊。
最終經過我的摸(luan)索(gao) ,我找到解決方案。下面就把代碼貼出來,但是我确實不知道怎麼解釋這個問題。也希望大神能夠指出,感激不盡
protected function initialize()
{
parent::initialize(); // TODO: Change the autogenerated stub
$this->name('order_1');//将table方法換位name方法
}
name方法是也是指定表名,隻是不帶表字首。經測試,傳入不含表字首的表名可行,我這裡的資料表設計的時候沒有表字首,是以傳的都一樣。
那麼改過之後,再來列印下之前的sql
經測試可以正常切換兩個表~
本次部落格隻在記錄,内容中有很多自己瞎猜的,站不住腳,讓大神見笑。非常希望能有大神指點