溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Java使用DateTimeFormatter實(shí)現(xiàn)格式化時(shí)間

發(fā)布時(shí)間:2020-09-24 07:14:26 來源:腳本之家 閱讀:200 作者:ZLC_hrkj 欄目:編程語言

用掃描器獲取輸入的時(shí)間(年月日時(shí)分),這個(gè)時(shí)間的格式是常用的格式,然后格式化這個(gè)時(shí)間,把格式化的時(shí)間輸出到控制臺(tái),可以在控制臺(tái)重復(fù)輸入時(shí)間.格式化的時(shí)間參考企業(yè)微信聊天記錄的展示時(shí)間.用DateTimeFormatter實(shí)現(xiàn),功能如下:
同年:
不同年:
同月:月日+上午/下午+時(shí)分
同年不同月:月日+時(shí)分
今天:上午/下午+時(shí)分
明天:明天+上午/下午+時(shí)分
昨天:昨天+上午/下午+時(shí)分
包括今天在內(nèi)的一周內(nèi):星期+上午/下午+時(shí)分

首先看一下測試類:

package hrkj;

import java.util.Scanner;

/**
 * 日期時(shí)間測試類
 * 
 * @author 張林琛<br>
 * @date 2020/01/10 08:35:29
 * @version 1.0
 */
public class DateTimeTest {

 public static void main(String[] args) {
 // 獲取Scanner掃描器
 Scanner sc = new Scanner(System.in);
 // 程序開始提示語句
 Hint.HINT_STAR.print();
 // 程序退出提示語句
 System.out.println(Order.ORDER_EXIT.getname());
 while (sc.hasNextLine()) {
  String str = sc.nextLine();
  // 判斷是否為退出指令
  if (Order.ORDER_EXIT.compare(str)) {
  Hint.HINT_END.print();
  //如果退出,則關(guān)閉掃描器
  sc.close();
  break;
  // 判斷輸入的是否為空格或回車
  } else if (str.equals(" ") || str.equals(""
   + "")) {
  Hint.BLANK_OR_ENTER.print();
  } else {
  // 開始處理日期時(shí)間
  DateTimeTool.dateTimeFormatter(str);
  }
  Hint.HINT_STAR.print();
  System.out.println(Order.ORDER_EXIT.getname());
 }
 }

}

這里比較好的一點(diǎn)就是,在程序結(jié)束之前,會(huì)關(guān)閉掃描器,程序開始和程序結(jié)束,都會(huì)有提示語句:

請輸入正確的時(shí)間,例如:
2020-01-10 10:10   2020.01.10 10:10   2020/01/10 10:10
退出請輸入:[exit, 退出]

上面效果分別對應(yīng)以下兩行代碼:

    // 程序開始提示語句
 Hint.HINT_STAR.print();
 // 程序退出提示語句
 System.out.println(Order.ORDER_EXIT.getname());

其中[exit, 退出]單獨(dú)為第二行代碼,與第一行代碼效果拼接在一起,值得注意的是,Hint是一個(gè)枚舉類,程序開始;運(yùn)行期間和結(jié)尾的提示語句,都定義為枚舉值,具體如下:

package hrkj;

/**
 * 日期時(shí)間類的提示語句,用于提示錯(cuò)誤信息
 * 
 * @author 張林琛<br>
 * @date 2020/01/10 08:39:27
 * @version 1.0
 */

public enum Hint {
 /**
 * 日期時(shí)間長度不合法
 */
 LENGTH_ILLEGAL("日期時(shí)間長度不合法,"),
 /**
 * 日期錯(cuò)誤
 */
 HINT_DATE("日期輸入錯(cuò)誤,"),
 /**
 * 月份輸入不合法
 */
 HINT_MONTH("月份輸入不合法,"),
 /**
 * 大月只有31天
 */
 HINT_BIGMONTH("大月只有31天,"),
 /**
 * 小月只有30天
 */
 HINT_TINYMONTH("小月只有30天,"),
 /**
 * 平年2月只有28天
 */
 HINT_TINY_TWOMONTH("平年二月只有28天,"),
 /**
 * 閏年二月只有29天
 */
 HINT_BIG_TWOMONTH("閏年二月只有28天,"),
 /**
 * 時(shí)間錯(cuò)誤
 */
 HINT_TIME("時(shí)間輸入錯(cuò)誤,"),
 /**
 * 輸入空格或回車
 */
 BLANK_OR_ENTER("不能輸入空格或直接回車,"),
 /**
 * 以空格分開日期和時(shí)間
 */
 DATE_BLANK_TIME("請以空格分割開日期和時(shí)間,"),
 /**
 * 開始提示語句
 */
 HINT_STAR("請輸入正確的時(shí)間,例如:\n2020-01-10 10:10  2020.01.10 10:10  2020/01/10 10:10\n退出請輸入:"),
 /**
 * 程序結(jié)束提示語句
 */
 HINT_END("程序結(jié)束,歡迎下次使用!");
 // 定義實(shí)例變量
 private final String name;

 /**
 * 單個(gè)參數(shù)構(gòu)造器
 * 
 * @param name
 */
 private Hint(String name) {
 this.name = name;

 }

 /**
 * 獲取實(shí)例變量的值
 * 
 * @return name
 */
 public String getName() {
 return name;
 }

 /**
 * 打印提示語句的方法
 */
 public void print() {
 System.out.print(name);
 }
}

因?yàn)槊杜e類中定義了一個(gè)print()方法,調(diào)用此方法可直接打印枚舉值.
上面說到,[exit, 退出]為退出指令,用戶輸入exit或退出則會(huì)關(guān)閉掃描器,接著關(guān)閉程序,在此也把退出定義為一個(gè)指令枚舉類,具體實(shí)現(xiàn)如下:

package hrkj;

import java.util.Arrays;

/**
 * 日期時(shí)間類的指令集,用于輸入相對應(yīng)的指令
 * 
 * @author 張林琛<br>
 * @date 2020/01/10 08:38:05
 * @version 1.0
 */
public enum Order {
 /**
 * 退出指令
 */
 ORDER_EXIT("exit","退出");
 // 定義實(shí)例變量數(shù)組name
 private final String[] name;

 /**
 * 形參可變的構(gòu)造器
 * @param name
 */
 private Order(String... name) {
 this.name = name;
 }

 /**
 * 獲取退出的指令
 * @return 獲取的指令
 */
 public String getname() {
 return Arrays.toString(name);
 }

 /**
 * 判斷輸入的是不是退出的指令
 * @param str 輸入的指令
 * @return 是否是退出指令
 */
 public boolean compare(String str) {
 for (int i = 0; i < name.length; i++) {
  if (str.equals(name[i])) {
  return true;
  }
 }
 return false;
 }
}

Order枚舉類雖然枚舉實(shí)例只有一個(gè)退出(如果有其他指令可繼續(xù)添加),但枚舉值為兩個(gè):退出和exit,用戶輸入任意一個(gè)就會(huì)達(dá)到退出的效果,在此類中定義一個(gè)比較方法,用于比較用戶輸入的是否與枚舉值相同,用到equals()方法,之所以沒有用contains()方法,是因?yàn)橐坏┯胏ontains()方法,那么用戶輸入e,x,it,等,只要是在枚舉值包含的,都可以退出.

最后便是工具類,用于格式化日期時(shí)間并輸出,實(shí)現(xiàn)細(xì)節(jié)如下

package hrkj;

import java.time.LocalDateTime;
import java.time.MonthDay;
import java.time.Year;
import java.time.format.DateTimeFormatter;
import java.time.format.FormatStyle;
import java.util.Arrays;

/**
 * 日期時(shí)間實(shí)現(xiàn)類,使用DateTimeFormatter實(shí)現(xiàn)
 * 
 * @author 張林琛<br>
 * @date 2020/01/10 08:36:37
 * @version 1.0
 */
public class DateTimeTool {
 /**
 * 有31天的月份
 */
 static int[] bigMonth = { 1, 3, 5, 7, 8, 10, 12 };
 /**
 * 有30天的月份
 */
 static int[] tinyMonth = { 4, 6, 9, 11 };

 /**
 * 處理日期和時(shí)間
 * 
 * @param str
 */
 public static void dateTimeFormatter(String str) {
 // 判斷日期時(shí)間的長度
 if (str.length() < 9 || str.length() > 16) {
  Hint.LENGTH_ILLEGAL.print();
  return;
 }
 // 判斷輸入的日期和時(shí)間是否以空格分割
 if (str.contains(" ")) {
  // 創(chuàng)建數(shù)組來接收分割后的日期時(shí)間
  String[] datetime = str.split(" ");
  // 獲取日期
  String[] date = splitDate(datetime[0]);
  // 判斷日期長度
  if (date.length != 3) {
  Hint.HINT_DATE.print();
  return;
  }
  // 獲取年
  Integer y = Integer.valueOf(date[0]);
  // 獲取月
  Integer M = Integer.valueOf(date[1]);
  // 獲取日
  Integer d = Integer.valueOf(date[2]);
  // 判斷是不是閏年
  if (!handleDate(y, M, d)) {
  // 如果月份大于12或小于1
  if (M > 12 || M < 1) {
   Hint.HINT_MONTH.print();
   return;
   // 如果大月天數(shù)超過31或小于1
  } else if (Arrays.binarySearch(bigMonth, M) > -1 && (d <= 0 || d > 31)) {
   Hint.HINT_BIGMONTH.print();
   return;
   // 如果小月天數(shù)超過30或小于1
  } else if (Arrays.binarySearch(tinyMonth, M) > -1 && (d <= 0 || d > 30)) {
   Hint.HINT_TINYMONTH.print();
   return;
   // 如果平年二月天數(shù)超過28或小于1
  } else if (y % 4 != 0 && y % 100 != 0 && M == 2 && (d <= 0 || d > 28)) {
   Hint.HINT_TINY_TWOMONTH.print();
   return;
   // 如果平年二月天數(shù)超過28或小于1
  } else if (y % 400 != 0 && M == 2 && (d <= 0 || d > 28)) {
   Hint.HINT_TINY_TWOMONTH.print();
   return;
   // 如果閏年二月天數(shù)超過29或小于1
  } else if (y % 4 == 0 && y % 100 != 0 && M == 2 && (d <= 0 || d > 29)) {
   Hint.HINT_BIG_TWOMONTH.print();
   return;
   // 如果閏年二月天數(shù)超過29或小于1
  } else if (y % 400 == 0 && M == 2 && (d <= 0 || d > 29)) {
   Hint.HINT_BIG_TWOMONTH.print();
   return;
  } else {
   return;
  }

  }
  // 獲取時(shí)間
  String time = datetime[1];
  // 判斷是否以正則分割
  boolean b = spiltTime(time);
  // 如果沒有以正則分割
  if (!b) {
  Hint.HINT_TIME.print();
  return;
  } else {
  // 進(jìn)行日期和時(shí)間的拼接
  String dateTime = y + "-" + M + "-" + d + " " + time;
  DateTimeFormatter ofPattern1 = DateTimeFormatter.ofPattern("y-M-d H:m");
  LocalDateTime parse1 = LocalDateTime.parse(dateTime, ofPattern1);
  // System.out.println(parse1);
  // 判斷是不是當(dāng)年
  if (y == Year.now().getValue()) {
   // 判斷是不是當(dāng)月
   if (M == MonthDay.now().getMonthValue()) {
   // 判斷是不是今天
   if (d == MonthDay.now().getDayOfMonth()) {
    printMessage("今天 a H:m", parse1);
    // 判斷是不是昨天
   } else if (d - MonthDay.now().getDayOfMonth() == -1) {
    printMessage("今天 a H:m", parse1);
    // 判斷是不是明天
   } else if (d - MonthDay.now().getDayOfMonth() == 1) {
    printMessage("明天 a H:m", parse1);
    // 判斷一周內(nèi)的哪一天
   } else if (d - MonthDay.now().getDayOfMonth() >= -7
    && d - MonthDay.now().getDayOfMonth() <= -2) {
    printMessage("E a H:m", parse1);
    // 在當(dāng)月內(nèi),但不在本周
   } else {
         printMessage("M-d a H:m", parse1);
   }
   // 當(dāng)前年的其他月
   } else {
   printMessage("M-d H:m", parse1);
   }
   // 不同年的情況下
  } else {
  DateTimeFormatter dtf = DateTimeFormatter.ofLocalizedDateTime(FormatStyle.LONG, FormatStyle.SHORT);
   System.out.println(parse1.format(dtf) + "分");
  }

  }
 } else {
  return;
 }

 }

 /**
 * 獲取時(shí)間格式器并解析,打印時(shí)間
 * 
 * @param info     模式字符串
 * @param localDateTime LocalDateTime對象
 */
 private static void printMessage(String info, LocalDateTime localDateTime) {
 //把模式字符串傳進(jìn)去,獲取到DateTimeFormatter對象
 DateTimeFormatter ofPattern2 = DateTimeFormatter.ofPattern(info);
 //用LocalDateTime對象解析獲取到的DateTimeFormatter對象
 System.out.println(localDateTime.format(ofPattern2));
 }

 /**
 * 判斷大小閏年方法
 * 其中判斷了平年和閏年內(nèi)大月和小月的天數(shù)
 * @param y 年
 * @param m 月
 * @param d 日
 * @return true為閏年,false為平年
 */
 private static boolean handleDate(int y, int m, int d) {
 // 是閏年二月情況下
 if (y % 4 == 0 && y % 100 != 0 && m == 2 && (d > 0 && d <= 29)) {
  return true;
  // 是閏年二月情況下
 } else if (y % 400 == 0 && m == 2 && (d > 0 && d <= 29)) {
  return true;
  // 不是閏年,但是二月情況下
 } else if (y % 4 != 0 && y % 400 != 0 && m == 2 && (d > 0 && d <= 28)) {
  // 不是閏年,2月28天
  return true;
  // 不是二月,判斷是否是大月
 } else if (Arrays.binarySearch(bigMonth, m) > -1 && (d > 0 && d <= 31)) {
  return true;
  // 不是二月,判斷是否是小月
 } else if (Arrays.binarySearch(tinyMonth, m) > -1 && (d > 0 && d <= 30)) {
  return true;
 }
 return false;
 }

 /**
 * 使用正則表達(dá)式限定時(shí)間
 * 
 * @param time 需要分割的時(shí)間
 * @return 分割后的結(jié)果
 */
 private static boolean spiltTime(String time) {
 String t = "([01]?[0-9]{1}|[2][0-3]):[0-5]?[0-9]";
 return time.matches(t) ? true : false;
 }

 /**
 * 使用正則表達(dá)式限定日期
 * 
 * @param date 需要分割的日期
 * @return 分割后的日期
 */
 private static String[] splitDate(String date) {
 // 分割年月日
 String r = "[\\./-]{1}";
 // 月份出現(xiàn)的位數(shù)
 String s = "\\d{1,2}";
 return date.matches("\\d+" + "(" + r + s + "){2}") ? date.split(r) : new String[0];
 }

}

在實(shí)現(xiàn)類中,使用了正則表達(dá)式(一個(gè)強(qiáng)大的字符串處理工具)來限定日期和時(shí)間的分隔符(. / - : 四種),輸入類型(只能為數(shù)字)等,得到自己想要的日期時(shí)間格式,然后對平閏年進(jìn)行判斷,完成后再拼接時(shí)間.然后把獲取時(shí)間格式器和解析抽象成一個(gè)方法(因?yàn)橛玫暮芏?如果不抽象成方法,會(huì)顯得代碼很冗長),每次使用直接調(diào)用方法.最后在不同年的情況下,使用了不同風(fēng)格的時(shí)間格式化器完成格式化.

測試雖然沒有任何Bug,但跨年(比如19年的12月31號是20年的1月1號的昨天),跨月(比如20年1月31號為星期五,2月2號為星期日)沒有實(shí)現(xiàn),如需其他需要,請自行添加邏輯實(shí)現(xiàn).

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持億速云。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI