4種解決json日期格式問題的辦法

 

開發中有時候須要從服務器端返回json格式的數據,在後臺代碼中若是有DateTime類型的數據使用系統自帶的工具類序列化後將獲得一個很長的數字表示日期數據,以下所示:javascript

複製代碼
           //設置服務器響應的結果爲純文本格式
            context.Response.ContentType = "text/plain";
           //學生對象集合
            List<Student> students = new List<Student>
            {
                new Student(){Name ="Tom",
                    Birthday =Convert.ToDateTime("2014-01-31 12:12:12")},
                new Student(){Name ="Rose",
                    Birthday =Convert.ToDateTime("2014-01-10 11:12:12")},
                new Student(){Name ="Mark",
                    Birthday =Convert.ToDateTime("2014-01-09 10:12:12")}
            };

            //javascript序列化器
            JavaScriptSerializer jss=new JavaScriptSerializer();
           //序列化學生集合對象獲得json字符
            string studentsJson=jss.Serialize(students);
           //將字符串響應到客戶端
            context.Response.Write(studentsJson);
           context.Response.End();
複製代碼

運行結果是:html

其中Tom所對應生日「2014-01-31」變成了1391141532000,這實際上是1970 年 1 月 1 日至今的毫秒數;1391141532000/1000/60/60/24/365=44.11年,44+1970=2014年,按這種方法能夠得出年月日時分秒和毫秒。這種格式是一種可行的表示形式但不是普通人能夠看懂的友好格式,怎麼讓這個格式變化?java

解決辦法:jquery

方法1:在服務器端將日期格式使用Select方法或LINQ表達式轉換後發到客戶端:正則表達式

複製代碼
using System;
using System.Collections.Generic;
using System.Web;

using System.Web.Script.Serialization;

namespace JsonDate1
{
    using System.Linq;

    /// <summary>
    /// 學生類,測試用
    /// </summary>
    public class Student
    {
        /// <summary>
        /// 姓名
        /// </summary>
        public String Name { get; set; }

        /// <summary>
        /// 生日
        /// </summary>
        public DateTime Birthday { get; set; }
    }

    /// <summary>
    /// 返回學生集合的json字符
    /// </summary>
    public class GetJson : IHttpHandler
    {
        public void ProcessRequest(HttpContext context)
        {
            //設置服務器響應的結果爲純文本格式
            context.Response.ContentType = "text/plain";
            //學生對象集合
            List<Student> students = new List<Student>
            {
                new Student(){Name ="Tom",Birthday =Convert.ToDateTime("2014-01-31 12:12:12")},
                new Student(){Name ="Rose",Birthday =Convert.ToDateTime("2014-01-10 11:12:12")},
                new Student(){Name ="Mark",Birthday =Convert.ToDateTime("2014-01-09 10:12:12")}
            };

            //使用Select方法從新投影對象集合將Birthday屬性轉換成一個新的屬性
            //注意屬性變化後要從新命名,並當即執行
            var studentSet =
                students.Select
                (
                p => new { p.Name, Birthday = p.Birthday.ToString("yyyy-mm-dd") }
                ).ToList();

            //javascript序列化器
            JavaScriptSerializer jss = new JavaScriptSerializer();
            //序列化學生集合對象獲得json字符
            string studentsJson = jss.Serialize(studentSet);
            //將字符串響應到客戶端
            context.Response.Write(studentsJson);
            context.Response.End();
        }

        public bool IsReusable
        {
            get
            {
                return false;
            }
        }
    }
}
複製代碼

Select方法從新投影對象集合將Birthday屬性轉換成一個新的屬性,注意屬性變化後要從新命名,屬性名能夠相同;這裏可使用select方法也可使用LINQ查詢表達式,也能夠選擇別的方式達到相同的目的;這種辦法能夠將集合中客戶端不用的屬性剔除,達到簡單優化性能的目的。json

運行結果:服務器

這時候的日期格式就已經變成友好格式了,不過在javascript中這只是一個字符串。app

方法二:ide

在javascript中將"Birthday":"\/Date(1391141532000)\/"中的字符串轉換成javascript中的日期對象,能夠將Birthday這個Key所對應的Value中的非數字字符以替換的方式刪除,到到一個數字1391141532000,而後實例化一個Date對象,將1391141532000毫秒做爲參數,獲得一個javascript中的日期對象,代碼以下:工具

複製代碼
<!DOCTYPE html>
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
    <title>json日期格式處理</title>
    <script src="Scripts/jquery-1.10.2.min.js" type="text/javascript"></script>
    <script type="text/javascript">
        $(function() {
            $.getJSON("getJson.ashx", function (students) {
                $.each(students, function (index, obj) {
                    $("<li/>").html(obj.Name).appendTo("#ulStudents");

                    //使用正則表達式將生日屬性中的非數字(\D)刪除
                    //並把獲得的毫秒數轉換成數字類型
                    var birthdayMilliseconds = parseInt(obj.Birthday.replace(/\D/igm, ""));
                    //實例化一個新的日期格式,使用1970 年 1 月 1 日至今的毫秒數爲參數
                    var birthday = new Date(birthdayMilliseconds);

                    $("<li/>").html(birthday.toLocaleString()).appendTo("#ulStudents"); ;
                });
            });
        });
    </script>
</head>
<body>
    <h2>json日期格式處理</h2>
    <ul id="ulStudents">
    </ul>
</body>
</html>
複製代碼

運行結果:

上的使用正則/\D/igm達到替換全部非數字的目的,\D表示非數字,igm是參數,分別表示忽視(ignore)大小寫;屢次、全局(global)替換;多行替換(multi-line);有一些時候還會出現+86的狀況,只須要變換正則一樣能夠達到目的。另外若是項目中反覆出現這種須要處理日期格式的問題,能夠擴展一個javascript方法,代碼以下:

複製代碼
$(function () {
            $.getJSON("getJson.ashx", function (students) {
                $.each(students, function (index, obj) {
                  $("<li/>").html(obj.Name).appendTo("#ulStudents");

                  //使用正則表達式將生日屬性中的非數字(\D)刪除
                    //並把獲得的毫秒數轉換成數字類型
                    var birthdayMilliseconds = parseInt(obj.Birthday.replace(/\D/igm, ""));
                  //實例化一個新的日期格式,使用1970 年 1 月 1 日至今的毫秒數爲參數
                    var birthday = new Date(birthdayMilliseconds);

                  $("<li/>").html(birthday.toLocaleString()).appendTo("#ulStudents");
                  $("<li/>").html(obj.Birthday.toDate()).appendTo("#ulStudents");
                });
            });
        });

        //在String對象中擴展一個toDate方法,能夠根據要求完善
        String.prototype.toDate = function () {
            var dateMilliseconds;
            if (isNaN(this)) {
                //使用正則表達式將日期屬性中的非數字(\D)刪除
                dateMilliseconds =this.replace(/\D/igm, "");
            } else {
                dateMilliseconds=this;
            }
            //實例化一個新的日期格式,使用1970 年 1 月 1 日至今的毫秒數爲參數
            return new Date(parseInt(dateMilliseconds));
        };
        
複製代碼

上面擴展的方法toDate不必定合理,也不夠強大,能夠根據須要修改。

方法三:

能夠選擇一些第三方的json工具類,其中不乏有一些已經對日期格式問題已處理好了的,常見的json序列化與反序列化工具庫有:

1.fastJSON.
2.JSON_checker.
3.Jayrock.
4.Json.NET - LINQ to JSON.
5.LitJSON.
6.JSON for .NET.
7.JsonFx.
8.JSONSharp.
9.JsonExSerializer.
10.fluent-json
11.Manatee Json 

這裏以litjson爲序列化與反序列化json的工具類做示例,代碼以下:

複製代碼
using System;
using System.Collections.Generic;
using System.Web;

using LitJson;

namespace JsonDate2
{
    using System.Linq;

    /// <summary>
    /// 學生類,測試用
    /// </summary>
    public class Student
    {
        /// <summary>
        /// 姓名
        /// </summary>
        public String Name { get; set; }

        /// <summary>
        /// 生日
        /// </summary>
        public DateTime Birthday { get; set; }
    }

    /// <summary>
    /// 返回學生集合的json字符
    /// </summary>
    public class GetJson : IHttpHandler
    {
        public void ProcessRequest(HttpContext context)
        {
            //設置服務器響應的結果爲純文本格式
            context.Response.ContentType = "text/plain";
            //學生對象集合
            List<Student> students = new List<Student>
            {
                new Student(){Name ="Tom",Birthday =Convert.ToDateTime("2014-01-31 12:12:12")},
                new Student(){Name ="Rose",Birthday =Convert.ToDateTime("2014-01-10 11:12:12")},
                new Student(){Name ="Mark",Birthday =Convert.ToDateTime("2014-01-09 10:12:12")}
            };

            //序列化學生集合對象獲得json字符
            string studentsJson = JsonMapper.ToJson(students);
            //將字符串響應到客戶端
            context.Response.Write(studentsJson);
            context.Response.End();
        }

        public bool IsReusable
        {
            get
            {
                return false;
            }
        }
    }
}
複製代碼

運行結果以下:

這時候的日期格式就基本正確了,只要在javascript中直接實例化日期就行了,

var date = new Date("01/31/2014 12:12:12");
alert(date.toLocaleString());

客戶端的代碼以下:

複製代碼
$(function () {
            $.getJSON("GetJson2.ashx", function (students) {
                $.each(students, function (index, obj) {
                    $("<li/>").html(obj.Name).appendTo("#ulStudents");

                    var birthday = new Date(obj.Birthday);
                    $("<li/>").html(birthday.toLocaleString()).appendTo("#ulStudents");
                });
            });
        });

        var date = new Date("01/31/2014 12:12:12");
        alert(date.toLocaleString());
複製代碼

方法四:

這點文字發到博客上有網友提出了他們寶貴的意見,我並無考慮在MVC中的狀況,其實MVC中也可使用handler,因此區別不是很大了,但MVC中有專門針對服務器響應爲JSON的Action,代碼以下:

複製代碼
using System;
using System.Web.Mvc;

namespace JSONDateMVC.Controllers
{
    public class HomeController : Controller
    {
        public JsonResult GetJson1()
        {
            //序列化當前日期與時間對象,並容許客戶端Get請求
            return Json(DateTime.Now, JsonRequestBehavior.AllowGet);
        }
    }
}
複製代碼

運行結果:

下載一個內容爲Application/json的文件,文件名爲GetJson1,內容是"\/Date(1391418272884)\/"

從上面的狀況看來MVC中序列化時並未對日期格式特別處理,咱們能夠反編譯看源碼:

Return調用的Json方法:

protected internal JsonResult Json(object data, JsonRequestBehavior behavior)
{
    return this.Json(data, null, null, behavior);
}

this.Json方法

複製代碼
protected internal virtual JsonResult Json(object data, string contentType, Encoding contentEncoding, JsonRequestBehavior behavior)
{
    return new JsonResult { Data = data, ContentType = contentType, ContentEncoding = contentEncoding, JsonRequestBehavior = behavior };
}
複製代碼

JsonResult類ActionResult類的子類,ExecuteResult方法:

從上面的代碼中不難看出微軟的JsonResult類仍然是使用了JavaScriptSerializer,因此返回的結果與方法一未處理時是同樣的,要解決這個問題咱們能夠派生出一個新的類,重寫ExecuteResult方法,使用Json.net來完成序列化工做,JsonResultPro.cs文件的代碼以下:

複製代碼
namespace JSONDateMVC.Common
{
    using System;
    using System.Web;

    using System.Web.Mvc;
    using Newtonsoft.Json;
    using Newtonsoft.Json.Converters;

    public class JsonResultPro : JsonResult
    {
        public JsonResultPro(){}
        public JsonResultPro(object data, JsonRequestBehavior behavior)
        {
            base.Data = data;
            base.JsonRequestBehavior = behavior;
            this.DateTimeFormat = "yyyy-MM-dd hh:mm:ss";
        }
        public JsonResultPro(object data, String dateTimeFormat)
        {
            base.Data = data;
            base.JsonRequestBehavior = JsonRequestBehavior.AllowGet;
            this.DateTimeFormat = dateTimeFormat;
        }

        /// <summary>
        /// 日期格式
        /// </summary>
        public string DateTimeFormat{ get; set; }

        public override void ExecuteResult(ControllerContext context)
        {
            if (context == null)
            {
                throw new ArgumentNullException("context");
            }
            if ((this.JsonRequestBehavior == JsonRequestBehavior.DenyGet) && string.Equals(context.HttpContext.Request.HttpMethod, "GET", StringComparison.OrdinalIgnoreCase))
            {                
                throw new InvalidOperationException("MvcResources.JsonRequest_GetNotAllowed");
            }
            HttpResponseBase base2 = context.HttpContext.Response;
            if (!string.IsNullOrEmpty(this.ContentType))
            {
                base2.ContentType = this.ContentType;
            }
            else
            {
                base2.ContentType = "application/json";
            }
            if (this.ContentEncoding != null)
            {
                base2.ContentEncoding = this.ContentEncoding;
            }
            if (this.Data != null)
            {
                //轉換System.DateTime的日期格式到 ISO 8601日期格式
                //ISO 8601 (如2008-04-12T12:53Z)
                IsoDateTimeConverter isoDateTimeConverter=new IsoDateTimeConverter();
                //設置日期格式
                isoDateTimeConverter.DateTimeFormat = DateTimeFormat;
                //序列化
                String jsonResult = JsonConvert.SerializeObject(this.Data,isoDateTimeConverter);
                //相應結果
                base2.Write(jsonResult);
            }

        }
    }
}
複製代碼

使用上面的JsonResultPro Action類型的代碼以下:

複製代碼
        public JsonResultPro GetJson2()
        {
            //序列化當前日期與時間對象,並容許客戶端Get請求,注意H是大寫
            return new JsonResultPro(DateTime.Now,"yyyy-MM-dd HH:mm");
        }
複製代碼

運行結果:

"2014-02-03 18:10"

這樣就能夠徹底按本身的意思來設置日期格式了,但須要注意日期格式如平時的Format是有區別的,如這裏表示時間的H若是大寫表示24小時制,若是小寫表示12小時制。另外還有幾個問題要問你們:

一、經過Reflector反編譯獲得的代碼中有不少變化,如屬性會變成get_Request()方法的形式,不知道你們有沒有更好的方法。

二、在反編譯獲得的代碼中使用到了資源文件MvcResources.JsonRequest_GetNotAllowed,怎麼在重寫時也可使用?

這裏講到了幾種解決json中序列化後的日期格式問題,應該還有更好更完善的方法,歡迎您告訴我。由於有不少學生問我因此我寫了這點文字,歡迎批評指正。

轉自:張果老師 原文地址: http://www.cnblogs.com/best/p/3537030.html
相關文章
相關標籤/搜索