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. DDOS反射攻击

    0x01 环境 包含3台主机 attact 作为攻击方,使用Centos7.2 reflect 作为流量放大器,安装有dns .ntp .memcached三种可以放大流量的服务 windows_se ...

  2. sql注入mysql注入

    #跨库查询及应用思路 information_schema表特性,记录数据库名.表名.列名对应表 information_schema.schemata:存储所有数据库名 schema_name:数据 ...

  3. 用ssh无密码登录远程linux

    登录linux常用的方式是:用户名+密码,多次输入密码非常不方便,所以推荐使用密钥登录,安全又方便,下面我说下怎么使用密钥登录. 生成密钥 使用密钥登录首先需要本地有ssh密钥 如果本地没有密钥,那么 ...

  4. select 中的timeout

    1. select 相关man 资料 /* According to POSIX.1-2001 */ #include <sys/select.h> /* According to ear ...

  5. k8s集群搭建过程详解

    准备工作 安装CentOS7虚拟机 略 安装Docker 略 关闭CentOS7自带的防火墙服务 systemctl disable firewalld systemctl stop firewall ...

  6. class文件和java文件区别

  7. 手撕代码:leetcode70爬楼梯

    装载于:https://blog.csdn.net/qq_35091252/article/details/90576779 题目描述 假设你正在爬楼梯.需要n阶你才能到达楼顶. 每次你可以爬1或2个 ...

  8. 三、MyCat主要配置介绍

    一.配置文件 1.server.xml Mycat的配置文件,设置账号.参数等2.schema.xml Mycat对应的物理数据库和数据库表的配置3.rule.xml Mycat分片(分库分表)规则 ...

  9. Redis缓存穿透、缓存雪崩、缓存击穿

    缓存穿透: ​ 缓存穿透,是指查询一个数据库一定不存在的数据.正常的使用缓存流程大致是,数据查询先进行缓存查询,如果key不存在或者key已经过期,再对数据库进行查询,并把查询到的对象,放进缓存.如果 ...

  10. element-ui 无法对绑定表单的对象中的对象属性进行验证

    <el-form-item label="类型" :label-width="formLabelWidth" prop="typeId" ...