ASP.NET MVC请求参数字符串之区分空与NULL

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

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

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

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。

posted @ 2018-01-18 13:29  coloc  阅读(2444)  评论(8编辑  收藏  举报