本文 基於 JQuery 擴展了一些 JS日期函數,包括: > 字元串 轉 Date 對象 萬能函數(性能僅 10W次/s,函數有路徑優化,字元串越詭異 耗時越長) > Date 轉 字元串 格式化 > 兩個 Date 的差值 (返回的結果類似 C# TimeSpan 對象) ...
背景:
JS 有自己的 時間類型 Date —— 但是,在某些情況下 這個對象似乎 不太好用。
本文 基於 JQuery 擴展了一些 JS日期函數,包括:
> 字元串 轉 Date 對象 萬能函數(性能僅 10W次/s,函數有路徑優化,字元串越詭異 耗時越長)
> Date 轉 字元串 格式化
> 兩個 Date 的差值 (返回的結果類似 C# TimeSpan 對象)
一言不合,直接上代碼:
1 /*感謝 InkFx (C) http://www.ink-fx.com 為 以下日期函數 作出的努力 */ 2 jQuery = window.jQuery || { }; 3 4 (function($) { 5 6 7 /*--數據類型轉換函數 Start----------------------------------*/ 8 9 $.isArray = function(obj) {var result = false;try {result = Object.prototype.toString.apply(obj) === '[object Array]';if (result) return true;} catch(e) {}try {result = obj.constructor === Array;if (result) return true;} catch(e) {}return false;}; 10 $.toInt = function(obj, dft) {try {dft = (typeof dft === 'number') ? dft : 0;} catch(e) {dft = 0;}try {if (obj == null) return dft;if (typeof obj === 'number') return parseInt(obj);var match = obj.toString().toLowerCase().match(/-*[0123456789abcdefx.]+/);var str = match == null || match.length <= 0 ? null : match[0];var result = null;if (str != null && str.length >= 2 && str.substr(0, 2) == "0x") {str = str.substr(2);result = parseInt(str, 16);} else {result = parseInt(str);}if (result == null || !isFinite(result)) return dft;return result;} catch(e) {return dft;}}; 11 $.toFloat = function(obj, dft) {try {dft = (typeof dft === 'number') ? dft : 0;} catch(e) {dft = 0;}try {if (obj == null) return dft;if (typeof obj === 'number') return parseFloat(obj);var match = obj.toString().toLowerCase().match(/-*[0123456789abcdefx.]+/);var str = match == null || match.length <= 0 ? null : match[0];var result = null;if (str != null && str.length >= 2 && str.substr(0, 2) == "0x") {str = str.substr(2);result = parseFloat(parseInt(str, 16));} else {result = parseFloat(str);}if (result == null || !isFinite(result)) return dft;return result;} catch(e) {return dft;}}; 12 $.toString = function(obj, dft) {try {dft = (typeof dft === 'string') ? dft : '';} catch(e) {dft = '';}try {if (obj == null) return dft;if (typeof obj === 'string') return obj;return obj.toString();} catch(e) {return dft;}}; 13 $.toDateTime = function(obj, dft) {try {dft = (dft instanceof Date) ? dft : new Date(1900, 00, 01);} catch(e) {dft = new Date(1900, 00, 01);}try {if (obj == null) return dft;if (obj instanceof Date) return obj;var result = parseDate(obj);if (result == null) return dft;return result;} catch(e) {return dft;}}; 14 $.toBoolean = function(obj, dft) {try {dft = (typeof dft === 'boolean') ? dft : false;} catch(e) {dft = false;}try {if (obj == null) return dft;if (obj == true) return true;if (obj == false) return false;if ($.isArray(obj)) return obj.length >= 1;if (typeof obj === 'boolean') return obj.toString().toLowerCase() == "true";if (typeof obj === 'number') return parseFloat(obj) > 0;/*if (typeof obj === 'string') return parseFloat(obj) > 0;*/if (obj.toString().toLowerCase() == "t") return true;if (obj.toString().toLowerCase() == "true") return true;return false;} catch(e) {return dft;}}; 15 16 17 /*--數據類型轉換函數 End------------------------------------*/ 18 19 20 21 /*--時間相關函數 Start--------------------------------------*/ 22 23 $.formatDate = function(date, fmt) {try {date = $.toDateTime(date);fmt = fmt || "yyyy-MM-dd HH:mm:ss"; o = {"M+": date.getMonth() + 1,"d+": date.getDate(),"H+": date.getHours(),"h+": date.getHours() /*(date.getHours() > 12 ? (date.getHours() - 12) : date.getHours())*/,"m+": date.getMinutes(),"s+": date.getSeconds(),"q+": Math.floor((date.getMonth() + 3) / 3),"S": date.getMilliseconds(),"f+": date.getMilliseconds()};if (/(y+)/.test(fmt)) fmt = fmt.replace(RegExp.$1, (date.getFullYear() + "").substr(4 - RegExp.$1.length));for (var k in o)if (new RegExp("(" + k + ")").test(fmt)) {var padLeft = (k == "S" || k == "f+") ? "000" : "00";fmt = fmt.replace(RegExp.$1, RegExp.$1.length == 1 ? o[k] : (padLeft + o[k]).substr(("" + o[k]).length));}return fmt;} catch(e) { return ""; }}; 24 $.isLeapYearDate = function(date) {try {date = $.toDateTime(date);return (0 == date.getFullYear() % 4 && ((date.getFullYear() % 100 != 0) || (date.getFullYear() % 400 == 0)));} catch(e) {return false;}}; 25 26 $.addYears = function(date, num) {try {date = $.toDateTime(date);num = $.toInt(num);var year = date.getFullYear();var month = date.getMonth();var day = date.getDate();var hour = date.getHours();var minute = date.getMinutes();var second = date.getSeconds();var millisecond = date.getMilliseconds();var result = new Date(year + num, month, day, hour, minute, second, millisecond);return result;} catch(e) { return new Date(1900, 01, 01); }}; 27 $.addMonths = function(date, num) {try {date = $.toDateTime(date);num = $.toInt(num);var year = date.getFullYear();var month = date.getMonth();var day = date.getDate();var hour = date.getHours();var minute = date.getMinutes();var second = date.getSeconds();var millisecond = date.getMilliseconds();var addYear = (num + month) / 12;month = (num + month) % 12;var result = new Date(year + addYear, month, day, hour, minute, second, millisecond);return result;} catch(e) { return new Date(1900, 01, 01); }}; 28 $.addDays = function(date, num) {try {date = $.toDateTime(date);num = $.toInt(num);var result = new Date(date.valueOf() + (num * 1000 * 60 * 60 * 24));return result;} catch(e) { return new Date(1900, 01, 01); }}; 29 $.addHours = function(date, num) {try {date = $.toDateTime(date);num = $.toInt(num);var result = new Date(date.valueOf() + (num * 1000 * 60 * 60));return result;} catch(e) { return new Date(1900, 01, 01); }}; 30 $.addMinutes = function(date, num) {try {date = $.toDateTime(date);num = $.toInt(num);var result = new Date(date.valueOf() + (num * 1000 * 60));return result;} catch(e) { return new Date(1900, 01, 01); }}; 31 $.addSeconds = function(date, num) {try {date = $.toDateTime(date);num = $.toInt(num);var result = new Date(date.valueOf() + (num * 1000));return result;} catch(e) { return new Date(1900, 01, 01); }}; 32 $.addMilliseconds = function(date, num) {try {date = $.toDateTime(date);num = $.toInt(num);var result = new Date(date.valueOf() + num);return result;} catch(e) { return new Date(1900, 01, 01); }}; 33 $.getTimeSpan = function(date1, date2) {var timestamp = 0;if((date1 instanceof Date || date2 instanceof Date || date2 == null) || (date1 != null && date2 != null)) {date1 = $.toDateTime(date1);date2 = $.toDateTime(date2);timestamp = $.toFloat(date1 - date2);} else {timestamp = $.toInt(date1);}var temp = $.toInt(timestamp / 1000);var totalMilliSecond = timestamp;var totalSecond = timestamp / (1000);var totalMinute = timestamp / (1000 * 60);var totalHour = timestamp / (1000 * 60 * 60);var totalDay = timestamp / (1000 * 60 * 60 * 24);var milliSecond = $.toInt(totalMilliSecond) % 1000;var second = $.toInt(totalSecond) % 60;var minute = $.toInt(totalMinute) % 60;var hour = $.toInt(totalHour) % 24;var day = $.toInt(totalDay);return {TimeStamp: timestamp,Days: day,Hours: hour,Minutes: minute,Seconds: second,Milliseconds: milliSecond,TotalDays: totalDay,TotalHours: totalHour,TotalMinutes: totalMinute,TotalSeconds: totalSecond,TotalMilliseconds: totalMilliSecond,valueOf: function() { return this.TimeStamp; },toString: function() { return this.Days + "." + $.padLeft(this.Hours, 2, "0") + ":" + $.padLeft(this.Minutes, 2, "0") + ":" + $.padLeft(this.Seconds, 2, "0") + "." + $.padLeft(this.Milliseconds, 3, "0"); } };} 34 35 /*--時間相關函數 End----------------------------------------*/ 36 37 38 })(jQuery);View Code
代碼比較長,當時我寫完之後,順手就把 JS壓縮了。
代碼調用如下:
var begin = new Date(); //執行15W次 自動識別轉換: //i3 CPU //Chrome: 5秒 平均: 30000次/s //IE 9 : 26秒 平均: 5770次/s //i5 CPU //Chrome: 14秒 20*100000次 平均: 107000次/s for (var i = 0; i < 10000; i++) { var result1 = $.toDateTime("2017-03-19 01:43:15 123453"); var result2 = $.toDateTime("2017-03-19T01:43:15 123453"); var result3 = $.toDateTime("2017-03-19"); var result4 = $.toDateTime("2017/03/19 01:43:15"); var result5 = $.toDateTime("03/19/2017 01:43:15"); var result6 = $.toDateTime("03/19/2017 01:43"); var result7 = $.toDateTime("01:43:15 123453"); var result8 = $.toDateTime("01:43:15.123453"); var result9 = $.toDateTime("01:43:15"); var result10 = $.toDateTime("01:43"); var result11 = $.toDateTime("2017/03/19"); var result12 = $.toDateTime("03/19/2017"); var result13 = $.toDateTime("Mon Mar 20 2017 02:46:06 GMT+0800 (中國標準時間)"); var result14 = $.toDateTime("Mon Mar 20 02:46:06 UTC+0800 2017"); var result15 = $.toDateTime("Mon Mar 20 2017 02:46:06"); var result16 = $.toDateTime("2017年3月19日 01時43分15秒"); var result17 = $.toDateTime("2017年03月19日01:43:15"); var result18 = $.toDateTime("2017年03月19日 01:43"); var result19 = $.toDateTime("2017年03月19日"); var result20 = $.toDateTime("1時43分15秒"); } var end = new Date(); document.writeln( "計算 " + (i) + "*20次, 耗時:" + "<br/>" + begin.format("yyyy-MM-dd HH:mm:ss fff") + "<br/>" + end.format("yyyy-MM-dd HH:mm:ss fff") + "<br/>" + "計算結果: " + "<br/> $.toDateTime(\"2017-03-19 01:43:15 123453\") >> " + result1.format("yyyy-MM-dd HH:mm:ss fff") + "<br/> $.toDateTime(\"2017-03-19T01:43:15 123453\") >> " + result2.format("yyyy-MM-dd HH:mm:ss fff") + "<br/> $.toDateTime(\"2017-03-19\") >> " + result3.format("yyyy-MM-dd HH:mm:ss fff") + "<br/> $.toDateTime(\"2017/03/19 01:43:15\") >> " + result4.format("yyyy-MM-dd HH:mm:ss fff") + "<br/> $.toDateTime(\"03/19/2017 01:43:15\") >> " + result5.format("yyyy-MM-dd HH:mm:ss fff") + "<br/> $.toDateTime(\"03/19/2017 01:43\") >> " + result6.format("yyyy-MM-dd HH:mm:ss fff") + "<br/> $.toDateTime(\"01:43:15 123453\") >> " + result7.format("yyyy-MM-dd HH:mm:ss fff") + "<br/> $.toDateTime(\"01:43:15.123453\") >> " + result8.format("yyyy-MM-dd HH:mm:ss fff") + "<br/> $.toDateTime(\"01:43:15\") >> " + result9.format("yyyy-MM-dd HH:mm:ss fff") + "<br/> $.toDateTime(\"01:43\") >> " + result10.format("yyyy-MM-dd HH:mm:ss fff") + "<br/> $.toDateTime(\"2017/03/19\") >> " + result11.format("yyyy-MM-dd HH:mm:ss fff") + "<br/> $.toDateTime(\"03/19/2017\") >> " + result12.format("yyyy-MM-dd HH:mm:ss fff") + "<br/> $.toDateTime(\"Mon Mar 20 2017 02:46:06 GMT+0800 (中國標準時間)\") >> " + result13.format("yyyy-MM-dd HH:mm:ss fff") + "<br/> $.toDateTime(\"Mon Mar 20 02:46:06 UTC+0800 2017\") >> " + result14.format("yyyy-MM-dd HH:mm:ss fff") + "<br/> $.toDateTime(\"Mon Mar 20 2017 02:46:06\") >> " + result15.format("yyyy-MM-dd HH:mm:ss fff") + "<br/> $.toDateTime(\"2017年3月19日 01時43分15秒\") >> " + result16.format("yyyy-MM-dd HH:mm:ss fff") + "<br/> $.toDateTime(\"2017年03月19日01:43:15\") >> " + result17.format("yyyy-MM-dd HH:mm:ss fff") + "<br/> $.toDateTime(\"2017年03月19日 01:43\") >> " + result18.format("yyyy-MM-dd HH:mm:ss fff") + "<br/> $.toDateTime(\"2017年03月19日\") >> " + result19.format("yyyy-MM-dd HH:mm:ss fff") + "<br/> $.toDateTime(\"1時43分15秒\") >> " + result20.format("yyyy-MM-dd HH:mm:ss fff") );
上面的代碼中:
1 "Mon Mar 20 2017 02:46:06 GMT+0800 (中國標準時間)" 2 "Mon Mar 20 02:46:06 UTC+0800 2017" 3 "Mon Mar 20 2017 02:46:06" 4 5 這三種,分別是 Chrome FireFox IE 的 Date 對象 toString() 的結果。 6 萬能函數 也已經進行了反向解析支持。
PS.
看客不禁哈哈大笑:“除非有BUG,否則這種 Date.toString() 的字元串 根本就不該出現在程式中。”
如果您覺得 這三種 字元串類型 不會出現。
—— 那我們看看 大名鼎鼎的 element-ui 庫中,DatePicker 控制項 是如何 鬧心的:
http://element-cn.eleme.io/1.4/#/zh-CN/component/date-picker
http://element-cn.eleme.io/2.4/#/zh-CN/component/date-picker (新版已經修正了BUG)
我們再來看看 各位網友是如何踩坑的:
https://www.cnblogs.com/Mrrabbit/p/7716552.html
https://blog.csdn.net/fabulous1111/article/details/79566624
https://blog.csdn.net/lyz571029230/article/details/77987835
—— 無一例外:控制項的值 不能直接獲取,只能通過 change 事件取值。
如何獲取時間差:
1 var timeSpan = $.getTimeSpan("2017-03-27 21:20:05 000", "2017-03-25 10:10:10 000"); 2 alert(timeSpan); //日期比較 是可以直接使用 > < 的 3 alert(timeSpan.TotalDays); //兩個時間相差的天數(有小數,比如 1.5天) 4 alert(timeSpan.TotalHours); //兩個時間相差的小時數(有小數) 5 alert(timeSpan.TotalMinutes); //兩個時間相差的分鐘數(有小數) 6 alert(timeSpan.TotalSeconds); //兩個時間相差的秒數(有小數) 7 alert(timeSpan.TotalMilliseconds); //兩個時間相差的毫秒數(有小數) 8 9 10 alert($.addYears("2017-03-27 21:20:05 123456", 5)); //在指定時間基礎上,增加 5 年 11 alert($.addMonths("2017-03-27 21:20:05 123456", 48)); //在指定時間基礎上,增加 48 月 12 alert($.addDays("2017-03-27 21:20:05 123456", 365)); //在指定時間基礎上,增加 365 天 13 alert($.addHours("2017-03-27 21:20:05 123456", 48)); //在指定時間基礎上,增加 48 消失 14 alert($.addMinutes("2017-03-27 21:20:05 123456", 120)); //在指定時間基礎上,增加 120分鐘 15 alert($.addSeconds("2017-03-27 21:20:05 123456", 480)); //在指定時間基礎上,增加 480秒 16 alert($.addMilliseconds("2017-03-27 21:20:05 123456", 10300)); //在指定時間基礎上,增加 10300毫秒
是的, JS 的 Date 對象, 好多你需要但是 底層不提供的 函數 —— 本文 都已經提供。
本文源碼,完全開源,才用 MIT 開源協議 —— 源碼可以用於 個人、商業,可以修改、添加、刪除,僅僅需要保留 作者的著作聲明。
源碼早在2017年完成,這次項目需要用到 TypeScript —— 於是,就將 之前的部分 底層Js代碼,進行了移植。
今天順帶踩了一下 element-ui 的坑,發文紀念一下 ~
InkFx
2018-08-16