webclz 2015-01-06 17:04:57 57724次浏览 22条评论 76 12 0

Yii2 用表单上传文件经常用到的,该怎样上传呢?

1、单个文件上传

首先建立一个模型 models/UploadForm.php,内容如下

namespace app\models;

use yii\base\Model;
use yii\web\UploadedFile;

/**
 * UploadForm is the model behind the upload form.
 */
class UploadForm extends Model
{
    /**
     * @var UploadedFile file attribute
     */
    public $file;

    /**
     * @return array the validation rules.
     */
    public function rules()
    {
        return [
            [['file'], 'file'],
        ];
    }
}

再建立一个视图文件,内容如下

<?php
use yii\widgets\ActiveForm;
?>

<?php $form = ActiveForm::begin(['options' => ['enctype' => 'multipart/form-data']]) ?>

<?= $form->field($model, 'file')->fileInput() ?>

<button>Submit</button>

<?php ActiveForm::end() ?>

最后建立控制器文件,内容如下

namespace app\controllers;

use Yii;
use yii\web\Controller;
use app\models\UploadForm;
use yii\web\UploadedFile;

class SiteController extends Controller
{
    public function actionUpload()
    {
        $model = new UploadForm();

        if (Yii::$app->request->isPost) {
            $model->file = UploadedFile::getInstance($model, 'file');

            if ($model->file && $model->validate()) {                
                $model->file->saveAs('uploads/' . $model->file->baseName . '.' . $model->file->extension);
            }
        }

        return $this->render('upload', ['model' => $model]);
    }
}

注意这里我们没有用model->load(...),而是用了UploadedFile::getInstance(...)。区别是后者不会执行$model->validate(),所以需要手动的去执行$model->validate()来检验数据的合法性。如果检验通过了,上传的文件保存在uploads文件夹下,即web目录下的uploads里。

一些可选的配置选项
上传文件不能为空

public function rules()
{
    return [
        [['file'], 'file', 'skipOnEmpty' => false],
    ];
}

上传类型,不仅可以根据扩展名检验,还可以根据文件的内容进行检验

public function rules()
{
    return [
        [['file'], 'file', 'extensions' => 'jpg, png', 'mimeTypes' => 'image/jpeg, image/png',],
    ];
}

2、多文件上传

如果你想一次上传多个文件,只需调节几个参数就可以达到目的

Model:

class UploadForm extends Model
{
    /**
     * @var UploadedFile|Null file attribute
     */
    public $file;

    /**
     * @return array the validation rules.
     */
    public function rules()
    {
        return [
            [['file'], 'file', 'maxFiles' => 10], // <--- here!
        ];
    }
}

View:

<?php
use yii\widgets\ActiveForm;

$form = ActiveForm::begin(['options' => ['enctype' => 'multipart/form-data']]);
?>

<?= $form->field($model, 'file[]')->fileInput(['multiple' => true]) ?>

    <button>Submit</button>

<?php ActiveForm::end(); ?>

与单文件上传不同的是下面这句

$form->field($model, 'file[]')->fileInput(['multiple' => true])

Controller:

namespace app\controllers;

use Yii;
use yii\web\Controller;
use app\models\UploadForm;
use yii\web\UploadedFile;

class SiteController extends Controller
{
    public function actionUpload()
    {
        $model = new UploadForm();

        if (Yii::$app->request->isPost) {
            $model->file = UploadedFile::getInstances($model, 'file');

            if ($model->file && $model->validate()) {
                foreach ($model->file as $file) {
                    $file->saveAs('uploads/' . $file->baseName . '.' . $file->extension);
                }
            }
        }

        return $this->render('upload', ['model' => $model]);
    }
}

这样就可以实现多文件上传了。
参考 https://github.com/yiisoft/yii2/blob/master/docs/guide/input-file-upload.md

觉得很赞
  • 评论于 2015-02-02 13:08 举报

    照着做的,怎么访问啊!

  • 评论于 2015-02-15 10:30 举报

    收藏一下,以后可能用得着

  • 评论于 2015-03-02 18:11 举报

    开始学习 Yii2,又开眼界了

    1 条回复
    评论于 2017-11-23 21:18 回复

    兄弟,怎么收藏啊

  • 评论于 2015-03-09 10:43 举报

    怎么表示才能显示我觉得很赞

  • 评论于 2015-04-02 11:55 举报

    码一个.......

  • 评论于 2015-04-13 16:04 举报

    必须手工先建立uploads目录,不然会出错。

    2 条回复
    评论于 2015-07-24 15:57 回复

    可以判断一下目录是否存在啊,不存在新建一个目录就可以了。

    评论于 2015-07-24 15:58 回复
    //自己设置的上传文件存放路径
    $filePath = $this->fileExists(Yii::$app->basePath.'/uploads/');  //上传路径
    public function fileExists($uploadpath)
    {
        if(!file_exists($uploadpath)){
            mkdir($uploadpath);
        }
        return $uploadpath;
    }
    
  • 评论于 2015-08-11 12:48 举报

    中文名的文件。。就不能正常了。。

  • 评论于 2016-07-06 11:27 举报

    单文件是可以的,改成多文件 $uploadsFile = UploadedFile::getInstance($model, 'image'); 取不到值? 求教

    1 条回复
    评论于 2017-05-19 15:15 回复

    多个请使用:UploadedFile::getInstances();

    , , 觉得很赞
  • 评论于 2016-07-13 20:52 举报

    已收藏,楼主继续

  • 评论于 2016-08-02 14:44 举报

    单文件上传,中文的文件名上传后是乱码

    觉得很赞
  • 评论于 2017-01-21 12:21 举报

    怎么设定上传文件大小啊

  • 评论于 2017-03-30 17:41 举报

    楼主厉害,全靠你了

  • 评论于 2017-03-31 14:59 举报

    为什么我的view里面只有,只有name 为file的input才能上传成功呢?我想弄两个input,名字不一样。但是只有名字为file的才成功了,这是为什么呢?求楼主大大指教。

  • 评论于 2017-04-27 09:05 举报

    为什么不可以传相同的图片 怎么上传到数据库 可以教教吗

  • 评论于 2017-05-08 16:24 举报
    $model->file = UploadedFile::getInstances($model, 'file');
    

    这里还要传入 $model 啊??要没有 $model 怎么办,因为是做一种文件上传接口,给其他的平台上传文件,这是要我用 $_FILE 了吗?

  • 评论于 2017-05-09 16:45 举报

    对这框架设计好无语的,上传文件,还非要建立个模型。简单功能是过渡设计。。。

  • 评论于 2017-07-06 09:05 举报

    这个还是很不错的,刚开始有些不适应,现在好多了,mark!

    2 条回复
    评论于 2017-07-24 15:24 回复

    这设计是非常不好,如果是图片是异步先上传,用这model验证方法,就一点都不好用了。

    评论于 2017-10-27 19:56 回复

    这个一般都放在前端验证比较好,能少用后端资源尽量少用后端资源。而且,还要写规则很烦。

  • 评论于 2017-09-02 14:52 举报

    为什么我的就是一直请求地址出错呢?这是为什么啊?

    觉得很赞
  • 评论于 2017-10-26 15:30 举报

    请问要如何在一份新闻后台里加上这段?我不是需要一个单独的上传,而是一个新闻,标题,内容,图片,作者,点击量,然后中间需要上传图片。

  • 评论于 2017-10-26 16:10 举报

    求问:如果说想要上传过程中直接由程序修改文件名,要在哪一步写

您需要登录后才可以评论。登录 | 立即注册