上传包含在 MVC 模型中的图像

Posted

技术标签:

【中文标题】上传包含在 MVC 模型中的图像【英文标题】:Upload image included in MVC model 【发布时间】:2013-12-25 03:26:06 【问题描述】:

我有以下型号:

public class Photo

    public int PhotoId  get; set; 
    public byte[] ImageData  get; set; 
    public DateTime DateUploaded  get; set; 
    public string Description  get; set; 
    public bool IsActive  get; set; 


我希望用户能够输入照片的详细信息,然后将模型发布到控制器。我的控制器动作如下:

[HttpPost]
    public ActionResult Create(WilhanWebsite.DomainClasses.Photo photo)
    
        if (ModelState.IsValid)
        
            photo.DateUploaded = DateTime.Now;
            _context.Photos.Add(photo);
            _context.SaveChanges();

            return RedirectToAction("Index");
        
        //we only get here if there was a problem
        return View(photo);
    

我的看法如下:

@using (html.BeginForm()) 

@Html.AntiForgeryToken()

<div class="form-horizontal">
    <h4>Photo</h4>
    <hr />
    @Html.ValidationSummary(true)

    <div class="form-group">
        @Html.LabelFor(model => model.ImageData, new  @class = "control-label col-md-2" )
        <div class="col-md-10">
            <input type="file" name="uploadImages" class="input-files" />
        </div>
    </div>

    <div class="form-group">
        @Html.LabelFor(model => model.DateUploaded, new  @class = "control-label col-md-2" )
        <div class="col-md-10">
            @Html.EditorFor(model => model.DateUploaded)
            @Html.ValidationMessageFor(model => model.DateUploaded)
        </div>
    </div>

    <div class="form-group">
        @Html.LabelFor(model => model.Description, new  @class = "control-label col-md-2" )
        <div class="col-md-10">
            @Html.EditorFor(model => model.Description)
            @Html.ValidationMessageFor(model => model.Description)
        </div>
    </div>

    <div class="form-group">
        @Html.LabelFor(model => model.IsActive, new  @class = "control-label col-md-2" )
        <div class="col-md-10">
            @Html.EditorFor(model => model.IsActive)
            @Html.ValidationMessageFor(model => model.IsActive)
        </div>
    </div>

    <div class="form-group">
        <div class="col-md-offset-2 col-md-10">
            <input type="submit" value="Create" class="btn btn-default" />
        </div>
    </div>
</div>

视图显示正常,允许用户从本地磁盘中选择文件并输入其他模型详细信息。 我的问题是,虽然模型已发布到控制器上,但描述、日期和 IsActive 标志已填充好 - 图像数据为空。

谁能告诉我我需要更改什么,以便将照片的字节数组包含在发布到控制器的模型中?

【问题讨论】:

【参考方案1】:

您视图中的文件输入名称为uploadImages。我在您的视图模型中看不到具有此名称的属性。您似乎有一些 ImageData 属性,它是一个字节数组,但您的视图中似乎没有具有此名称的相应输入字段。

这解释了为什么你会得到 null。您可以通过遵守约定来完成这项工作。因此,例如,如果您打算在视图中有这样一个输入字段:

<input type="file" name="uploadImages" class="input-files" />

然后确保您的视图模型上有一个同名的属性。当然还有HttpPostedFileBase

public HttpPostedFileBase UploadImages  get; set; 

在您看来,请确保您设置了正确的 multipart/form-data 内容类型:

@using (Html.BeginForm(null, null, FormMethod.Post, new  enctype = "multipart/form-data" ))

    ...

您可能希望通过following blog post 更好地熟悉如何在 ASP.NET MVC 中上传文件的基础知识。我还写了一个similar answer here,你可以参考一下。

因此,一旦在视图模型中添加带有 UploadImages 名称的 HttpPostedFileBase 属性,您就可以调整控制器操作以读取字节数组并将其存储为您的 ImageData 属性:

[HttpPost]
public ActionResult Create(WilhanWebsite.DomainClasses.Photo photo)

    if (ModelState.IsValid)
    
        photo.DateUploaded = DateTime.Now;
        photo.ImageData = new byte[photo.UploadImages.ContentLength];
        photo.UploadImages.Read(photo.ImageData, 0, photo.ImageData.Length);

        _context.Photos.Add(photo);
        _context.SaveChanges();

        return RedirectToAction("Index");
    

    //we only get here if there was a problem
    return View(photo);

现在请记住,这是一个绝对糟糕的解决方案。永远不要在现实世界的应用程序中这样做。在正确设计的应用程序中,您将拥有一个视图模型,您的控制器操作将作为参数。您永远不会直接使用自动生成的 EF 模型作为控制器操作的参数。您将拥有一个带有 HttpPostedFileBase 属性的视图模型,该属性将映射到您的域模型。

因此,在设计合理的应用程序中,您将拥有一个 PhotoViewModel 视图模型类,您的控制器操作将执行该类。

【讨论】:

另一个很好的答案感谢达林。我将创建一个专用的视图模型。我为此解决方案使用了一个存储库项目,但我发现额外的抽象层确实使我的 moq'd 单元测试变得复杂。 不应该是这样的:photo.UploadImages.InputStream.Read【参考方案2】:

改变这一行:

@using (Html.BeginForm()) 

到这里:

@using (Html.BeginForm(null, null, FormMethod.Post, new  enctype = "multipart/form-data" ))

然后改变:

<input type="file" name="uploadImages" class="input-files" />

收件人:

<input type="file" name="ImageData" class="input-files" />

然后改变这一行:

 public byte[] ImageData  get; set; 

到这里:

 public HttpPostedFileBase ImageData  get; set; 

最后,使用类似这样的代码将图像读入字节数组:

 var bs = new byte[ImageData.ContentLength];
 using (var fs = ImageData.InputStream)
 
     var offset = 0;
     do
     
         offset += fs.Read(bs, offset, bs.Length - offset);
      while (offset < bs.Length);
 

【讨论】:

【参考方案3】:

查看:

@using (Html.BeginForm(null, null, FormMethod.Post, new  enctype = "multipart/form-data" ))

   ...
   <input type="file" id="ImageFile" name="ImageFile" .../>
   ...

控制器:

[HttpPost]
public ActionResult Create(Photo photo, HttpPostedFileBase ImageFile)

    byte[] buf = new byte[ImageFile.ContentLength];
    ImageFile.InputStream.Read(buf, 0, buf.Length);
    photo.ImageData = buf;
    ...

【讨论】:

以上是关于上传包含在 MVC 模型中的图像的主要内容,如果未能解决你的问题,请参考以下文章

ASP.Net MVC 5 图像上传到文件夹

是或否:MVC 中的模型是不是应该包含应用程序逻辑?

MVC 中的模型类与模型

MVC 3 应用程序中的模型、视图模型、DTO

如何在 joomla 管理组件中上传图像?

自定义MVC框架之工具类-模型类