此文意在記錄 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 也由你自己定義。
<form action="your url" method='post' enctype="multipart/form-data" id='fileform'> <p style='display:inline-block'>文件上傳 </p><input id='file1' name='file1' type='file' ></input> <br /> <input type='submit' value='上傳'> </form>
這是服務器端接收並保存文件的代碼,文件最後保存到了 attached 文件夾的 file 文件夾裡:
if(isset($_FILES['file1'])) { $xlsfile = $_FILES['file1']; $tmp_name = $xlsfile['tmp_name']; /*獲取文件名*/ $file_name = basename($xlsfile_name); if($xlsfile['error'] > 0) { echo "文件上傳出錯!請重試。<br />"; 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 文件夾,且創建失敗!<br />"; 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));