Yii2框架的幾個隱蔽的坑
摘要:Yii2是一款優秀的通用Web后端框架,結構簡單優雅、實用功能豐富、擴展性強、性能搞是他最突出的優點。它優秀的地方你在使用過程中總能輕易的發現,無須贅述。而這些隱蔽的小瑕疵,顯得更有必要告訴大家。
- 博客: http://www.cnblogs.com/jhzhu
- 郵箱: jhzhuustc@gmail.com
- 作者: 知明所以
- 時間: 2015-08-17
目錄
說點閑話
距離上次寫博客,已經有三個月了。在動手寫之前,總是帶着深深的罪惡感。被它折磨許久,終於,還是,動手了。
值得慶祝的一件事:最近開始,每天早上8:30起來健身了。有兩個視頻很好用,只需8分鍾,照着做一遍保證你(生)爽(不)到(如)爆(死)。(8分鍾腹肌鍛煉第2級-下載,8分鍾胸肌鍛煉第2級-下載)
值得反思的一件事:最近看了《叔本華美學隨筆》,改變了我一直以來對閱讀的看法。我曾經以為閱讀是進步的源動力,卻被這本書深深的打臉了。來,先給大家分享一段:
我們只管所見的外在環境並不像閱讀物那樣,把某已確定的見解強加給我們的頭腦,而只是為我們提供了素材和機會。去思考與我們的頭腦能力相稱、與當下的情緒相符的事情。所以,太多的閱讀會是我們的精神失去彈性,就像把一重物持續壓在一條彈簧上面就會是彈簧失去彈性一樣;而讓自己沒有自己思想的最穩妥的辦法就是在空閑的每一分鍾馬上隨手拿起一本書。
思考才是進步的源動力!
好了,扯淡完畢,步入正題。
ActiveRecord被莫名寫入?
准備知識
ActiveRecord
的基本用法。如果不理解,可參考這里。
代碼現場
/** * @property integer $id * @property string $name * @property string $detail * @property double $price * @property integer $area **/ class OcRoom extends ActivieRecord { ... } $room = OcRoom::find() //先取出一個對象。 ->select(['id']) //只取出'id'列 ->where(['id'=>20]) ->one(); $room->save(); //保存,會發現此行的其它字段都被寫成默認值了。
總結問題
這個例子的問題在於:
- 我從數據庫中取出了一行,也就是代碼中的
$room
,但是只取出了id
字段,而其他字段自然就是默認值。 - 當我
$room->save()
的時候,那些是默認值的字段也被保存到數據庫里去了。what!? - 也就是說,當你想節約資源,不取出所有字段的時候,一定要注意不能保存,否則,很多數據會被莫名修改為默認值。
解決方法
然而,我們有什么解決辦法呢?提供幾種思路:
- 自己時刻注意,避免未完全取出的
ActiveRecord
的保存。 - 修改或繼承
ActiveRecord
, 使得,當此對象由find()
新建,且字段沒有完全取出,調用save()
方法,拋出異常。 - 修改或繼承
ActiveRecord
,使得,當此對象由find()
新建,且字段沒有完全取出,調用save()
方法時,只保存取出過的字段,其他字段被忽略。
你的Transaction生效了嗎?
代碼現場
/** * @property integer $id * @property string $name **/ class OcRoom extends ActiveRecord { public function rules() { return [['name','string','min'=>2,'max'=>10]]; } ... } class OcHouse extends ActiveRecord { public function rules() { return [['name','string','max'=>10]]; } ... } $a = new OcRoom(); $a->name = ''; //name為空字符串,不滿足rules()條件。 $b = new OcHouse(); $b->name = '我的房間'; //name合法,可以保存。 $transaction = Yii::$app->db->beginTransaction(); try{ $a->save(); //name字段不合法,無法驗證通過,在validate()階段已經返回false,不會進行數據庫存儲的步驟,所以也不會拋出異常。 $b->save(); //name字段合法,可以正常保存。 $transaction->commit(); //提交后,發現$a保存失敗,而$b保存成功。 } catch (Exception $e) { Yii::error($e->getTraceAsString(),__METHOD__); $transaction->rollBack(); }
問題總結
這段代碼的問題在於:
- 大家知道
$transaction
的存在意義是保證整段數據庫存儲代碼要么全成功,要么全失敗。 - 顯然,在這個例子中,
transaction
並沒有達到我們想要的效果:$a
因為validate()
都沒過,所以$transation->commit()
的時候並不會報錯。
解決方法
在$transation
塊內,所有的save()
都要判斷下返回值,如果為false
,則直接拋出異常。
'Y-m-d'不被識別?
代碼現場
OcRenterBill extends ActiveRecord { public function rules() { return [ ['start_time','date','format'=>'Y-m-d'], ]; } } $a = new OcRenterBill(); $a = '2015-09-12'; $a->save(); //會報錯,說格式不對。
問題總結
如果一開始,Yii框架就報錯,這個還不算坑。坑的是我在Mac上開發時,這個可以完全正常的工作,而發布到線上環境(Ubuntu)后,就彈出“屬性start_time格式無效”的錯誤。而參考官方文檔,發現這種格式是允許的官方文檔。
啊啊啊。各種試錯,最后發現如果改成php:Y-m-d
,世界就清凈了。所以,如果你遇到這種問題,感激我吧。