开发中经常会写增删改查的功能,这里记录下在更新操作时遇到的一个问题。

假设一个模型对应数据库中某一张表,在更新时便需要区分是一次性更新全部字段还是仅更新部分字段。希望能做到传递某个参数时便更新,未传递时不更新。

先定义一个用户模型,如下:

public class UserModel
{
public int Id { get; set; }
public string Name { get; set; }
public string Address { get; set; }
}

服务端以这种方式来接收:

public IActionResult Update(UserModel user)
{
// 执行数据库更新操作
return Content(user.Id + user.Name + user.Address);
}

然后客户端以下面的方式来请求(这里用GET方式):

/user/update?id=1&address=hang&name=Hale
/user/update?id=1&address=hang&name=
/user/update?id=1&address=hang

上面有三种传参方式,一般情况下没有问题,但对于第二种形式,&name= 的方式,原本是希望将name字段更新为空值,但是在Action里接收时会发现,user.Name == null 。这样便无法区分是要将Name更新为空值,还是不做更新。

ModelBinder的方式

默认MVC在构造参数模型时没有区分这两种情况,要实现我们的需求就需要自定义一个ModelBinder。定义一个类,并实现IModelBinder接口即可。

public class StringBinder : IModelBinder
{
public object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
{
throw new NotImplementedException();
}
public Task BindModelAsync(ModelBindingContext bindingContext)
{
var valueProviderResult = bindingContext.ValueProvider.GetValue(bindingContext.FieldName); if (valueProviderResult == ValueProviderResult.None)
return Task.CompletedTask; bindingContext.Result = ModelBindingResult.Success(valueProviderResult.FirstValue);
return Task.CompletedTask;
}
}

它提供两个方法来绑定模型,一个同步的一个异步的。我用的 asp.net core 是调用的异步版方法,所以这里仅实现了BindModelAsync。

以这种方式来用这个StringBinder:

public class UserModel
{
public int Id { get; set; }
[ModelBinder(typeof(StringBinder))]
public string Name { get; set; }
[ModelBinder(typeof(StringBinder))]
public string Address { get; set; }
}

这样,再以上面第二种方式传参时,便会发现user.Name == "",而不是 user.Name == null。

这个StringBinder还可以直接用在Action的参数上:

public IActionResult Update(UserModel user, [ModelBinder(typeof(StringBinder))]string v, string v2)
{
return Content(user.Id + user.Name + v + v2);
}

这里参数 v 和 v2,一个指定了Binder一个未指定,以下面方式调用以下即可看出区别:

/user/update?id=1&address=&name=Ingo&v=&v2=

会发现 user.Address == "" ,v == "", v2 == null。

简单的方式

除了自定义ModelBinder的方式,还可以通过直接修改属性的set访问器的办法来区分null和空字符串。

修改下UserModel的代码,新增一个Phone成员:

private string phone;
public string Phone
{
get => phone;
set => phone = string.IsNullOrEmpty(value) ? string.Empty : value;
}

采用与上面相同的方式传值,会发现当传递&phone=时,user.Phone == "", 同样也能区分phone是传递的空字符串还是没传递phone参数。

因为mvc在收到&phone=参数时会调用set访问器,只是value为null。而未收到&phone=xx参数时,不会调用set访问器,所以用这种办法也可以区分空值和null。

ASP.NET MVC请求参数字符串之区分空与NULL的更多相关文章

  1. 自动化CodeReview - ASP.NET Core请求参数验证

    自动化CodeReview系列目录 自动化CodeReview - ASP.NET Core依赖注入 自动化CodeReview - ASP.NET Core请求参数验证 参数验证实现 在做服务端开发 ...

  2. 如何将ASP.NET MVC所有参数均自动设置为默认

    今天看到CSDN上有个问题觉得有点意思:"可不可以ASP.NET MVC所有参数均自动设置为默认" public class HomeController : Controller ...

  3. Spring3 MVC请求参数获取的几种方法

    Spring3 MVC请求参数获取的几种方法 一.      通过@PathVariabl获取路径中的参数 @RequestMapping(value="user/{id}/{name}&q ...

  4. ASP.NET MVC传送参数至服务端

    ASP.NET MVC传送参数至服务端,前端与服务端的写法,你可以参考与采用适合你的需求的.当你只传递一两个参数也许觉得没有什么,如果一个方法中带的参数多的话,可以考虑model,前端可以考虑对象进行 ...

  5. Spring3 MVC请求参数获取的几种方法[转]

    Spring3 MVC请求参数获取的几种方法 Spring3 MVC请求参数获取的几种方法 一.      通过@PathVariabl获取路径中的参数 @RequestMapping(value=& ...

  6. Asp.net Mvc 请求是如何到达 MvcHandler的——UrlRoutingModule、MvcRouteHandler分析,并造个轮子

    这个是转载自:http://www.cnblogs.com/keyindex/archive/2012/08/11/2634005.html(那个比较容易忘记,希望博主不要生气的) 前言 本文假定读者 ...

  7. 【MVC】ASP.NET MVC 请求生命周期

    当一个asp.net mvc应用程序提出请求,为了响应请求,包含一些请求执行流程步骤! 在asp.net mvc应用程序Http request和Http response 过程中,主要包含8个步骤: ...

  8. Spring MVC请求参数绑定 自定义类型转化 和获取原声带额servlet request response信息

    首先还在我们的框架的基础上建立文件 在domian下建立Account实体类 import org.springframework.stereotype.Controller; import org. ...

  9. ASP.NET MVC显示HTML字符串

    一些html经HtmlEncode后,如“<span>测试数据</span>”.现需要把这些内容正常显示于asp.net mvc的视图内. 举个例子来解决与说明,先创建一个mo ...

随机推荐

  1. 利用vue-meta管理头部标签

    在 Vue SPA 应用中,如果想要修改HTML的头部标签,或许,你会在代码里,直接这么做 // 改下title document.title = 'what?' // 引入一段script let ...

  2. Painting The Fence(贪心+优先队列)

    Painting The Fence(贪心+优先队列) 题目大意:给 m 种数字,一共 n 个,从前往后填,相同的数字最多 k 个在一起,输出构造方案,没有则输出"-1". 解题思 ...

  3. HTML盒子模型冷知识!!!

    一.边框    1.边框颜色     border-color 边框颜色(可以设置上边框,如:border-top-color,也可以整体设置,但是要注意顺序)     2.边框粗细     bord ...

  4. 当变量超过任意设定的变量限制时终止fluent模拟【翻译】

    一些时候某个特定的变量(压力,速度,温度等)发散会造成不合理的计算结果.在许多算例,当变量超过某些合理的限制时,自动停止/打断模拟是有帮助的. 解决方法是联合UDF和scheme文件.UDF将会遍历所 ...

  5. docker容器启动后添加端口映射

    DOCKER 给运行中的容器添加映射端口 方法1 1.获得容器IP 将container_name 换成实际环境中的容器名 docker inspect `container_name` | grep ...

  6. flask 实现最简单的登录功能

    视图函数如下: # Sample.py from flask import Flask, render_template, url_for, request, redirect app = Flask ...

  7. SpringBoot——读取配置文件@Value和@Configuration比较

    1.@Configuration package com.xgcd.springboot.bean; import org.springframework.boot.context.propertie ...

  8. java里的数组和list分别在什么情况下使用?

    数组长度固定,List未限定长度,且支持的功能更多,最常用的ArrayList底层实际上也是使用数组实现. 不需要复杂功能和确定长度的情况下,使用数组效率更高,通常情况建议使用List.

  9. GIS地理处理脚本案例教程——批量栅格分割-批量栅格裁剪-批量栅格掩膜-深度学习样本批量提取

    GIS地理处理脚本案例教程--批量栅格分割-批量栅格裁剪-批量栅格掩膜-深度学习样本批量提取 商务合作,科技咨询,版权转让:向日葵,135-4855_4328,xiexiaokui#qq.com 关键 ...

  10. semi-join子查询优化 -- FirstMatch策略

    FirstMatch执行semi-join子查询的一种策略. 类似于MySQL 5.x中如何执行in.exists子查询. 让我们以搜索拥有大城市的国家为例: select * from Countr ...