程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> 網頁編程 >> PHP編程 >> PHP綜合 >> PHP的Yii框架使用中的一些錯誤解決方法與建議

PHP的Yii框架使用中的一些錯誤解決方法與建議

編輯:PHP綜合

 此文意在記錄 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)); 

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved