Yii 不完全解决方案(二)

最近遇到的 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;
}

5.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));

Yii 不完全解决方案(二),布布扣,bubuko.com

时间: 2024-11-08 18:18:39

Yii 不完全解决方案(二)的相关文章

Yii 不完全解决方案

此文意在记录 Yii 开发过程中的小问题解决方案,不全面,不权威,不是教程.自己写过,觉得可以解决问题,以后也可能用上,就记记吧. 1. Yii 中 Js 和 Css 文件的引入. 我们就从最简单的问题开始吧,说起来也不是问题,只是语法罢了.假设我们的 js 文件都放在和 protected 同一层的 js 文件夹里,css 文件都放在和 protected 同一层的 css 文件夹里,好吧,规范就是这样的...那我们可以在对应的 view 界面按下面这样写,css 和 js 函数的参数是不同的

跨域解决方案二:使用JSONP实现跨域

跨域的实现方式有多种,除了 上篇文章 提到的CORS外,常见的还有JSONP.HTML5.Flash.iframe.xhr2等. 这篇文章对JSONP的跨域原理进行了探索,并将我的心得记录在这里和大家分享. JSONP跨域原理探秘 我们知道,使用 XMLHTTPRequest 对象发送HTTP请求时,会遇到 同源策略 问题,域不同请求会被浏览器拦截. 那么是否有方法能绕过 XMLHTTPRequest 对象进行HTTP跨域请求呢? 换句话说,不使用 XMLHTTPRequest 对象是否可以发送

ios7.1安装提示&quot;无法安装应用程序 因为证书无效&quot;的解决方案二(dropbox被封项目转移到Appharbor上)

6月18日起dropbox被天朝封了(这个真是无力吐槽),而ios7.1要求使用ssl安全连接,则需要重新找到一个支持https的免费服务器.Appharbor是个不错的选择,操作简单,此外需要添加配置文件来识别plist,ipa文件,有关如何使用Appharbor转自: 免费ASP和ASP.NET空间Webweb和.NET云计算空间Appharbor 一.Appharbor免费.NET云计算空间申请 PS:Appharbor官网: https://appharbor.com/ 1.进入Apph

yii 整合phpqrcode生成二维码附在线演示地址

1,先到官网下载包  http://phpqrcode.sourceforge.net/ 下载官网提供的类库后,只需要使用phpqrcode.php就可以生成二维码了,当然您的PHP环境必须开启支持GD2. phpqrcode.php提供了一个关键的png()方法,其中参数$text表示生成二位的的信息文本:参数$outfile表示是否输出二维码图片 文件,默认否:参数$level表示容错率,也就是有被覆盖的区域还能识别,分别是 L(QR_ECLEVEL_L,7%),M(QR_ECLEVEL_M

Yii 不完全解决方案(一)

此文意在记录 Yii 开发过程中的小问题解决方案 1. Yii 中 Js 和 Css 文件的引入. 我们就从最简单的问题开始吧,说起来也不是问题,只是语法罢了.假设我们的 js 文件都放在和 protected 同一层的 js 文件夹里,css 文件都放在和 protected 同一层的 css 文件夹里,好吧,规范就是这样的...那我们可以在对应的 view 界面按下面这样写,css 和 js 函数的参数是不同的哦...(之前因为这个调了一个小时..) 注册 js 文件的第二个参数是 js 所

基于MVC和Bootstrap的权限框架解决方案 二.添加增删改查按钮

上一期我们已经搭建了框架并且加入了列表的显示, 本期我们来加入增删改查按钮 整体效果如下 HTML部分,在HTML中找到中意的按钮按查看元素,复制HTML代码放入工程中 <a class="btn btn-small element" data-original-title="新增" href="/Customer/Add" data-toggle="tooltip" data-placement="top&q

回车键转tab键解决方案二

<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%><%@ taglib prefix="s" uri="/struts-tags"%><!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional

离线地图解决方案(二):地图基本控制

地图鹰眼,比例尺,地图缩放控件应用,地图放大.缩小.设置中心点,设置瓦片显示层级等.功能没什么好说的,直接上图: <!DOCTYPE HTML PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> <html xmlns="http://www.w3.org/1999/xhtml&

(学习笔记)页面布局解决方案二

垂直居中解决方法: 方法1:table-cell+vertical-alignl .parent{display:table-cell;vertical-align:middle;} 方法2:absolute+transform .parent{position:relativee;} .child{position:absolut;top:50%;transform:translateY(-50%);} 方法3:flex+align-items .parent{display:flex;ali