C# ASP.net MVC JsonValueProvider字节类型问题

C# ASP.net MVC JsonValueProvider字节类型问题,c#,asp.net-mvc-3,value-provider,C#,Asp.net Mvc 3,Value Provider,我正在将JSON发布到一个控制器方法中,当ASP.NETMVC3中包含的JsonValueProvider试图将JSON数字转换为字节类型时,它似乎已损坏 基本上,需要转换为字节的任何非null值都是null,并且我的模型状态有错误,没有错误消息。如返回JSON中的问题列表所示。dnbscore和region有错误,但评级没有错误。返回的数据将dnbscore和region设置为null,即使在传递JSON时,这些字段都有值。我可以使用自定义模型绑定器来处理这个问题,但我想知道是否有其他方法可以

我正在将JSON发布到一个控制器方法中,当ASP.NETMVC3中包含的JsonValueProvider试图将JSON数字转换为字节类型时,它似乎已损坏

基本上,需要转换为字节的任何非null值都是null,并且我的模型状态有错误,没有错误消息。如返回JSON中的问题列表所示。dnbscore和region有错误,但评级没有错误。返回的数据将dnbscore和region设置为null,即使在传递JSON时,这些字段都有值。我可以使用自定义模型绑定器来处理这个问题,但我想知道是否有其他方法可以让JsonValueProider像我期望的那样工作

这就是我试图构建的对象:

public class APICustomerDetailsDTO
{
    public int customerid { get; set; }

    [StringLength(30)]
    public string customername { get; set; }
    public decimal? globalcredit { get; set; }
    [Range(0,5)]
    public byte? rating { get; set; }
    [Range(0, 100)]
    public byte? dnbscore { get; set; }
    [Range(0, 8)]
    public byte? region { get; set; }
    [Required]
    public bool isnotactive { get; set; }
    public string salesperson { get; set; }
    [StringLength(30)]
    public string newmill_venderid { get; set; }
    [StringLength(30)]
    public string newmill_supplierid { get; set; }
    [StringLength(30)]
    public string edi_receiverid { get; set; }
    [Required]
    public bool edi_invoice { get; set; }
    [StringLength(15)]
    public string bill_code { get; set; }
}
这是我在post请求中发送的JSON:

{"bill_code":"good","customerid":50,"customername":"Ted","dnbscore":80,"edi_invoice":false,"edi_receiverid":null,"globalcredit":null,"isnotactive":false,"newmill_supplierid":null,"newmill_venderid":null,"rating":null,"region":0,"salesperson":null}
检查模型状态的方法的一部分:

        if (!ModelState.IsValid)
        {
            var issues = ModelState.Where(m => m.Value.Errors.Any())
                                   .Select((m)=> new {field = m.Key, error = m.Value.Errors.FirstOrDefault().ErrorMessage})
                                   .ToArray();

            var result = new
            {
                result = "Failure",
                message = "Invalid data received. See issues for details.",
                issues = issues,
                data = cust
            };
返回JSON:

{"result":"Failure","message":"Invalid data received. See issues for details.","issues":[{"field":"dnbscore","error":""},{"field":"region","error":""}],"data":{"customerid":50,"customername":"Ted","globalcredit":null,"rating":null,"dnbscore":null,"region":null,"isnotactive":false,"salesperson":null,"newmill_venderid":null,"newmill_supplierid":null,"edi_receiverid":null,"edi_invoice":false,"bill_code":"good"}}
为了完整起见,以下是我为解决这个问题所做的:

根据Darin提供的信息,我意识到这是一个比字节更大的问题?到int?转换。这对于默认绑定器提供的任何非int转换都是一个问题。因此,我制作了一个自定义活页夹,它可以很好地处理字节、小数等数字。见下文:

public class APICustomerDetailsDTOBinder : IModelBinder
{
    public object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
    {
        APICustomerDetailsDTO model = (APICustomerDetailsDTO)bindingContext.Model ??
            (APICustomerDetailsDTO)DependencyResolver.Current.GetService(typeof(APICustomerDetailsDTO));

        bool hasPrefix = bindingContext.ValueProvider.ContainsPrefix(bindingContext.ModelName);
        string searchPrefix = (hasPrefix) ? bindingContext.ModelName + "." : "";

        int customerid = 0;
        int.TryParse(GetValue(bindingContext, searchPrefix, "customerid"), out customerid);
        model.customerid = customerid;

        string customername = GetValue(bindingContext, searchPrefix, "customername");
        if (!String.IsNullOrEmpty(customername))
        { model.customername = customername; }
        else
        { model.customername = null; }

        decimal globalcredit;
        if (decimal.TryParse(GetValue(bindingContext, searchPrefix, "globalcredit"), out globalcredit))
        { model.globalcredit = globalcredit; }
        else
        { model.globalcredit = null; }

        byte rating;
        if (byte.TryParse(GetValue(bindingContext, searchPrefix, "rating"), out rating))
        { model.rating = rating; }
        else
        { model.rating = null; }

        byte dnbscore;
        if (byte.TryParse(GetValue(bindingContext, searchPrefix, "dnbscore"), out dnbscore))
        { model.dnbscore = dnbscore; }
        else
        { model.dnbscore = null; }

        byte region;
        if (byte.TryParse(GetValue(bindingContext, searchPrefix, "region"), out region))
        { model.region = region; }
        else
        { model.region = null; }

        bool isnotactive;
        if (bool.TryParse(GetValue(bindingContext, searchPrefix, "isnotactive"), out isnotactive))
        { model.isnotactive = isnotactive; }
        else
        { model.isnotactive = false; }

        string salesperson = GetValue(bindingContext, searchPrefix, "salesperson");
        if (!String.IsNullOrEmpty(salesperson))
        { model.salesperson = salesperson; }
        else
        { model.salesperson = null; }

        string newmill_venderid = GetValue(bindingContext, searchPrefix, "newmill_venderid");
        if (!String.IsNullOrEmpty(newmill_venderid))
        { model.newmill_venderid = newmill_venderid; }
        else
        { model.newmill_venderid = null; }

        string newmill_supplierid = GetValue(bindingContext, searchPrefix, "newmill_supplierid");
        if (!String.IsNullOrEmpty(newmill_supplierid))
        { model.newmill_supplierid = newmill_supplierid; }
        else
        { model.newmill_supplierid = null; }

        string edi_receiverid = GetValue(bindingContext, searchPrefix, "edi_receiverid");
        if (!String.IsNullOrEmpty(edi_receiverid))
        { model.edi_receiverid = edi_receiverid; }
        else
        { model.edi_receiverid = null; }


        bool edi_invoice;
        if (bool.TryParse(GetValue(bindingContext, searchPrefix, "edi_invoice"), out edi_invoice))
        { model.edi_invoice = edi_invoice; }
        else
        { model.edi_invoice = false; }

        model.bill_code = GetValue(bindingContext, searchPrefix, "bill_code");

        return model;
    }

    private string GetValue(ModelBindingContext context, string prefix, string key)
    {
        ValueProviderResult vpr = context.ValueProvider.GetValue(prefix + key);
        return vpr == null ? null : vpr.AttemptedValue;
    }

    private bool GetCheckedValue(ModelBindingContext context, string prefix, string key)
    {
        bool result = false;
        ValueProviderResult vpr = context.ValueProvider.GetValue(prefix + key);
        if (vpr != null)
        {
            result = (bool)vpr.ConvertTo(typeof(bool));
        }
        return result;
    }
}

您可以将它们作为字符串发送:

{ 
    "bill_code": "good", 
    "customerid": 50, 
    "customername": "Ted", 
    "dnbscore": "80", 
    "edi_invoice": false, 
    "edi_receiverid": null, 
    "globalcredit": null, 
    "isnotactive": false, 
    "newmill_supplierid": null, 
    "newmill_venderid": null, 
    "rating": null, 
    "region": "0", 
    "salesperson": null 
}

如果您对更多详细信息感兴趣,可以查看。

我的自制解决方案。这是为了创建一个javascript函数,确保所有数值都用引号进行序列化,如下所示

$.toJson = function (data) {
    return JSON.stringify(param, function (k, v) { return typeof v === 'number' ? v.toString() : v; });
};

这个JSON序列化问题正是我的问题所在。当解析方法获取字符串值时,使用字符串是有意义的。我可以控制我发布到页面上的数据,但是当我开始将这些值转换为JSON时,有没有办法确保这些值是字符串?你让我意识到这个问题不仅仅是字节问题?我认为这是有限的领域。非常感谢。