在.Net的MVC开发中,经常会使用到Json对象,于是,系统提供了JsonResult这个对象,其本质是调用.Net系统自带的Json序列化类JavaScriptSerializer对数据对象进行序列化。但是这个系统自带的Json序列化对象方法没有Json.Net好用,于是打算有些时候用Json.Net替代默认的实现。
要实现有时候用Json.Net,有时候用默认实现,那么就要保证系统中两种实现并存。对于Server将对象序列化成Json传给Client很简单,我们只需要建立一个新的ActionResult,我们命名为JsonNetResult,然后在Get时,return这个JsonNetResult即可。JsonNetResult的代码实现为:
System;
System.Collections.Generic;
System.Linq;
System.Text;
<span class="kwrd">namespace MvcJsonNet
{
<span class="kwrd">using System.IO;
<span class="kwrd">using System.Web;
<span class="kwrd">using System.Web.Mvc;
<span class="kwrd">using Newtonsoft.Json;
<span class="kwrd">public</span> <span class="kwrd">class</span> JsonNetResult : JsonResult
{
<span class="kwrd">public</span> JsonNetResult()
{
Settings = <span class="kwrd">new</span> JsonSerializerSettings
{
ReferenceLoopHandling = ReferenceLoopHandling.Error
};
}
<span class="kwrd">public</span> JsonNetResult(<span class="kwrd">object</span> data,JsonRequestBehavior behavior = JsonRequestBehavior.AllowGet,<span class="kwrd">string</span> contentType=<span class="kwrd">null</span>,Encoding contentEncoding=<span class="kwrd">null</span>)
{
<span class="kwrd">this</span>.Data = data;
<span class="kwrd">this</span>.JsonRequestBehavior = behavior;
<span class="kwrd">this</span>.ContentEncoding = contentEncoding;
<span class="kwrd">this</span>.ContentType = contentType;
}
<span class="kwrd">public</span> JsonSerializerSettings Settings { get; <span class="kwrd">private</span> set; }
<span class="kwrd">public</span> <span class="kwrd">override</span> <span class="kwrd">void</span> ExecuteResult(ControllerContext context)
{
<span class="kwrd">if</span> (context == <span class="kwrd">null</span>)
<span class="kwrd">throw</span> <span class="kwrd">new</span> ArgumentNullException(<span class="str">"context"</span>);
<span class="kwrd">if</span> (<span class="kwrd">this</span>.JsonRequestBehavior == JsonRequestBehavior.DenyGet && <span class="kwrd">string</span>.Equals(context.HttpContext.Request.HttpMethod,<span class="str">"GET"</span>,StringComparison.OrdinalIgnoreCase))
<span class="kwrd">throw</span> <span class="kwrd">new</span> InvalidOperationException(<span class="str">"JSON GET is not allowed"</span>);
HttpResponseBase response = context.HttpContext.Response;
response.ContentType = <span class="kwrd">string</span>.IsNullOrEmpty(<span class="kwrd">this</span>.ContentType) ? <span class="str">"application/json"</span> : <span class="kwrd">this</span>.ContentType;
<span class="kwrd">if</span> (<span class="kwrd">this</span>.ContentEncoding != <span class="kwrd">null</span>)
response.ContentEncoding = <span class="kwrd">this</span>.ContentEncoding;
<span class="kwrd">if</span> (<span class="kwrd">this</span>.Data == <span class="kwrd">null</span>)
<span class="kwrd">return</span>;
var scriptSerializer = JsonSerializer.Create(<span class="kwrd">this</span>.Settings);
<span class="kwrd">using</span> (var sw = <span class="kwrd">new</span> StringWriter())
{
scriptSerializer.Serialize(sw,<span class="kwrd">this</span>.Data);
response.Write(sw.ToString());
}
}
}
}
要返回一个Json.Net序号列后的对象,那么调用方法是:
ActionResult GetJsonNet()
{
var myClass = InitClass();
JsonNetResult(myClass);
}
这是Get方法,但是对于ClientPost一个Json回Server,那么就比较麻烦了,需要修改好几处地方:
1,建立Json.Net的ValueProviderFactory,这个类主要就是用于Json字符串的反序列化。
System;
System.Collections.Generic;
System.Linq;
System.Web;
<span class="kwrd">namespace MvcJsonNet
{
<span class="kwrd">using System.Collections;
<span class="kwrd">using System.Dynamic;
<span class="kwrd">using System.Globalization;
<span class="kwrd">using System.IO;
<span class="kwrd">using System.Web.Mvc;
<span class="kwrd">using System.Web.Script.Serialization;
<span class="kwrd">using Newtonsoft.Json;
<span class="kwrd">public</span> <span class="kwrd">class</span> JsonNetValueProviderFactory : ValueProviderFactory
{
<span class="kwrd">private</span> <span class="kwrd">void</span> AddToBackingStore(Dictionary<<span class="kwrd">string</span>,<span class="kwrd">object</span>> backingStore,<span class="kwrd">string</span> prefix,<span class="kwrd">object</span> <span class="kwrd">value</span>)
{
IDictionary<<span class="kwrd">string</span>,<span class="kwrd">object</span>> d = <span class="kwrd">value</span> <span class="kwrd">as</span> IDictionary<<span class="kwrd">string</span>,<span class="kwrd">object</span>>;
<span class="kwrd">if</span> (d != <span class="kwrd">null</span>)
{
<span class="kwrd">foreach</span> (KeyValuePair<<span class="kwrd">string</span>,<span class="kwrd">object</span>> entry <span class="kwrd">in</span> d)
{
AddToBackingStore(backingStore,MakePropertyKey(prefix,entry.Key),entry.Value);
}
<span class="kwrd">return</span>;
}
IList l = <span class="kwrd">value</span> <span class="kwrd">as</span> IList;
<span class="kwrd">if</span> (l != <span class="kwrd">null</span>)
{
<span class="kwrd">for</span> (<span class="kwrd">int</span> i = 0; i < l.Count; i++)
{
AddToBackingStore(backingStore,MakeArrayKey(prefix,i),l[i]);
}
<span class="kwrd">return</span>;
}
<span class="rem">// primitive</span>
backingStore[prefix] = <span class="kwrd">value</span>;
}
<span class="kwrd">private</span> <span class="kwrd">object</span> GetDeserializedObject(ControllerContext controllerContext)
{
<span class="kwrd">if</span> (!controllerContext.HttpContext.Request.ContentType.StartsWith(<span class="str">"application/json"</span>,StringComparison.InvariantCultureIgnoreCase))
{
<span class="rem">// not JSON request</span>
<span class="kwrd">return</span> <span class="kwrd">null</span>;
}
StreamReader reader = <span class="kwrd">new</span> StreamReader(controllerContext.HttpContext.Request.InputStream);
<span class="kwrd">string</span> bodyText = reader.ReadToEnd();
<span class="kwrd">if</span> (String.IsNullOrEmpty(bodyText))
{
<span class="rem">// no JSON data</span>
<span class="kwrd">return</span> <span class="kwrd">null</span>;
}
<span class="rem">//接下来的代码是关键,判断content type,如果是json.net,那么就使用Json.Net的反序列化方法,如果不是,那么就使用系统默认的反序列化方法</span>
<span class="kwrd">if</span> (controllerContext.HttpContext.Request.ContentType.StartsWith(<span class="str">"application/json.net"</span>,StringComparison.InvariantCultureIgnoreCase))
{
var jsonData = JsonConvert.DeserializeObject<ExpandoObject>(bodyText);
<span class="kwrd">return</span> jsonData;
}
<span class="kwrd">else</span>
{
JavaScriptSerializer serializer = <span class="kwrd">new</span> JavaScriptSerializer();
<span class="kwrd">object</span> jsonData = serializer.DeserializeObject(bodyText);
<span class="kwrd">return</span> jsonData;
}
}
<span class="kwrd">public</span> <span class="kwrd">override</span> IValueProvider GetValueProvider(ControllerContext controllerContext)
{
<span class="kwrd">if</span> (controllerContext == <span class="kwrd">null</span>)
{
<span class="kwrd">throw</span> <span class="kwrd">new</span> ArgumentNullException(<span class="str">"controllerContext"</span>);
}
<span class="kwrd">object</span> jsonData = GetDeserializedObject(controllerContext);
<span class="kwrd">if</span> (jsonData == <span class="kwrd">null</span>)
{
<span class="kwrd">return</span> <span class="kwrd">null</span>;
}
Dictionary<<span class="kwrd">string</span>,<span class="kwrd">object</span>> backingStore = <span class="kwrd">new</span> Dictionary<<span class="kwrd">string</span>,<span class="kwrd">object</span>>(StringComparer.OrdinalIgnoreCase);
AddToBackingStore(backingStore,String.Empty,jsonData);
<span class="kwrd">return</span> <span class="kwrd">new</span> DictionaryValueProvider<<span class="kwrd">object</span>>(backingStore,CultureInfo.CurrentCulture);
}
<span class="kwrd">private</span> <span class="kwrd">string</span> MakeArrayKey(<span class="kwrd">string</span> prefix,<span class="kwrd">int</span> index)
{
<span class="kwrd">return</span> prefix + <span class="str">"["</span> + index.ToString(CultureInfo.InvariantCulture) + <span class="str">"]"</span>;
}
<span class="kwrd">private</span> <span class="kwrd">string</span> MakePropertyKey(<span class="kwrd">string</span> prefix,<span class="kwrd">string</span> propertyName)
{
<span class="kwrd">return</span> (String.IsNullOrEmpty(prefix)) ? propertyName : prefix + <span class="str">"."</span> + propertyName;
}
}
}
2,在初始化MVC时替换掉默认的JsonValueProviderFactory。 在Global.asax的Application_Start时,写入以下代码:
().FirstOrDefault());
ValueProviderFactories.Factories.Add( JsonNetValueProviderFactory());
3,建立新的ModelBinder,命名为JsonNetModelBinder。
MvcJsonNet
{
System;
System.ComponentModel;
System.Diagnostics;
System.Globalization;
System.Linq;
System.Web.Mvc;
Newtonsoft.Json;
<span class="kwrd">public</span> <span class="kwrd">class</span> JsonNetModelBinder : DefaultModelBinder
{
<span class="kwrd">protected</span> <span class="kwrd">override</span> <span class="kwrd">void</span> BindProperty(ControllerContext controllerContext,ModelBindingContext bindingContext,PropertyDescriptor propertyDescriptor)
{
Debug.WriteLine(<span class="str">"BindProperty"</span>);
<span class="kwrd">if</span> (!controllerContext.HttpContext.Request.ContentType.StartsWith(<span class="str">"application/json.net"</span>,StringComparison
.InvariantCultureIgnoreCase))
{
<span class="rem">//根据Content type来判断,只有json.net这种content type的才会使用该ModelBinder,否则使用默认的Binder</span>
<span class="kwrd">base</span>.BindProperty(controllerContext,bindingContext,propertyDescriptor);
<span class="kwrd">return</span>;
}
<span class="rem">// need to skip properties that aren't part of the request,else we might hit a StackOverflowException</span>
<span class="kwrd">string</span> name = propertyDescriptor.Name;
<span class="kwrd">foreach</span> (<span class="kwrd">object</span> attribute <span class="kwrd">in</span> propertyDescriptor.Attributes)
{
<span class="kwrd">if</span> (attribute <span class="kwrd">is</span> JsonPropertyAttribute)
{
var jp = attribute <span class="kwrd">as</span> JsonPropertyAttribute;
name = jp.PropertyName;
}
}
<span class="kwrd">string</span> fullPropertyKey = CreateSubPropertyName(bindingContext.ModelName,name);
<span class="kwrd">if</span> (!bindingContext.ValueProvider.ContainsPrefix(fullPropertyKey))
{
<span class="kwrd">return</span>;
}
<span class="rem">// call into the property's model binder</span>
IModelBinder propertyBinder = Binders.GetBinder(propertyDescriptor.PropertyType);
<span class="kwrd">object</span> originalPropertyValue = propertyDescriptor.GetValue(bindingContext.Model);
ModelMetadata propertyMetadata = bindingContext.PropertyMetadata[propertyDescriptor.Name];
propertyMetadata.Model = originalPropertyValue;
var innerBindingContext = <span class="kwrd">new</span> ModelBindingContext
{
ModelMetadata = propertyMetadata,ModelName = fullPropertyKey,ModelState = bindingContext.ModelState,ValueProvider = bindingContext.ValueProvider
};
<span class="kwrd">object</span> newPropertyValue = GetPropertyValue(controllerContext,innerBindingContext,propertyDescriptor,propertyBinder);
propertyMetadata.Model = newPropertyValue;
<span class="rem">// validation</span>
ModelState modelState = bindingContext.ModelState[fullPropertyKey];
<span class="kwrd">if</span> (modelState == <span class="kwrd">null</span> || modelState.Errors.Count == 0)
{
<span class="kwrd">if</span> (OnPropertyValidating(controllerContext,newPropertyValue))
{
SetProperty(controllerContext,newPropertyValue);
OnPropertyValidated(controllerContext,newPropertyValue);
}
}
<span class="kwrd">else</span>
{
SetProperty(controllerContext,newPropertyValue);
<span class="rem">// Convert FormatExceptions (type conversion failures) into InvalidValue messages</span>
<span class="kwrd">foreach</span> (
ModelError error <span class="kwrd">in</span>
modelState.Errors.Where(err => String.IsNullOrEmpty(err.ErrorMessage) && err.Exception != <span class="kwrd">null</span>)
.ToList())
{
<span class="kwrd">for</span> (Exception exception = error.Exception; exception != <span class="kwrd">null</span>; exception = exception.InnerException)
{
<span class="kwrd">if</span> (exception <span class="kwrd">is</span> FormatException)
{
<span class="kwrd">string</span> displayName = propertyMetadata.GetDisplayName();
<span class="kwrd">string</span> errorMessageTemplate = <span class="str">"The value '{0}' is not valid for {1}."</span>;
<span class="kwrd">string</span> errorMessage = String.Format(CultureInfo.CurrentCulture,errorMessageTemplate,modelState.Value.AttemptedValue,displayName);
modelState.Errors.Remove(error);
modelState.Errors.Add(errorMessage);
<span class="kwrd">break</span>;
}
}
}
}
}
}
}
4,建立一个VModel的基类,为该基类添加Attribute,然后在Global中添加Model和Binder的映射。
(JsonNetModelBinder))]
VEntity
{
Id { get; set; }
}
Global.asax中Application_Start添加代码:
(VEntity), JsonNetModelBinder());
5在前端Post Json时,指定content type为application/json.net
PostJsonNet() {
jsonstr = $()[0].innerHTML;
$.ajax({
url: ,type: ,data: jsonstr,contentType: ,dataType: ,success: (data) {
alert(data);
}
});
}</pre>
我们这样处理后,Client在往Server传送Json数据时,如果指定了contentType是application/json,那么就使用系统默认的方法来反序列化对象,如果是application/json.net,那么就使用Json.Net来反序列化。
(编辑:李大同)
【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!
|