成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

Java 時(shí)間與日期處理

HelKyle / 2428人閱讀

摘要:時(shí)間與日期處理從屬于筆者的現(xiàn)代開發(fā)系列文章,涉及到的引用資料聲明在學(xué)習(xí)與實(shí)踐資料索引中。可以通過靜態(tài)構(gòu)造方法很容易的創(chuàng)建,定義了與之間的轉(zhuǎn)化關(guān)系時(shí)差類以年月日來表示日期差,而以秒與毫秒來表示時(shí)間差適用于處理與機(jī)器時(shí)間。

Java 時(shí)間與日期處理 從屬于筆者的現(xiàn)代 Java 開發(fā)系列文章,涉及到的引用資料聲明在 Java 學(xué)習(xí)與實(shí)踐資料索引中。

Java 時(shí)間與日期處理

在 Java 8 之前,我們最常見的時(shí)間與日期處理相關(guān)的類就是 Date、Calendar 以及 SimpleDateFormatter 等等。不過 java.util.Date 也是被詬病已久,它包含了日期、時(shí)間、毫秒數(shù)等眾多繁雜的信息,其內(nèi)部利用午夜 12 點(diǎn)來區(qū)分日期,利用 1970-01-01 來計(jì)算時(shí)間;并且其月份從 0 開始計(jì)數(shù),而且用于獲得年、月、日等信息的接口也是太不直觀。除此之外,java.util.DateSimpleDateFormatter 都不是類型安全的,而 JSR-310 中的 LocalDate LocalTime 等則是不變類型,更加適合于并發(fā)編程。JSR 310 實(shí)際上有兩個(gè)日期概念。第一個(gè)是 Instant,它大致對應(yīng)于 java.util.Date 類,因?yàn)樗砹艘粋€(gè)確定的時(shí)間點(diǎn),即相對于標(biāo)準(zhǔn) Java 紀(jì)元(1970年1月1日)的偏移量;但與 java.util.Date 類不同的是其精確到了納秒級別。另一個(gè)則是 LocalDate、LocalTime 以及 LocalDateTime 這樣代表了一般時(shí)區(qū)概念、易于理解的對象。

Class / Type Description
Year Represents a year.
YearMonth A month within a specific year.
LocalDate A date without an explicitly specified time zone.
LocalTime A time without an explicitly specified time zone.
LocalDateTime A combination date and time without an explicitly specified time zone.

最新 JDBC 映射將把數(shù)據(jù)庫的日期類型和 Java 8 的新類型關(guān)聯(lián)起來:

SQL Java
date LocalDate
time LocalTime
timestamp LocalDateTime
datetime LocalDateTime
時(shí)間與日期基礎(chǔ)概念 標(biāo)準(zhǔn)時(shí)間

GMT 即「格林威治標(biāo)準(zhǔn)時(shí)間」( Greenwich Mean Time,簡稱 G.M.T. ),指位于英國倫敦郊區(qū)的皇家格林威治天文臺的標(biāo)準(zhǔn)時(shí)間,因?yàn)楸境踝游缇€被定義為通過那里的經(jīng)線。然而由于地球的不規(guī)則自轉(zhuǎn),導(dǎo)致 GMT 時(shí)間有誤差,因此目前已不被當(dāng)作標(biāo)準(zhǔn)時(shí)間使用。UTC 是最主要的世界時(shí)間標(biāo)準(zhǔn),是經(jīng)過平均太陽時(shí)(以格林威治時(shí)間 GMT 為準(zhǔn))、地軸運(yùn)動修正后的新時(shí)標(biāo)以及以「秒」為單位的國際原子時(shí)所綜合精算而成的時(shí)間。UTC 比 GMT 來得更加精準(zhǔn)。其誤差值必須保持在 0.9 秒以內(nèi),若大于 0.9 秒則由位于巴黎的國際地球自轉(zhuǎn)事務(wù)中央局發(fā)布閏秒,使 UTC 與地球自轉(zhuǎn)周期一致。不過日常使用中,GMT 與 UTC 的功能與精確度是沒有差別的。協(xié)調(diào)世界時(shí)區(qū)會使用 “Z” 來表示。而在航空上,所有使用的時(shí)間劃一規(guī)定是協(xié)調(diào)世界時(shí)。而且 Z 在無線電中應(yīng)讀作 “Zulu”(可參見北約音標(biāo)字母),協(xié)調(diào)世界時(shí)也會被稱為 “Zulu time”。

TimeZone&UTC Offsets: 時(shí)區(qū)與偏移

人們經(jīng)常會把時(shí)區(qū)與 UTC 偏移量搞混,UTC 偏移量代表了某個(gè)具體的時(shí)間值與 UTC 時(shí)間之間的差異,通常用 HH:mm 形式表述。而 TimeZone 則表示某個(gè)地理區(qū)域,某個(gè) TimeZone 中往往會包含多個(gè)偏移量,而多個(gè)時(shí)區(qū)可能在一年的某些時(shí)間有相同的偏移量。譬如 America/Chicago, America/Denver, 以及 America/Belize 在一年中不同的時(shí)間都會包含 -06:00 這個(gè)偏移。

時(shí)間戳

Unix 時(shí)間戳表示當(dāng)前時(shí)間到 1970 年 1 月 1 日 00:00:00 UTC 對應(yīng)的秒數(shù)。注意,JavaScript 內(nèi)的時(shí)間戳指的是當(dāng)前時(shí)間到 1970 年 1 月 1 日 00:00:00 UTC 對應(yīng)的毫秒數(shù),和 Unix 時(shí)間戳不是一個(gè)概念,后者表示秒數(shù),差了 1000 倍。

時(shí)間數(shù)字字符串格式 RFC2822
YYYY/MM/DD HH:MM:SS ± timezone(時(shí)區(qū)用4位數(shù)字表示)
// eg 1992/02/12 12:23:22+0800
ISO 8601

國際標(biāo)準(zhǔn)化組織的國際標(biāo)準(zhǔn) ISO 8601 是日期和時(shí)間的表示方法,全稱為《數(shù)據(jù)存儲和交換形式·信息交換·日期和時(shí)間的表示方法》。目前最新為第三版 ISO8601:2004,第一版為 ISO8601:1988,第二版為 ISO8601:2000。年由 4 位數(shù)組成,以公歷公元 1 年為 0001 年,以公元前 1 年為 0000 年,公元前 2 年為 -0001 年,其他以此類推。應(yīng)用其他紀(jì)年法要換算成公歷,但如果發(fā)送和接受信息的雙方有共同一致同意的其他紀(jì)年法,可以自行應(yīng)用。

YYYY-MM-DDThh:mm:ss ± timezone(時(shí)區(qū)用HH:MM表示)
1997-07-16T08:20:30Z
// “Z”表示UTC標(biāo)準(zhǔn)時(shí)區(qū),即"00:00",所以這里表示零時(shí)區(qū)的`1997年7月16日08時(shí)20分30秒`
//轉(zhuǎn)換成位于東八區(qū)的北京時(shí)間則為`1997年7月17日16時(shí)20分30秒`
1997-07-16T19:20:30+01:00
// 表示東一區(qū)的1997年7月16日19時(shí)20秒30分,轉(zhuǎn)換成UTC標(biāo)準(zhǔn)時(shí)間的話是1997-07-16T18:20:30Z
時(shí)間戳

在 Java 8 之前,我們使用 java.sql.Timestamp 來表示時(shí)間戳對象,可以通過以下方式創(chuàng)建與獲取對象:

// 利用系統(tǒng)標(biāo)準(zhǔn)時(shí)間創(chuàng)建
Timestamp timestamp = new Timestamp(System.currentTimeMillis());

// 從 Date 對象中創(chuàng)建
new Timestamp((new Date()).getTime());

// 獲取自 1970-01-01 00:00:00 GMT 以來的毫秒數(shù)
timestamp.getTime();

在 Java 8 中,即可以使用 java.time.Instant 來表示自從 1970-01-01T00:00:00Z 之后經(jīng)過的標(biāo)準(zhǔn)時(shí)間:

// 基于靜態(tài)函數(shù)創(chuàng)建
Instant instant = Instant.now();

// 基于 Date 或者毫秒數(shù)轉(zhuǎn)換
Instant someInstant = someDate.toInstant();
Instant someInstant = Instant.ofEpochMilli(someDate.getTime());

// 基于 TimeStamp 轉(zhuǎn)換
Instant instant = timestamp.toInstant();

// 從 LocalDate 轉(zhuǎn)化而來
LocalDate.now().atStartOfDay().toInstant(ZoneOffset.UTC)

// 從 LocalDateTime 轉(zhuǎn)化而來
ldt.atZone(ZoneId.systemDefault()).toInstant();

// 獲取毫秒
long timeStampMillis = instant.toEpochMilli();

// 獲取秒
long timeStampSeconds = instant.getEpochSecond();

Clock 方便我們?nèi)プx取當(dāng)前的日期與時(shí)間。Clock 可以根據(jù)不同的時(shí)區(qū)來進(jìn)行創(chuàng)建,并且可以作為System.currentTimeMillis()的替代。這種指向時(shí)間軸的對象即是Instant類。Instants 可以被用于創(chuàng)建java.util.Date對象。

Clock clock = Clock.systemDefaultZone();
long millis = clock.millis();

Instant instant = clock.instant();
Date legacyDate = Date.from(instant);   // legacy java.util.Date
Date
// 默認(rèn)創(chuàng)建
Date date0 = new Date();

// 從 TimeStamp 中創(chuàng)建
Date date1 = new Date(time);

// 基于 Instant 創(chuàng)建
Date date = Date.from(instant);

// 從格式化字符串中獲取
SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd");
java.util.Date dt=sdf.parse("2005-2-19");

// 從 LocalDateTime 中轉(zhuǎn)化而來
Date out = Date.from(ldt.atZone(ZoneId.systemDefault()).toInstant());

基于 Date 的日期比較常常使用以下方式:

使用 getTime() 方法獲取兩個(gè)日期(自1970年1月1日經(jīng)歷的毫秒數(shù)值),然后比較這兩個(gè)值。

使用方法 before(),after() 和 equals()。例如,一個(gè)月的12號比18號早,則 new Date(99, 2, 12).before(new Date (99, 2, 18)) 返回true。

使用 compareTo() 方法,它是由 Comparable 接口定義的,Date 類實(shí)現(xiàn)了這個(gè)接口。

Calendar

Date 用于記錄某一個(gè)含日期的、精確到毫秒的時(shí)間。重點(diǎn)在代表一剎那的時(shí)間本身。 Calendar 用于將某一日期放到歷法中的互動——時(shí)間和年、月、日、星期、上午、下午、夏令時(shí)等這些歷法規(guī)定互相作用關(guān)系和互動。我們可以通過 Calendar 內(nèi)置的構(gòu)造器來創(chuàng)建實(shí)例:

Calendar.Builder builder =new Calendar.Builder();
Calendar calendar1 = builder.build();
Date date = calendar.getTime();

在 Calendar 中我們則能夠獲得較為直觀的年月日信息:

// 2017,不再是 2017 - 1900 = 117
int year =calendar.get(Calendar.YEAR);

int month=calendar.get(Calendar.MONTH)+1;

int day =calendar.get(Calendar.DAY_OF_MONTH);

int hour =calendar.get(Calendar.HOUR_OF_DAY);

int minute =calendar.get(Calendar.MINUTE);

int seconds =calendar.get(Calendar.SECOND);    

除此之外,Calendar 還提供了一系列 set 方法來允許我們動態(tài)設(shè)置時(shí)間,還可以使用 add 等方法進(jìn)行日期的加減。

SimpleDateFormat

SimpleDateFormat 用來進(jìn)行簡單的數(shù)據(jù)格式化轉(zhuǎn)化操作:

Date dNow = new Date( );
SimpleDateFormat ft = new SimpleDateFormat ("E yyyy.MM.dd "at" hh:mm:ss a zzz");
LocalDateTime LocalDate
// 取當(dāng)前日期:
LocalDate today = LocalDate.now();

// 根據(jù)年月日取日期,12月就是12:
LocalDate crischristmas = LocalDate.of(2017, 5, 15); 

// 根據(jù)指定格式字符串取
LocalDate endOfFeb = LocalDate.parse("2017-05-15"); // 嚴(yán)格按照ISO yyyy-MM-dd驗(yàn)證,02寫成2都不行,當(dāng)然也有一個(gè)重載方法允許自己定義格式
LocalDate.parse("2014-02-29"); // 無效日期無法通過:DateTimeParseException: Invalid date

// 通過自定義時(shí)間字符串格式獲取
DateTimeFormatter germanFormatter =
    DateTimeFormatter
        .ofLocalizedDate(FormatStyle.MEDIUM)
        .withLocale(Locale.GERMAN);

LocalDate xmas = LocalDate.parse("24.12.2014", germanFormatter);
System.out.println(xmas);   // 2014-12-24

// 獲取其他時(shí)區(qū)下日期
LocalDate localDate = LocalDate.now(ZoneId.of("GMT+02:30"));

// 從 LocalDateTime 中獲取實(shí)例
LocalDateTime localDateTime = LocalDateTime.now();
LocalDate localDate = localDateTime.toLocalDate();
日期操作
// 取本月第1天
LocalDate firstDayOfThisMonth = today.with(TemporalAdjusters.firstDayOfMonth()); // 2014-12-01

// 取本月第2天
LocalDate secondDayOfThisMonth = today.withDayOfMonth(2); // 2014-12-02

// 取本月最后一天,再也不用計(jì)算是28,29,30還是31
LocalDate lastDayOfThisMonth = today.with(TemporalAdjusters.lastDayOfMonth()); // 2014-12-31

// 取下一天
LocalDate firstDayOf2015 = lastDayOfThisMonth.plusDays(1); // 變成了2015-01-01

// 取2015年1月第一個(gè)周一
LocalDate firstMondayOf2015 = LocalDate.parse("2015-01-01").with(TemporalAdjusters.firstInMonth(DayOfWeek.MONDAY)); // 2015-01-05
LocalTime
// 獲取其他時(shí)區(qū)下時(shí)間
LocalTime localTime = LocalTime.now(ZoneId.of("GMT+02:30"));

// 從 LocalDateTime 中獲取實(shí)例
LocalDateTime localDateTime = LocalDateTime.now();
LocalTime localTime = localDateTime.toLocalTime();


- 12:00
- 12:01:02
- 12:01:02.345
LocalDateTime
// 通過時(shí)間戳創(chuàng)建
LocalDateTime localDateTime = LocalDateTime.ofInstant(Instant.ofEpochSecond(1450073569l), TimeZone.getDefault().toZoneId());

// 通過 Date 對象創(chuàng)建
Date in = new Date();
LocalDateTime ldt = LocalDateTime.ofInstant(in.toInstant(), ZoneId.systemDefault());

// 通過解析時(shí)間字符串創(chuàng)建
DateTimeFormatter formatter =
    DateTimeFormatter
        .ofPattern("MMM dd, yyyy - HH:mm");

LocalDateTime parsed = LocalDateTime.parse("Nov 03, 2014 - 07:13", formatter);
String string = formatter.format(parsed);
System.out.println(string);     // Nov 03, 2014 - 07:13

獲取年、月、日等信息

LocalDateTime sylvester = LocalDateTime.of(2014, Month.DECEMBER, 31, 23, 59, 59);

DayOfWeek dayOfWeek = sylvester.getDayOfWeek();
System.out.println(dayOfWeek);      // WEDNESDAY

Month month = sylvester.getMonth();
System.out.println(month);          // DECEMBER

long minuteOfDay = sylvester.getLong(ChronoField.MINUTE_OF_DAY);
System.out.println(minuteOfDay);    // 1439

時(shí)間格式化展示

DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm");
LocalDateTime dateTime = LocalDateTime.of(1986, Month.APRIL, 8, 12, 30);
String formattedDateTime = dateTime.format(formatter); // "1986-04-08 12:30"
時(shí)間操作
localDateTime.plusDays(1);
localDateTime.minusHours(2);
時(shí)區(qū)轉(zhuǎn)換

Timezones 以 ZoneId 來區(qū)分??梢酝ㄟ^靜態(tài)構(gòu)造方法很容易的創(chuàng)建,Timezones 定義了 Instants 與 Local Dates 之間的轉(zhuǎn)化關(guān)系:

System.out.println(ZoneId.getAvailableZoneIds());
// prints all available timezone ids

ZoneId zone1 = ZoneId.of("Europe/Berlin");
ZoneId zone2 = ZoneId.of("Brazil/East");
System.out.println(zone1.getRules());
System.out.println(zone2.getRules());

// ZoneRules[currentStandardOffset=+01:00]
// ZoneRules[currentStandardOffset=-03:00]
LocalDateTime ldt = ...
ZonedDateTime zdt = ldt.atZone(ZoneId.systemDefault());
Date output = Date.from(zdt.toInstant());
ZoneId losAngeles = ZoneId.of("America/Los_Angeles");
ZoneId berlin = ZoneId.of("Europe/Berlin");

// 2014-02-20 12:00
LocalDateTime dateTime = LocalDateTime.of(2014, 02, 20, 12, 0);

// 2014-02-20 12:00, Europe/Berlin (+01:00)
ZonedDateTime berlinDateTime = ZonedDateTime.of(dateTime, berlin);

// 2014-02-20 03:00, America/Los_Angeles (-08:00)
ZonedDateTime losAngelesDateTime = berlinDateTime.withZoneSameInstant(losAngeles);

int offsetInSeconds = losAngelesDateTime.getOffset().getTotalSeconds(); // -28800

// a collection of all available zones
Set allZoneIds = ZoneId.getAvailableZoneIds();

// using offsets
LocalDateTime date = LocalDateTime.of(2013, Month.JULY, 20, 3, 30);
ZoneOffset offset = ZoneOffset.of("+05:00");

// 2013-07-20 03:30 +05:00
OffsetDateTime plusFive = OffsetDateTime.of(date, offset);

// 2013-07-19 20:30 -02:00
OffsetDateTime minusTwo = plusFive.withOffsetSameInstant(ZoneOffset.ofHours(-2));
時(shí)差

Period 類以年月日來表示日期差,而 Duration 以秒與毫秒來表示時(shí)間差;Duration 適用于處理 Instant 與機(jī)器時(shí)間。

// periods

LocalDate firstDate = LocalDate.of(2010, 5, 17); // 2010-05-17
LocalDate secondDate = LocalDate.of(2015, 3, 7); // 2015-03-07
Period period = Period.between(firstDate, secondDate);

int days = period.getDays(); // 18
int months = period.getMonths(); // 9
int years = period.getYears(); // 4
boolean isNegative = period.isNegative(); // false

Period twoMonthsAndFiveDays = Period.ofMonths(2).plusDays(5);
LocalDate sixthOfJanuary = LocalDate.of(2014, 1, 6);

// add two months and five days to 2014-01-06, result is 2014-03-11
LocalDate eleventhOfMarch = sixthOfJanuary.plus(twoMonthsAndFiveDays);


// durations

Instant firstInstant= Instant.ofEpochSecond( 1294881180 ); // 2011-01-13 01:13
Instant secondInstant = Instant.ofEpochSecond(1294708260); // 2011-01-11 01:11

Duration between = Duration.between(firstInstant, secondInstant);

// negative because firstInstant is after secondInstant (-172920)
long seconds = between.getSeconds();

// get absolute result in minutes (2882)
long absoluteResult = between.abs().toMinutes();

// two hours in seconds (7200)
long twoHoursInSeconds = Duration.ofHours(2).getSeconds();

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/67466.html

相關(guān)文章

  • 20 個(gè)案例教你在 Java 8 中如何處理日期時(shí)間?

    摘要:前言前面一篇文章寫了如何安全的使用里面介紹了如何處理日期時(shí)間,以及如何保證線程安全,及其介紹了在中的處理時(shí)間日期默認(rèn)就線程安全的類。引入了全新的日期時(shí)間格式工具,線程安全而且使用方便。 前言 前面一篇文章寫了《SimpleDateFormat 如何安全的使用?》, 里面介紹了 SimpleDateFormat 如何處理日期/時(shí)間,以及如何保證線程安全,及其介紹了在 Java 8 中的處...

    Rango 評論0 收藏0
  • Java8新特性之日期處理

    摘要:處理日期日歷和時(shí)間的不足之處將設(shè)定為可變類型,以及的非線程安全使其應(yīng)用非常受限。最完整的日期時(shí)間,包含時(shí)區(qū)和相對或格林威治的時(shí)差。獲取當(dāng)前的日期中的用于表示當(dāng)天日期。 簡介 伴隨 lambda表達(dá)式、streams 以及一系列小優(yōu)化,Java 8 推出了全新的日期時(shí)間API。 Java處理日期、日歷和時(shí)間的不足之處:將 java.util.Date 設(shè)定為可變類型,以及 SimpleD...

    Airy 評論0 收藏0
  • 全面解析Java日期時(shí)間API

    摘要:類是一個(gè)抽象類,它為特定瞬間與一組諸如等日歷字段之間的轉(zhuǎn)換提供了一些方法,并為操作日歷字段例如獲得下星期的日期提供了一些方法。是不變的日期時(shí)間對象代表一個(gè)日期時(shí)間,往往被視為年月日時(shí)分秒。獲得實(shí)例的日期和時(shí)間。 時(shí)區(qū) GMT(Greenwich Mean Time):格林尼治時(shí)間,格林尼治標(biāo)準(zhǔn)時(shí)間的正午是指當(dāng)太陽橫穿格林尼治子午線時(shí)(也就是在格林尼治上空最高點(diǎn)時(shí))的時(shí)間。 UTC(U...

    robin 評論0 收藏0
  • Java8實(shí)戰(zhàn)》-第十二章筆記(新的日期時(shí)間API)

    摘要:合并日期和時(shí)間這個(gè)復(fù)合類名叫,是和的合體。截至目前,我們介紹的這些日期時(shí)間對象都是不可修改的,這是為了更好地支持函數(shù)式編程,確保線程安全,保持領(lǐng)域模式一致性而做出的重大設(shè)計(jì)決定。 新的日期和時(shí)間API Java的API提供了很多有用的組件,能幫助你構(gòu)建復(fù)雜的應(yīng)用。不過,Java API也不總是完美的。我們相信大多數(shù)有經(jīng)驗(yàn)的程序員都會贊同Java 8之前的庫對日期和時(shí)間的支持就非常不理想...

    khlbat 評論0 收藏0
  • 樂字節(jié)-Java8新特性之Date API

    摘要:上一篇文章,小樂給大家?guī)砹诵绿匦灾?,接下來本文將會給大家介紹新特性之前言通過發(fā)布新的來進(jìn)一步加強(qiáng)對日期與時(shí)間的處理。 上一篇文章,小樂給大家?guī)砹薐ava8新特性之Optional,接下來本文將會給大家介紹Java8新特性之Date API 前言: Java 8通過發(fā)布新的Date-Time API來進(jìn)一步加強(qiáng)對日期與時(shí)間的處理。 舊版的 Java 中,日期時(shí)間 API 存在諸多問題...

    suemi 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<