package com.study.test;

import java.io.Serializable;
import java.text.SimpleDateFormat;
import java.util.*; /**
* 实现Java中日期的简单格式化,支持以下字段:
* yyyy:年
* MM:月
* dd:日
* hh:1~12小时制(1-12)
* HH:24小时制(0-23)
* mm:分
* ss:秒
* S:毫秒
* E:星期几
* a: 上午/下午
*/
public class DateFormatter implements Serializable { private static Map<Character, Integer> tokenFieldMap = new HashMap<>();
private static final int NO_LENGTH = 1000; //不用自动补0的属性,例如毫秒
private static final int LOCALE = NO_LENGTH + 1000; //用Locale获取值的属性,例如星期 //tokenFieldMap,将字符与Calendar中的field对应
static {
tokenFieldMap.put('y', Calendar.YEAR);
tokenFieldMap.put('M', Calendar.MONTH);
tokenFieldMap.put('d', Calendar.DATE);
tokenFieldMap.put('h', Calendar.HOUR);
tokenFieldMap.put('H', Calendar.HOUR_OF_DAY);
tokenFieldMap.put('m', Calendar.MINUTE);
tokenFieldMap.put('s', Calendar.SECOND);
tokenFieldMap.put('a', Calendar.AM_PM + LOCALE);
tokenFieldMap.put('S',Calendar.MILLISECOND + NO_LENGTH);
tokenFieldMap.put('E',Calendar.DAY_OF_WEEK + LOCALE);
} private static class Token implements Serializable {
int field; // Calendar中的field对应
int length; // 自动补0的长度 public Token(int field, int length) {
this.field = field;
this.length = length;
}
} //解析出的token,可能是token或者是String
private List<Object> tokens; private String format; public DateFormatter(String format) {
this.format = format;
parseTokens();
} public String getFormat() {
return format;
} public String format(Date date){
if(date == null){
throw new IllegalArgumentException("null argument!");
}
StringBuilder sb = new StringBuilder();
Calendar calendar = Calendar.getInstance();
calendar.setTime(date);
for(Object token : tokens){
if(token instanceof Token){
token2String(sb,(Token)token,calendar);
}else{
sb.append(token);
}
}
return sb.toString();
} private void parseTokens() {
tokens = new ArrayList<>();
StringBuilder temp = new StringBuilder();
for (int i = 0; i < format.length(); i++) {
char c = format.charAt(i);
Integer field = tokenFieldMap.get(c);
if (field != null) {
checkStr(temp);
if(field < NO_LENGTH){
int num = 1;
while (i < format.length() - 1 && format.charAt(i + 1) == c) {
i++;
num++;
}
tokens.add(new Token(field, num));
}else{
tokens.add(new Token(field, 0));
}
}else{
temp.append(c);
}
}
checkStr(temp);
} private void checkStr(StringBuilder temp) {
if (temp.length() > 0) {
tokens.add(temp.toString());
temp.setLength(0);
}
} @SuppressWarnings("MagicConstant")
private void token2String(StringBuilder sb, Token token, Calendar calendar){
int field = token.field;
if(field > LOCALE){
sb.append(calendar.getDisplayName(field - LOCALE,Calendar.SHORT,Locale.getDefault()));
}else if(field > NO_LENGTH){
sb.append(calendar.get(field - NO_LENGTH));
}else{
int val = calendar.get(field);
if(field == Calendar.MONTH){
val++; //如果是月,取出的范围是0-11,需要加一
}
String value = String.valueOf(val);
if(value.length() > token.length){
if(token.field == Calendar.YEAR){ //如果是年,才截取,比如2018可以截取成18年,小时分钟不能截取
sb.append(value.substring(value.length() - token.length));
}else{
sb.append(value);
}
}else{
for(int i=0;i<token.length - value.length();i++){ //根据length自动补0
sb.append('0');
}
sb.append(value);
}
}
} public static void main(String[] args)throws Exception {
String fmt = "yy-M-dd a h:m:s E";
Date date = new Date(); DateFormatter formatter = new DateFormatter(fmt);
String result = formatter.format(date);
System.out.println(result); SimpleDateFormat simpleDateFormat = new SimpleDateFormat(fmt);
System.out.println(simpleDateFormat.format(date)); //性能测试
long t1 = System.currentTimeMillis();
for(int i=0;i<30000;i++){
simpleDateFormat.format(new Date(System.currentTimeMillis() + i * 3000 ));
}
long t2 = System.currentTimeMillis();
for(int i=0;i<30000;i++){
formatter.format(new Date(System.currentTimeMillis() + i * 3000));
}
long t3 = System.currentTimeMillis(); System.out.println("formatter cost : " + (t3 - t2));
System.out.println("java format cost : " + (t2 - t1)); }
}

Java中日期格式化的实现算法的更多相关文章

  1. 震惊!java中日期格式化的大坑!

    前言 我们都知道在java中进行日期格式化使用simpledateformat.通过格式 yyyy-MM-dd 等来进行格式化,但是你知道其中微小的坑吗? yyyy 和 YYYY 示例代码 @Test ...

  2. Java中日期格式化YYYY-DD的坑

    写这篇博文是记录下跨年的bug.去年隔壁组的小伙伴就是计算两个日期之间间隔的天数,因为跨年的原因计算有误. 当时测试组的小姐姐也没有模拟出来这种场景,导致上生产环境直接影响线上的数据. 今天逛技术论论 ...

  3. Java中日期格式化SimpleDateFormat类包含时区的处理方法

    1.前言 需要把格式为“2017-02-23T08:04:02+01:00”转化成”23-02-2017-T15:04:02“格式(中国时区为+08:00所以是15点),通过网上查找答案,发现没有我需 ...

  4. Java基础-日期格式化DateFormat类简介

    Java基础-日期格式化DateFormat类简介 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.DateFormat类概述 DateFormat 是日期/时间格式化子类的抽象 ...

  5. java向MySQL插入当前时间的四种方式和java时间日期格式化的几种方法(案例说明)

    转载地址:http://www.devba.com/index.php/archives/4581.html java向MySQL插入当前时间的四种方式和java时间日期格式化的几种方法(案例说明); ...

  6. (转)java向MySQL插入当前时间的四种方式和java时间日期格式化的几种方法(案例说明)

    java向MySQL插入当前时间的四种方式和java时间日期格式化的几种方法(案例说明);部分资料参考网络资源 1. java向MySQL插入当前时间的四种方式 第一种:将java.util.Date ...

  7. java中日期常用

    Java中日期的几种常见操作 —— 取值.转换.加减.比较 Java 的开发过程中免不了与 Date 类型纠缠,准备总结一下项目经常使用的日期相关操作,JDK 版本 1.7,如果能够帮助大家节约那么几 ...

  8. JAVA中日期 yyyy-MM-dd HH:mm:ss和yyyy-MM-dd hh:mm:ss的区别

    JAVA中日期 yyyy-MM-dd HH:mm:ss和yyyy-MM-dd hh:mm:ss的区别 : HH:24小时制 hh:12小时制 package time; import java.tex ...

  9. 聊聊 Java 中日期的几种常见操作 —— 取值、转换、加减、比较

    Java 的开发过程中免不了与 Date 类型纠缠,准备总结一下项目经常使用的日期相关操作,JDK 版本 1.7,如果能够帮助大家节约那么几分钟起身活动一下,去泡杯咖啡,便是极好的,嘿嘿.当然,我只提 ...

随机推荐

  1. python单ip端口扫描器

    from socket import * import threading #导入线程相关模块 # qianxiao996精心制作 #博客地址:https://blog.csdn.net/qq_363 ...

  2. python练习册 每天一个小程序 第0004题

    1 #-*-coding:utf-8-*- 2 __author__ = 'Deen' 3 ''' 4 题目描述:任一个英文的纯文本文件,统计其中的单词出现的个数. 5 参考学习链接: 6 re ht ...

  3. P1030

    题面 给出一棵二叉树的中序排列与后序排列.求出它的先序排列.(约定树结点用不同的大写字母表示,长度≤8). 输入格式 2行,均为大写字母组成的字符串,表示一棵二叉树的中序排列与后序排列. 输出格式 1 ...

  4. 简单了解一下pinia的结构

    随着 Vue3 的正式转正,Pinia 也渐渐火了起来.所以要更新一下自己的知识树了.这里主要是看看新的状态是什么"形态". 状态的容器还是"reactive" ...

  5. java 中有几种方法可以实现一个线程?

    继承 Thread 类 实现 Runnable 接口 实现 Callable 接口,需要实现的是 call() 方法

  6. 写一段代码在遍历 ArrayList 时移除一个元素?

    该问题的关键在于面试者使用的是 ArrayList 的 remove() 还是 Iterator 的 remove()方法.这有一段示例代码,是使用正确的方式来实现在遍历的过程中移 除元素,而不会出现 ...

  7. 详解 IOC

    什么是IOC: IOC-Inversion Of Control,即"控制反转",不是什么技术,而是一种设计思想.在Java开发中,IOC意味着将你设计好的对象交给容器控制,而不是 ...

  8. springboot使用自定义注解和反射实现一个简单的支付

    优点: 未使用if else,就算以后增加支付类型,也不用改动之前代码 只需要新写一个支付类,给添加自定义注解@Pay 首先: 定义自定义注解 Pay 定义 CMBPay ICBCPay 两种支付 根 ...

  9. 学习MFS(五)

     ########################################################## mfs master 安装 建议 cp eth0 eth0:0 ifup eth ...

  10. js技术之转换为大写toUpperCase()

    案例:把所有单词以空格为分割并将首字母转为大写 <!DOCTYPE html><html lang="en"><head> <meta c ...