一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - PHP教程 - Yii2框架中一些折磨人的坑

Yii2框架中一些折磨人的坑

2021-09-23 16:13有痣青年 PHP教程

這篇文章主要給大家介紹了關于Yii2框架中一些折磨人的坑,文中通過示例代碼介紹的非常詳細,對大家學習或者使用Yii2框架具有一定的參考學習價值,需要的朋友們下面來一起學習學習吧

說點閑話

距離上次寫博客,已經有一年了。在動手寫之前,總是帶著深深的罪惡感。被它折磨許久,終于,還是,動手了。

值得慶祝的一件事:最近開始健身了。每天動感單車45分鐘,游泳45分鐘,真的是(生)爽(不)到(如)爆(死)。

好了,扯淡完畢,步入正題。

ActiveRecord被莫名寫入?

準備知識

ActiveRecord的基本用法。如果不理解,可參考這里

代碼現場

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
/**
 * @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();    //保存,會發現此行的其它字段都被寫成默認值了。

總結問題

這個例子的問題在于:

  1. 我從數據庫中取出了一行,也就是代碼中的$room,但是只取出了id字段,而其他字段自然就是默認值。
  2. 當我$room->save()的時候,那些是默認值的字段也被保存到數據庫里去了。what!?
  3. 也就是說,當你想節約資源,不取出所有字段的時候,一定要注意不能保存,否則,很多數據會被莫名修改為默認值。

解決方法

然而,我們有什么解決辦法呢?提供幾種思路:

  1. 自己時刻注意,避免未完全取出的ActiveRecord的保存。
  2. 修改或繼承ActiveRecord, 使得,當此對象由find()新建,且字段沒有完全取出,調用save()方法,拋出異常。
  3. 修改或繼承ActiveRecord,使得,當此對象由find()新建,且字段沒有完全取出,調用save()方法時,只保存取出過的字段,其他字段被忽略。

你的Transaction生效了嗎?

代碼現場

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
/**
 * @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();
}

問題總結

這段代碼的問題在于:

  1. 大家知道$transaction的存在意義是保證整段數據庫存儲代碼要么全成功,要么全失敗。
  2. 顯然,在這個例子中,transaction并沒有達到我們想要的效果:$a因為validate()都沒過,所以$transation->commit()的時候并不會報錯。

解決方法

在$transation塊內,所有的save()都要判斷下返回值,如果為false,則直接拋出異常。

'Y-m-d'不被識別?

代碼現場

?
1
2
3
4
5
6
7
8
9
10
11
12
13
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,世界就清凈了。所以,如果你遇到這種問題,感激我吧。

內存泄露

代碼現場

?
1
2
3
4
5
6
7
8
9
10
public static function actionTest() {
  $total = 10;
  var_dump('開始內存'.memory_get_usage());
  while($total){
   $ret=User::findOne(['id'=>910002]);
   var_dump('end內存'.memory_get_usage());
   unset($ret);
   $total--;
  }
 }

上面代碼的內存一直在增長, 按照原本想法來看, 變量被釋放了,內存就算增長也不會一直增長。因為每循環一次內存都會被釋放。

分析問題 上面這段代碼涉及到了數據庫的操作,而我們知道,數據庫的很多地方都能引起內存泄漏。 所以先屏蔽數據庫相關操作, 我手寫了一個原生的數據庫查詢操作, 發現內存正常,沒有問題。

?
1
2
3
4
5
6
7
8
9
$dsn = "mysql:dbname=test;host=localhost";
$db_user = 'root';
$db_pass = 'admin';
//查詢
$sql = "select * from buyer";
$res = $pdo->query($sql);
foreach($res as $row) {
 echo $row['username'].'<br/>';
}

這時候答案呼之欲出--- 是yii2框架搞了鬼

定位問題 既然知道了是yii2 框架的問題那就可以進一步縮小問題。

?
1
2
3
4
5
6
7
8
9
10
public static function actionTest() {
  $total = 10;
  var_dump('開始內存'.memory_get_usage());
  while($total){
   $ret= new User();
   var_dump('end內存'.memory_get_usage());
   unset($ret);
   $total--;
  }
 }

內存還是一直增長。 這時候我測試了一個其他的yii2類 發覺內存不增長了。 這就可以聯想到是在new 對象的時候yii2內部自己執行了什么操作,然后導致內存泄漏。 什么方法是new 的時候就執行的呢。。。 對的 構造方法 __construct 。 然后 我一步一步的從model 查到object 發覺都沒有能引起泄漏的地方。

這個時候我們不妨換個思路, 既然是yii2框架下出現的泄漏, 那肯定就是yii2獨有的功能, 那什么功能是yii2獨有的,又是在new 對象的時候就會執行的呢?

行為(Behavior) 發覺我的模型類里面果然有用了行為

?
1
2
3
4
5
6
public function behaviors()
 {
  return [
   TimestampBehavior::class,
  ];
 }

最普通不過的代碼。 我們知道 行為最后調用的地方是 yii\base\Component->attachBehaviors 最后定位到

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
private function attachBehaviorInternal($name, $behavior)
 {
  if (!($behavior instanceof Behavior)) {
   $behavior = Yii::createObject($behavior);
  }
  if (is_int($name)) {
   $behavior->attach($this);
   $this->_behaviors[] = $behavior;
  } else {
   if (isset($this->_behaviors[$name])) {
    $this->_behaviors[$name]->detach();
   }
   $behavior->attach($this);
   $this->_behaviors[$name] = $behavior;
  }
 
  return $behavior;
 }

我們觀察這段代碼,發覺他把自己傳進去了$behavior->attach($this); 最后調用的是 yii\base\Behavior->attach

?
1
2
3
4
5
6
7
public function attach($owner)
 {
  $this->owner = $owner;
  foreach ($this->events() as $event => $handler) {
   $owner->on($event, is_string($handler) ? [$this, $handler] : $handler);
  }
 }

問題總結

這個時候答案已經呼之欲出, Yii2為了實現行為這一功能, 把自身this傳進去,以便能注冊事件、觸發事件、解除事件。 這就導致了一個循環引用的問題。 所以導致對象refcount一直不為0 一直回收不了。

接下來就好辦了。將查詢換成原始的連接試試。果然,內存上升的非常慢了,可以說這才是正常現象。現在的內存也就是50m左右,cpu也穩定在7%左右。

代碼優化后,再跑腳本,1分鐘左右吧,腳本就跑完了。重點是不會再報出內存錯誤了。所以,以后考慮問題還是要深入。敢于質疑。以后如果遇到這種內存錯誤,一定要先檢查自己的代碼是不是有內存泄漏的地方。不要想著先設置php的內存。這樣只會治標不治本。

總結

1、從開發速度方面,借助于gii腳手架,可以快速生成代碼,也就是說搭建一個可以增刪改查的系統可能一行代碼都不用寫,而且集成了jquery和bootstrap,特效和樣式基本也不需要寫了,這對于設計和審美能力普遍較差的后端程序員來說簡直是一大福利。不過在前后端完全的分離的趨勢下,Yii2前后端的耦合的還是有些重了。

2、從代碼的可讀性方面,Yii不會為了刻板地遵照某種設計模式而對代碼進行過度的設計?;旧项愒贗DE里不借助第三方組件是可以跳轉閱讀源碼的。這點上Yii要比Laravel略勝一籌。

3、從開源生態圈方面,Yii因為人少,稍微偏門一點的資料就很少,需要強大的谷歌能力和閱讀英文文檔的能力。

不可否認,Yii是一個優秀的開發框架,值得PHP開發者上手學習,踩坑的過程也是一種成長與積累。最后祝愿PHP小伙伴們都健健康康,事業有成。

原文鏈接:https://www.cnblogs.com/zydj333/p/12038025.html

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 色综合久久中文字幕 | 狠狠色96视频 | 亚洲国产精品综合久久一线 | 国产精品美女福利视频免费专区 | 啾咪成人漫画免费 | 男gay男gay男gay野外 | 500av导航大全精品 | 成人影院观看 | 色综合视频一区二区三区 | 国产成人一区二区三区影院免费 | 亚洲va欧美va天堂v国产综合 | 青草视频在线观看免费资源 | 欧美日韩国产一区二区三区伦 | 亚洲精品视 | 亚洲精品在看在线观看 | 亚洲天堂h | 激情小视频 | 91亚洲精品国产自在现线 | 欧美一级在线播放 | aigao视频| 我与白丝同桌的故事h文 | 成人国产在线视频在线观看 | 日韩有码 | 国产精品久久久精品视频 | 色四虎| 91大神在线观看精品一区 | 亚洲bt区 | 和肥岳在厨房激情 | chinaspanking调教 chanelpreston欧美网站 | 亚洲天堂视频在线免费观看 | 女女性恋爱免费 | 四虎影视免费观看免费观看 | 18young第一次 | 日本黄色一区 | 出轨同学会2在线观看 | 国产在线综合网 | 亚洲国产一区二区三区a毛片 | 午夜国产在线视频 | 极品91 | 美女黄板视频 | 91精品国产综合久久 |