数据格式化

简介: 数据格式化

摘要: 



格式化是通过格式操作使任意类型的数据转换成一个字符串。例如下面这样


<script>
console.log(chopper.format('{0} - {1} - {2}', 12, 24, 25)); // outputs "12 - 24 - 25"
</script>


下面是一个完整的代码,可以复制到自己的项目中。


1 <!DOCTYPE html>
  2 <html>
  3     <head>
  4         <meta http-equiv="Content-Type" content="text/html;charset=utf-8">
  5     </head>
  6     <body>
  7         <script src="http://code.jquery.com/jquery-1.9.1.min.js"></script>
  8         <script>
  9         (function() {
 10             var chopper = window.chopper = window.chopper || { cultures: {} },
 11                 math = Math,
 12                 formatRegExp = /\{(\d+)(:[^\}]+)?\}/g,
 13                 FUNCTION = "function",
 14                 STRING = "string",
 15                 NUMBER = "number",
 16                 OBJECT = "object",
 17                 NULL = "null",
 18                 BOOLEAN = "boolean",
 19                 UNDEFINED = "undefined",
 20                 slice = [].slice,
 21                 globalize = window.Globalize,
 22                 standardFormatRegExp =  /^(n|c|p|e)(\d*)$/i,
 23                 literalRegExp = /(\\.)|(['][^']*[']?)|(["][^"]*["]?)/g,
 24                 commaRegExp = /\,/g,
 25                 EMPTY = "",
 26                 POINT = ".",
 27                 COMMA = ",",
 28                 SHARP = "#",
 29                 ZERO = "0",
 30                 PLACEHOLDER = "??",
 31                 EN = "en-US",
 32                 objectToString = {}.toString;
 33 
 34             //cultures
 35             chopper.cultures["en-US"] = {
 36                 name: EN,
 37                 numberFormat: {
 38                     pattern: ["-n"],
 39                     decimals: 2,
 40                     ",": ",",
 41                     ".": ".",
 42                     groupSize: [3],
 43                     percent: {
 44                         pattern: ["-n %", "n %"],
 45                         decimals: 2,
 46                         ",": ",",
 47                         ".": ".",
 48                         groupSize: [3],
 49                         symbol: "%"
 50                     },
 51                     currency: {
 52                         pattern: ["($n)", "$n"],
 53                         decimals: 2,
 54                         ",": ",",
 55                         ".": ".",
 56                         groupSize: [3],
 57                         symbol: "$"
 58                     }
 59                 },
 60                 calendars: {
 61                     standard: {
 62                         days: {
 63                             names: ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"],
 64                             namesAbbr: ["Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"],
 65                             namesShort: [ "Su", "Mo", "Tu", "We", "Th", "Fr", "Sa" ]
 66                         },
 67                         months: {
 68                             names: ["January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December"],
 69                             namesAbbr: ["Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"]
 70                         },
 71                         AM: [ "AM", "am", "AM" ],
 72                         PM: [ "PM", "pm", "PM" ],
 73                         patterns: {
 74                             d: "M/d/yyyy",
 75                             D: "dddd, MMMM dd, yyyy",
 76                             F: "dddd, MMMM dd, yyyy h:mm:ss tt",
 77                             g: "M/d/yyyy h:mm tt",
 78                             G: "M/d/yyyy h:mm:ss tt",
 79                             m: "MMMM dd",
 80                             M: "MMMM dd",
 81                             s: "yyyy'-'MM'-'ddTHH':'mm':'ss",
 82                             t: "h:mm tt",
 83                             T: "h:mm:ss tt",
 84                             u: "yyyy'-'MM'-'dd HH':'mm':'ss'Z'",
 85                             y: "MMMM, yyyy",
 86                             Y: "MMMM, yyyy"
 87                         },
 88                         "/": "/",
 89                         ":": ":",
 90                         firstDay: 0,
 91                         twoDigitYearMax: 2029
 92                     }
 93                 }
 94             };
 95 
 96 
 97              function findCulture(culture) {
 98                 if (culture) {
 99                     if (culture.numberFormat) {
100                         return culture;
101                     }
102 
103                     if (typeof culture === STRING) {
104                         var cultures = chopper.cultures;
105                         return cultures[culture] || cultures[culture.split("-")[0]] || null;
106                     }
107 
108                     return null;
109                 }
110 
111                 return null;
112             }
113 
114             function getCulture(culture) {
115                 if (culture) {
116                     culture = findCulture(culture);
117                 }
118 
119                 return culture || chopper.cultures.current;
120             }
121 
122             function expandNumberFormat(numberFormat) {
123                 numberFormat.groupSizes = numberFormat.groupSize;
124                 numberFormat.percent.groupSizes = numberFormat.percent.groupSize;
125                 numberFormat.currency.groupSizes = numberFormat.currency.groupSize;
126             }
127 
128             chopper.culture = function(cultureName) {
129                 var cultures = chopper.cultures, culture;
130 
131                 if (cultureName !== undefined) {
132                     culture = findCulture(cultureName) || cultures[EN];
133                     culture.calendar = culture.calendars.standard;
134                     cultures.current = culture;
135 
136                     if (globalize && !globalize.load) {
137                         expandNumberFormat(culture.numberFormat);
138                     }
139 
140                 } else {
141                     return cultures.current;
142                 }
143             };
144 
145 
146             chopper.culture(EN);
147 
148             //number formatting
149             function formatNumber(number, format, culture) {
150                 culture = getCulture(culture);
151 
152                 var numberFormat = culture.numberFormat,
153                     groupSize = numberFormat.groupSize[0],
154                     groupSeparator = numberFormat[COMMA],
155                     decimal = numberFormat[POINT],
156                     precision = numberFormat.decimals,
157                     pattern = numberFormat.pattern[0],
158                     literals = [],
159                     symbol,
160                     isCurrency, isPercent,
161                     customPrecision,
162                     formatAndPrecision,
163                     negative = number < 0,
164                     integer,
165                     fraction,
166                     integerLength,
167                     fractionLength,
168                     replacement = EMPTY,
169                     value = EMPTY,
170                     idx,
171                     length,
172                     ch,
173                     hasGroup,
174                     hasNegativeFormat,
175                     decimalIndex,
176                     sharpIndex,
177                     zeroIndex,
178                     hasZero, hasSharp,
179                     percentIndex,
180                     currencyIndex,
181                     startZeroIndex,
182                     start = -1,
183                     end;
184 
185                 //return empty string if no number
186                 if (number === undefined) {
187                     return EMPTY;
188                 }
189 
190                 if (!isFinite(number)) {
191                     return number;
192                 }
193 
194                 //if no format then return number.toString() or number.toLocaleString() if culture.name is not defined
195                 if (!format) {
196                     return culture.name.length ? number.toLocaleString() : number.toString();
197                 }
198 
199                 formatAndPrecision = standardFormatRegExp.exec(format);
200 
201                 // standard formatting
202                 if (formatAndPrecision) {
203                     format = formatAndPrecision[1].toLowerCase();
204 
205                     isCurrency = format === "c";
206                     isPercent = format === "p";
207 
208                     if (isCurrency || isPercent) {
209                         //get specific number format information if format is currency or percent
210                         numberFormat = isCurrency ? numberFormat.currency : numberFormat.percent;
211                         groupSize = numberFormat.groupSize[0];
212                         groupSeparator = numberFormat[COMMA];
213                         decimal = numberFormat[POINT];
214                         precision = numberFormat.decimals;
215                         symbol = numberFormat.symbol;
216                         pattern = numberFormat.pattern[negative ? 0 : 1];
217                     }
218 
219                     customPrecision = formatAndPrecision[2];
220 
221                     if (customPrecision) {
222                         precision = +customPrecision;
223                     }
224 
225                     //return number in exponential format
226                     if (format === "e") {
227                         return customPrecision ? number.toExponential(precision) : number.toExponential(); // toExponential() and toExponential(undefined) differ in FF #653438.
228                     }
229 
230                     // multiply if format is percent
231                     if (isPercent) {
232                         number *= 100;
233                     }
234 
235                     number = round(number, precision);
236                     negative = number < 0;
237                     number = number.split(POINT);
238 
239                     integer = number[0];
240                     fraction = number[1];
241 
242                     //exclude "-" if number is negative.
243                     if (negative) {
244                         integer = integer.substring(1);
245                     }
246 
247                     value = integer;
248                     integerLength = integer.length;
249 
250                     //add group separator to the number if it is longer enough
251                     if (integerLength >= groupSize) {
252                         value = EMPTY;
253                         for (idx = 0; idx < integerLength; idx++) {
254                             if (idx > 0 && (integerLength - idx) % groupSize === 0) {
255                                 value += groupSeparator;
256                             }
257                             value += integer.charAt(idx);
258                         }
259                     }
260 
261                     if (fraction) {
262                         value += decimal + fraction;
263                     }
264 
265                     if (format === "n" && !negative) {
266                         return value;
267                     }
268 
269                     number = EMPTY;
270 
271                     for (idx = 0, length = pattern.length; idx < length; idx++) {
272                         ch = pattern.charAt(idx);
273 
274                         if (ch === "n") {
275                             number += value;
276                         } else if (ch === "$" || ch === "%") {
277                             number += symbol;
278                         } else {
279                             number += ch;
280                         }
281                     }
282 
283                     return number;
284                 }
285 
286                 //custom formatting
287                 //
288                 //separate format by sections.
289 
290                 //make number positive
291                 if (negative) {
292                     number = -number;
293                 }
294 
295                 if (format.indexOf("'") > -1 || format.indexOf("\"") > -1 || format.indexOf("\\") > -1) {
296                     format = format.replace(literalRegExp, function (match) {
297                         var quoteChar = match.charAt(0).replace("\\", ""),
298                             literal = match.slice(1).replace(quoteChar, "");
299 
300                         literals.push(literal);
301 
302                         return PLACEHOLDER;
303                     });
304                 }
305 
306                 format = format.split(";");
307                 if (negative && format[1]) {
308                     //get negative format
309                     format = format[1];
310                     hasNegativeFormat = true;
311                 } else if (number === 0) {
312                     //format for zeros
313                     format = format[2] || format[0];
314                     if (format.indexOf(SHARP) == -1 && format.indexOf(ZERO) == -1) {
315                         //return format if it is string constant.
316                         return format;
317                     }
318                 } else {
319                     format = format[0];
320                 }
321 
322                 percentIndex = format.indexOf("%");
323                 currencyIndex = format.indexOf("$");
324 
325                 isPercent = percentIndex != -1;
326                 isCurrency = currencyIndex != -1;
327 
328                 //multiply number if the format has percent
329                 if (isPercent) {
330                     number *= 100;
331                 }
332 
333                 if (isCurrency && format[currencyIndex - 1] === "\\") {
334                     format = format.split("\\").join("");
335                     isCurrency = false;
336                 }
337 
338                 if (isCurrency || isPercent) {
339                     //get specific number format information if format is currency or percent
340                     numberFormat = isCurrency ? numberFormat.currency : numberFormat.percent;
341                     groupSize = numberFormat.groupSize[0];
342                     groupSeparator = numberFormat[COMMA];
343                     decimal = numberFormat[POINT];
344                     precision = numberFormat.decimals;
345                     symbol = numberFormat.symbol;
346                 }
347 
348                 hasGroup = format.indexOf(COMMA) > -1;
349                 if (hasGroup) {
350                     format = format.replace(commaRegExp, EMPTY);
351                 }
352 
353                 decimalIndex = format.indexOf(POINT);
354                 length = format.length;
355 
356                 if (decimalIndex != -1) {
357                     fraction = number.toString().split("e");
358                     if (fraction[1]) {
359                         fraction = round(number, Math.abs(fraction[1]));
360                     } else {
361                         fraction = fraction[0];
362                     }
363                     fraction = fraction.split(POINT)[1] || EMPTY;
364                     zeroIndex = format.lastIndexOf(ZERO) - decimalIndex;
365                     sharpIndex = format.lastIndexOf(SHARP) - decimalIndex;
366                     hasZero = zeroIndex > -1;
367                     hasSharp = sharpIndex > -1;
368                     idx = fraction.length;
369 
370                     if (!hasZero && !hasSharp) {
371                         format = format.substring(0, decimalIndex) + format.substring(decimalIndex + 1);
372                         length = format.length;
373                         decimalIndex = -1;
374                         idx = 0;
375                     } if (hasZero && zeroIndex > sharpIndex) {
376                         idx = zeroIndex;
377                     } else if (sharpIndex > zeroIndex) {
378                         if (hasSharp && idx > sharpIndex) {
379                             idx = sharpIndex;
380                         } else if (hasZero && idx < zeroIndex) {
381                             idx = zeroIndex;
382                         }
383                     }
384 
385                     if (idx > -1) {
386                         number = round(number, idx);
387                     }
388                 } else {
389                     number = round(number);
390                 }
391 
392                 sharpIndex = format.indexOf(SHARP);
393                 startZeroIndex = zeroIndex = format.indexOf(ZERO);
394 
395                 //define the index of the first digit placeholder
396                 if (sharpIndex == -1 && zeroIndex != -1) {
397                     start = zeroIndex;
398                 } else if (sharpIndex != -1 && zeroIndex == -1) {
399                     start = sharpIndex;
400                 } else {
401                     start = sharpIndex > zeroIndex ? zeroIndex : sharpIndex;
402                 }
403 
404                 sharpIndex = format.lastIndexOf(SHARP);
405                 zeroIndex = format.lastIndexOf(ZERO);
406 
407                 //define the index of the last digit placeholder
408                 if (sharpIndex == -1 && zeroIndex != -1) {
409                     end = zeroIndex;
410                 } else if (sharpIndex != -1 && zeroIndex == -1) {
411                     end = sharpIndex;
412                 } else {
413                     end = sharpIndex > zeroIndex ? sharpIndex : zeroIndex;
414                 }
415 
416                 if (start == length) {
417                     end = start;
418                 }
419 
420                 if (start != -1) {
421                     value = number.toString().split(POINT);
422                     integer = value[0];
423                     fraction = value[1] || EMPTY;
424 
425                     integerLength = integer.length;
426                     fractionLength = fraction.length;
427 
428                     if (negative && (number * -1) >= 0) {
429                         negative = false;
430                     }
431 
432                     //add group separator to the number if it is longer enough
433                     if (hasGroup) {
434                         if (integerLength === groupSize && integerLength < decimalIndex - startZeroIndex) {
435                             integer = groupSeparator + integer;
436                         } else if (integerLength > groupSize) {
437                             value = EMPTY;
438                             for (idx = 0; idx < integerLength; idx++) {
439                                 if (idx > 0 && (integerLength - idx) % groupSize === 0) {
440                                     value += groupSeparator;
441                                 }
442                                 value += integer.charAt(idx);
443                             }
444 
445                             integer = value;
446                         }
447                     }
448 
449                     number = format.substring(0, start);
450 
451                     if (negative && !hasNegativeFormat) {
452                         number += "-";
453                     }
454 
455                     for (idx = start; idx < length; idx++) {
456                         ch = format.charAt(idx);
457 
458                         if (decimalIndex == -1) {
459                             if (end - idx < integerLength) {
460                                 number += integer;
461                                 break;
462                             }
463                         } else {
464                             if (zeroIndex != -1 && zeroIndex < idx) {
465                                 replacement = EMPTY;
466                             }
467 
468                             if ((decimalIndex - idx) <= integerLength && decimalIndex - idx > -1) {
469                                 number += integer;
470                                 idx = decimalIndex;
471                             }
472 
473                             if (decimalIndex === idx) {
474                                 number += (fraction ? decimal : EMPTY) + fraction;
475                                 idx += end - decimalIndex + 1;
476                                 continue;
477                             }
478                         }
479 
480                         if (ch === ZERO) {
481                             number += ch;
482                             replacement = ch;
483                         } else if (ch === SHARP) {
484                             number += replacement;
485                         }
486                     }
487 
488                     if (end >= start) {
489                         number += format.substring(end + 1);
490                     }
491 
492                     //replace symbol placeholders
493                     if (isCurrency || isPercent) {
494                         value = EMPTY;
495                         for (idx = 0, length = number.length; idx < length; idx++) {
496                             ch = number.charAt(idx);
497                             value += (ch === "$" || ch === "%") ? symbol : ch;
498                         }
499                         number = value;
500                     }
501 
502                     length = literals.length;
503 
504                     if (length) {
505                         for (idx = 0; idx < length; idx++) {
506                             number = number.replace(PLACEHOLDER, literals[idx]);
507                         }
508                     }
509                 }
510 
511                 return number;
512             }
513 
514             var round = function(value, precision) {
515                 precision = precision || 0;
516 
517                 value = value.toString().split('e');
518                 value = Math.round(+(value[0] + 'e' + (value[1] ? (+value[1] + precision) : precision)));
519 
520                 value = value.toString().split('e');
521                 value = +(value[0] + 'e' + (value[1] ? (+value[1] - precision) : -precision));
522 
523                 return value.toFixed(precision);
524             };
525 
526             var toString = function(value, fmt, culture) {
527                 if (fmt) {
528                     if (typeof value === NUMBER) {
529                         return formatNumber(value, fmt, culture);
530                     }
531                 }
532 
533                 return value !== undefined ? value : "";
534             };
535 
536             if (globalize && !globalize.load) {
537                 toString = function(value, format, culture) {
538                     if ($.isPlainObject(culture)) {
539                         culture = culture.name;
540                     }
541 
542                     return globalize.format(value, format, culture);
543                 };
544             }
545 
546             chopper.format = function(fmt) {
547                 var values = arguments;
548 
549                 return fmt.replace(formatRegExp, function(match, index, placeholderFormat) {
550                     var value = values[parseInt(index, 10) + 1];
551 
552                     return toString(value, placeholderFormat ? placeholderFormat.substring(1) : "");
553                 });
554             };
555         })();
556         </script>
557     </body>
558 </html>


API:



chopper.format('{0} is playing {1}', 'Xiaoming', 'basketball'); // outputs "Xiaoming is playing basketball"
// 价格
chopper.format('{0:c} - {1:c}', 10, 20); // outputs "10.00−10.00−20.00"
// 指数
chopper.format('指数: {0:e}', 25); // outputs "指数: 2.5e+1"
// 百分数
chopper.format('百分数: {0:p}', 25); // outputs "百分数: 2,500.00 %"
// 小数
chopper.format('小数: {0:n}', 25); // outputs "小数: 25.00"


小结:


 

开发中格式化数据还是经常用到的,比如我们要根据变量提示不同的信息,但是内容模板都是一样的,这样的话我们就可以使用此方法。如果你的项目使用jQuery,你也可以将上面的javascript封装成jQuery插件。

相关文章
数据解析之xpath 太6了
数据解析之xpath 太6了
ThingWorx 支持哪些数据格式
ThingWorx 平台支持多种数据格式,包括但不限于 JSON、XML、CSV 和 MQTT 协议等,以满足不同设备和应用的数据交换需求。
60 13
JSON 数据格式化方法
JSON 数据格式化方法
142 3
|
8月前
|
数据的格式与描述
数据的格式与描述
70 4
中间件数据格式的转换
中间件数据格式转换涉及在系统间传递数据时调整格式,以适应不同标准。常见转换包括:JSON与XML互转、文本到二进制、结构化与非结构化数据转换、不同数据标准间的转换及处理自定义格式。实现转换通常借助编程语言(如Python的json、xml库)和工具(如Apache NiFi、StreamSets)。设计时需考虑数据源、目标、传输和性能需求。
105 2
中间件数据格式文本与二进制之间的转换
中间件数据格式文本与二进制之间的转换
46 2
JSON格式转换工具:快速、简单、高效处理JSON数据
JSON格式转换工具:快速、简单、高效处理JSON数据
601 1