最近遇到的 Yii 问题还挺多,再次献上几个 Yii 问题的解决方案。
1.关联表查询相同字段出错。
有时候我们建了两个表,但是两个表有相同的字段,在用 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);
2.文件上传
说起来这个不算是 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('无法移动文件!');
}
3. 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);
4.检查日期格式合法性
有时我们需要检验用户填写的日期是否合法,可以用下面的函数。
function checkDatetime($dateStr, $format = "Y-m-d H:i:s")
{
$time = strtotime($dateStr);
$checkDate = date($format, $time);
return $checkDate == $dateStr;
}
相信新手都有疑惑,_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));