ASP.NET Core教程【三】实体字段属性、链接标签、并发数据异常、文件上传及读取

前文索引: ASP.NET Core教程【二】从保存数据看Razor Page的特有属性与服务端验证 ASP.NET Core教程【一】关于Razor Page的知识

实体字段属性

再来看看我们的实体类

    public class Movie    {        public int ID { get; set; }        public string Title { get; set; }        [Display(Name = "Release Date")]        [DataType(DataType.Date)]        public DateTime ReleaseDate { get; set; }        public string Genre { get; set; }        public decimal Price { get; set; }    }

说明,上面的代码需要引用:using System.ComponentModel.DataAnnotations;

Display属性标志这个字段在页面上显示的时候,需要显示什么名字;

我们在上一篇文章中用到的:

<label asp-for="Movie.Title" class="control-label"></label>

这里就会显示Display属性指定的名字;

DataType属性标志这个字段是什么类型的;

上一章中我们说到的,数据类型的验证工作,就是依据这里标志的数据类型来完成的

比如你可以增加如下数据约束

[StringLength(60, MinimumLength = 3)][Required]
[Range(1, 100)]
[RegularExpression(@"^[A-Z]+[a-zA-Z''-'\s]*$")]
[DataType(DataType.Currency)]

如果你想格式化输出的内容,你可以使用如下的属性注释

[DisplayFormat(DataFormatString = "{0:yyyy-MM-dd}"]

你可以在同一行代码中标记多个属性,如下:

[RegularExpression(@"^[A-Z]+[a-zA-Z''-'\s]*$"), Required, StringLength(30)]

更多说明文档,可以查阅:https://docs.microsoft.com/zh-cn/aspnet/mvc/overview/older-versions/mvc-music-store/mvc-music-store-part-6

链接标签

在上一篇文章中我们简单说了一下链接标签,再来看第一章中提到的这个场景:

<a asp-page="./Edit" asp-route-id="@item.ID">Edit</a> <a asp-page="./Details" asp-route-id="@item.ID">Details</a><a asp-page="./Delete" asp-route-id="@item.ID">Delete</a>

asp-page和asp-route-id两个属性共同决定了最终编译出来的链接地址;

以前我们可能要拼字符串来构造href属性,现在不用了,可以分开写,代码更优雅;

上面的代码,编译后生成的href属性的值是这样的:

http://localhost:5000/Movies/Details?id=2

现在我们分别打开Edit.cshtml、Details.cshtml、Delete.cshtml

把页面中的第一个命令:@page,修改为:@page "{id:int}"

重新编译运行,发现上面的链接变成了:

http://localhost:5000/Movies/Details/1

看到这里你会说“呦~”吗??

如果这个时候你请求这个地址:

http://localhost:5000/Movies/Details

并没有传入ID的值,那么服务器会返回404,

如果你的设计是希望ID是一个可选的传入参数,那么你可以把page指令修改成:

@page "{id:int?}"

这样就好啦!

如果你想让页面接收一个字符串,可以把这个“路由模版”写成如下这个样子:

@page "{searchString?}"

并发数据异常

当一个用户删除了一个实体,另一个用户同时又要更新这个实体的话

第二个用户的请求就会抛出并发数据异常(这里姑且称作并发,微软官网就是这么说的),来看代码:

public async Task<IActionResult> OnPostAsync(){    if (!ModelState.IsValid)    {        return Page();    }    _context.Attach(Movie).State = EntityState.Modified;    try    {        await _context.SaveChangesAsync();    }    catch (DbUpdateConcurrencyException)    {        if (!_context.Movie.Any(e => e.ID == Movie.ID))        {            return NotFound();        }        else        {            throw;        }    }    return RedirectToPage("./Index");}

上面代码中DbUpdateConcurrencyException就是专门针对这种异常定义的异常类;

NotFound方法将返回404异常

文件上传及读取

如果你想上传一个文件,可以撰写如下razor page的代码(只提供一部分表单域)

<div class="form-group">  <label asp-for="FileUpload.UploadPublicSchedule" class="control-label"></label>  <input asp-for="FileUpload.UploadPublicSchedule" type="file" class="form-control" style="height:auto" />  <span asp-validation-for="FileUpload.UploadPublicSchedule" class="text-danger"></span></div>

这个表单域对应的实体如下

using Microsoft.AspNetCore.Http;using System.ComponentModel.DataAnnotations;namespace RazorPagesMovie.Models{    public class FileUpload    {        [Required]        [Display(Name="Title")]        [StringLength(60, MinimumLength = 3)]        public string Title { get; set; }        [Required]        [Display(Name="Public Schedule")]        public IFormFile UploadPublicSchedule { get; set; }    }}

我们只要关注第二个字段即可,UploadPublicSchedule是一个IFormFile类型的字段;

当表单提交后,ASP.NET CORE 也会把文件流绑定到这个字段上;

如果上传的是一个文本文件,那么我们看看怎么直接读取这个文本文件;

public static async Task<string> ProcessFormFile(IFormFile formFile, ModelStateDictionary modelState){            var fieldDisplayName = string.Empty;            // 通过反射拿到实例的字段,再拿到字段的DisplayAttribute            MemberInfo property = typeof(FileUpload).GetProperty(formFile.Name.Substring(formFile.Name.IndexOf(".") + 1));            if (property != null)            {                var displayAttribute = property.GetCustomAttribute(typeof(DisplayAttribute)) as DisplayAttribute;                if (displayAttribute != null)                {                    fieldDisplayName = $"{displayAttribute.Name} ";                }            }            // 通过Path.GetFileName拿到文件名            var fileName = WebUtility.HtmlEncode(Path.GetFileName(formFile.FileName));            if (formFile.ContentType.ToLower() != "text/plain")            {                modelState.AddModelError(formFile.Name, $"The {fieldDisplayName}file ({fileName}) must be a text file.");            }            // 判断文件长度            if (formFile.Length == 0)            {                modelState.AddModelError(formFile.Name, $"The {fieldDisplayName}file ({fileName}) is empty.");            }            else            {                try                {                    string fileContents;                    using (var reader = new StreamReader(formFile.OpenReadStream(), new UTF8Encoding(encoderShouldEmitUTF8Identifier: false, throwOnInvalidBytes: true), detectEncodingFromByteOrderMarks: true))                    {                        fileContents = await reader.ReadToEndAsync();                        // 再验证一遍文件内容的长度,以防文件只有一个BOM头                        if (fileContents.Length > 0)                        {                            return fileContents;                        }                        else                        {                            modelState.AddModelError(formFile.Name, $"The {fieldDisplayName}file ({fileName}) is empty.");                        }                    }                }                catch (IOException ex)                {                    modelState.AddModelError(formFile.Name, $"The {fieldDisplayName}file ({fileName}) upload failed. Please contact the Help Desk for support.");                }            }            return string.Empty;}

调用上面方法的代码如下:

var publicScheduleData = await FileHelpers.ProcessFormFile(FileUpload.UploadPublicSchedule, ModelState);

其中ModelState是PageModel特有的属性

在本示例中,用于给页面添加错误信息~

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏me的随笔

ASP.NET MVC5中View-Controller间数据的传递

使用ASP.NET MVC做开发时,经常需要在页面(View)和控制器(Controller)之间传递数据,那么都有哪些数据传递的方式呢?

18110
来自专栏vue

.Net—反射

新建一个空白解决方案,添加一个控制台应用程序和一个名为Common的类库。在Common里面添加一个Person类和Student类,代码如下

24730
来自专栏芋道源码1024

哪个更快:Java 堆还是本地内存

使用Java的一个好处就是你可以不用亲自来管理内存的分配和释放。当你用new关键字来实例化一个对象时,它所需的内存会自动的在Java堆中分配。堆会被垃圾回收器进...

12940
来自专栏GreenLeaves

C# 文件读写系列二

读取文件原则上非常简单,但它不是通过FileInfo和DirectoryInfo来完成的,关于FileInfo和DirectoryInfo请参考C# 文件操作系...

35890
来自专栏大内老A

让我们的ASP.NET MVC应用可以单独维护验证消息

在项目开发中,我们会使用到很多的描述性文字,比如验证消息、错误消息和确认消息等,让这些文本消息具有可维护性具有重要的意义。虽然我们可以将它们存储于资源文件中,并...

21170
来自专栏张善友的专栏

在Linux和Windows平台上操作MemoryMappedFile(简称MMF)

操作系统很早就开始使用内存映射文件(Memory Mapped File)来作为进程间的共享存储区,这是一种非常高效的进程通讯手段。.NET 4.0新增加了一个...

23260
来自专栏Android知识点总结

2-SIII-Android数据固化之Xml的Pull解析和存储

13130
来自专栏圣杰的专栏

ABP入门系列(4)——创建应用服务

一、解释下应用服务层 应用服务用于将领域(业务)逻辑暴露给展现层。展现层通过传入DTO(数据传输对象)参数来调用应用服务,而应用服务通过领域对象来执行相应的业务...

32370
来自专栏生信宝典

Linux学习-文件排序和FASTA文件操作

环境变量的补充 PATH只是众多环境变量中的一个变量,用于存储可执行文件所在的目录,以便在用户输入命令时可以查询的到。尤其是自己写的脚本或安装的程序,系统不会知...

288100
来自专栏开发 & 算法杂谈

Hiredis源码阅读(一)

Hiredis库主要包含三类API:同步api、异步api以及回复解析api。首先介绍一下同步api以及回复解析api。

644110

扫码关注云+社区

领取腾讯云代金券