PHP的Yii架構使用中的一些錯誤解決方案與建議,yii架構_PHP教程

來源:互聯網
上載者:User

PHP的Yii架構使用中的一些錯誤解決方案與建議,yii架構


此文意在記錄 Yii 開發過程中的小問題解決方案,不全面,不權威,不是教程。自己寫過,覺得可以解決問題,以後也可能用上,就記記吧。

1. Yii 中 Js 和 Css 檔案的引入。
我們就從最簡單的問題開始吧,說起來也不是問題,只是文法罷了。假設我們的 js 檔案都放在和 protected 同一層的 js 檔案夾裡,css 檔案都放在和 protected 同一層的 css 檔案夾裡,好吧,規範就是這樣的...那我們可以在對應的 view 介面按下面這樣寫,css 和 js 函數的參數是不同的哦...(之前因為這個調了一個小時..)
註冊 js 檔案的第二個參數是 js 所放的位置,可選三個:CClientScript::POS_HEAD 放在 Head 部分 CClientScript::POS_BEGIN 放在 Body 開始處 CClientScript::POS_END 放在 Body 結束處,沒有特別要求就不用填了...註冊 Css 檔案的第二個參數是 media,,有興趣的同學點這裡,目前還是預設就好...
對於 Jquery 這樣的 js ,用 registerCoreScript 不會造成莫名奇妙的錯誤...

//註冊 js 檔案 Yii::app()->clientScript->registerScriptFile(Yii::app()->baseUrl.'/js/project1.js',CClientScript::POS_HEAD); //註冊 css 檔案 Yii::app()->clientScript->registerCssFile(Yii::app()->baseUrl.'/css/project1.css'); //註冊 Jquery 檔案 Yii::app()->clientScript->registerCoreScript('jquery'); 

2. Yii isNewRecord 修複
Yii 的 Model 的 isNewRecord 屬性是很好用的,可以根據這個屬性進行分情況討論。但是,假如我們開啟了事務機制或是其他情況,造成資料插入後又被復原了,這時資料庫裡沒有該條記錄,但是 isNewRecord 是 flase,即認為已經不是新紀錄了。解決方案是用主鍵去訪問資料庫,判斷究竟是不是新紀錄,而我們在用到這個屬性之前要先按下面處理一下。以下 Model 是 Post,主鍵是 id:

if(!$model->isNewRecord) {   $db_exist = Post::model()->findByPk($model->id);   if($db_exist == NULL)     $model->isNewRecord = true; } 

3.Yii 產生 隱藏輸入欄位
雖然自己寫一個輸入欄位很容易(不就是 display:none 嘛),但是有時架不住需要按照 Yii 的表單代碼格式呀,反正就一句話...

<?php echo $form->hiddenField($model,'name'); ?> <?php if($model->isNewRecord) echo $form->hiddenField($model,'path',array('size'=>60,'maxlength'=>128,'id'=>'path1')); ?> 

4. Yii 產生下拉式功能表
很多時候我們在 form 裡需要一個下拉式功能表,這時候 Chtml 的 listdata 就很好用的。假如我們資料庫裡的欄位只有很少的可能,比如 0 和 1,可以按下面寫:

echo $form->dropDownList($model,'is_marry',array('0'=>'否','1'=>'是')); 

這時候,你看到的就是 是 和 否 的下拉式功能表,選擇 '是' 提交的時候這個欄位填的就是 1 ,'否' 就是 0 。當然,經常不只這麼簡單,我們可以在 Model 裡面添加一個函數用於產生下拉式功能表的數組,然後在 view 裡去調用就行了。這個函數的資料可以自己寫的,或者在資料庫尋找得來的。下面用了 listdata, 具體意思是以 model 中 id 為 鍵, name 為值。

/* 寫在 model 裡 */ public function getUserOptions() {   $models = User::model()->findAll();   $models = User::model()->findAllByAttributes(array('is_regeister'=>'1'));   return CHtml::listdata($models, 'id', 'name'); }  /* 寫在 view 的介面裡 */ echo $form->dropDownList($model,'user_id',User::model()->getUserOptions()); 

5.Yii 開啟事務機制
在你同時儲存幾條記錄到資料庫時,你可能很有必要開啟事務機制。Yii 開啟事務機制很容易,只要三句話就夠了。

/*開啟事務機制*/ $transaction = Yii::app()->db->beginTransaction(); try {   /* 成功則 commit */   $transaction->commit(); } catch(Exception $e) {   $transaction->rollBack(); } 

比較完整的像這樣:

if($_POST['ModelA']) {   /*開啟事務機制*/   $transaction = Yii::app()->db->beginTransaction();   try   {     /*此處省略一堆邏輯*/     $modelA->save();     $modelB->save();      /* 成功則 commit */     $transaction->commit();     $this->redirect(array('view','id'=>$model->id));   }   catch(Exception $e)   {     $transaction->rollBack();   } } 

不過我一般會像下面這樣,有什麼好處請自行體會...

if($_POST['ModelA']) {   /*開啟事務機制*/   $transaction = Yii::app()->db->beginTransaction();   try   {     $validated = true;      /*此處省略一堆邏輯*/     $valid = $modelA->save();     $validated = $valid & $validated;      /*此處繼續省略一堆邏輯*/     $valid = $modelB->save();     $validated = $valid & $validated;      /* 成功則 commit */     if($validated)     {       $transaction->commit();       $this->redirect(array('view','id'=>$model->id));     }     else     {       /*不成功即復原 */       $transaction->rollBack();     }   }   catch(Exception $e)   {     $transaction->rollBack();   } } 

6.關聯表查詢相同欄位出錯。
有時候我們建了兩個表,但是兩個表有相同的欄位,在用 CDbCriteria 進行 with 關聯查詢搜尋時候,如果沒有進行額外設定,那會出現查詢錯誤,大概的意思就是 Mysql 語句模糊。這時候,我們在主表設定一個別名就好了,然後查詢相關欄位的時候注意把 名字加上就行。
比如:兩個 Model, Post 和 User,都有一個 id, 在 我們可以像下面這樣寫:

$criteria=new CDbCriteria;  $criteria->alias = "post";  $criteria->with = array('user');  $criteria->compare('post.id',$Post->id,true);  $model = Post::model()->find($criteria); 

7.檔案上傳
說起來這個不算是 Yii 的,基本都是原生的 HTML 和 PHP,懶得分,就直接放這裡吧。
下面是 HTML,action 改為你自己的 url, id 和 name 也由你自己定義。

 

這是伺服器端接收並儲存檔案的代碼,檔案最後儲存到了 attached 檔案夾的 file 檔案夾裡:

if(isset($_FILES['file1'])) {   $xlsfile = $_FILES['file1'];   $tmp_name = $xlsfile['tmp_name'];   /*擷取檔案名稱*/   $file_name = basename($xlsfile_name);    if($xlsfile['error'] > 0)   {     echo "檔案上傳出錯!請重試。
"; exit; } else { if(file_exists("attached/tmp/".$file_name)) echo "檔案已存在!本次不予儲存!"; else { if(!is_dir("attached/tmp/")) { /*建立檔案夾,預設許可權 777, true 意味著可以遞迴從建立*/ if(!mkdir("attached/tmp/",0777,true)) { echo "找不到 attached/tmp 檔案夾,且建立失敗!
"; exit; } } /*這個函數僅用於上傳檔案的移動*/ move_uploaded_file($tmp_name,"attached/tmp/".$file_name); } } }

下面是把已存在的檔案從 old_file 路徑移到 attached/file 裡面的當前日期檔案夾。這裡的移動用 rename

/*建立檔案夾*/ $date = date('Y-m-d',time()); $date = str_replace('-',"",$date); $dir = "attached/file/".$date.'/'; if(!is_dir($dir)) {   if(!mkdir($dir,0777,true))   {     exit('無法建立檔案夾!');   } }  /*移動檔案*/ $file_name = basename($old_file); $finish = rename($old_file,$dir.$file_name); if(!$finish) {   exit('無法移動檔案!'); } 

8.YIi 情境與安全欄位
查看當前 Model 情境:

var_dump($model->scenario); 

查看情境的安全欄位。安全欄位的意思是說這些資料由使用者提交的時候不會被 Yii 過濾掉。有次發現網頁提交上來的東西有些有有些沒,調了很久才知道在那個情境下部分被過濾了。

$arr = $model->getSafeAttributeNames($model->scenario); var_dump($arr); 

強制賦值避免 rule 規則過濾欄位。用 setAttributes 可以強製取消 Yii 的安全過濾,只要第二個參數賦值為 false 就好。但是這也只能對這個 Model 產生時就擁有的欄位生效,如果要對包括自己定義的所有欄位不過濾,還是要定義情境然後在 rule 裡指定安全欄位比較好。

if(isset($_GET['Po']))   $model->setAttributes($_GET['Post'],false); 

檢查日期格式合法性
有時我們需要檢驗使用者填寫的日期是否合法,可以用下面的函數。

function checkDatetime($dateStr, $format = "Y-m-d H:i:s") {   $time = strtotime($dateStr);   $checkDate = date($format, $time);    return $checkDate == $dateStr; } 

Yii 渲染多個 model
相信新手都有疑惑,_form 裡面的表單都是渲染一個 model 然後提交給 controller 儲存資料的,如果想要渲染多個 model 怎麼辦呢?
下面,我們假設有兩個 model 類,分別叫做 Person 和 Addr,我們想要做的是在一個 Person 的 _form 裡再渲染幾個 Addr 的 model ,意思是一個人可以有幾個地址。基本思路其實還是很簡單,就是你在 controller 裡定義要渲染的 model 然後傳給 view 介面,最後依然在 controller 裡接收 Post 過來的資料。主要是寫法問題而已,我相信下面大家都能看懂,有疑問的童鞋再留言好了。

//在 controller 裡面  $model=new Person; /* $addrs 儲存 Addr model 的數組,放幾個你就看著辦吧*/ $addrs = array();  if(isset($_POST['Person'])) {   $model->attributes = $_POST['Person'];   /*此處省略一堆邏輯*/   foreach($_POST['Addr'] as $one_addr)   {     $addr = new Addr();     $addr->attributes = $one_addr;     /*此處省略另一堆邏輯*/   } }  $this->render('create',array(   'model'=>$model,   'addrs' => $addrs, ));   //在 view 裡面  /*可以迴圈輸出你的多個 model */ $num = count($addrs); for($i = 0;$i < $num;++$i) {   echo $form->labelEx($addrs[$i],"[{$i}]postcode");   echo $form->textField($addrs[$i],"[{$i}]postcode",array('size'=>10,'maxlength'=>10));   ...; }  /*也可以通過數字指定輸出某個 model */ echo $form->labelEx($addrs[0],"[0]postcode"); echo $form->textField($addrs[0],"[0]postcode",array('size'=>10,'maxlength'=>10)); 

http://www.bkjia.com/PHPjc/1049138.htmlwww.bkjia.comtruehttp://www.bkjia.com/PHPjc/1049138.htmlTechArticlePHP的Yii架構使用中的一些錯誤解決方案與建議,yii架構 此文意在記錄 Yii 開發過程中的小問題解決方案,不全面,不權威,不是教程。自己...

  • 聯繫我們

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