Upgrade to Pro — share decks privately, control downloads, hide ads and more …

Java 8 Date & Time API (in Lithuanian) - Vilniu...

Java 8 Date & Time API (in Lithuanian) - Vilnius JUG #25

Talk about Java 8 Date/Time API updates given at Vilnius jug MEETUP #25

Vaidas Pilkauskas

March 12, 2014
Tweet

More Decks by Vaidas Pilkauskas

Other Decks in Programming

Transcript

  1. aš • programuotojas • pagrindinė kalba - Java • domiuosi

    kitomis kalbomis - Scala, JavaScript, Python, Ruby • Kai neprogramuoju - dviračiai, snieglentė, roko muzika, šiuolaikinis menas, Vilnius JUG https://www.google.com/+VaidasPilkauskas
  2. Legal This work is licensed under the Creative Commons Attribution

    3.0 Unported License. To view a copy of this license, visit http://creativecommons.org/licenses/by/3.0/ or send a letter to Creative Commons, 444 Castro Street, Suite 900, Mountain View, California, 94041, USA. Slides 10-62 are derived from The new JDK 1.8 Date and Time API – JSR-310 by Stephen Colebourne given at JAX 2013. All the other slides are based on various publicly available internet resources listed at the end in the reference list.
  3. Kodėl reikalingas naujas Date/Time API? java.util.Date is a testament to

    the fact that even brilliant programmers can screw up. java.util.Calendar, which Sun licensed to rectify the Date mess, is a testament to the fact that average programmers can screw up, too. http://stackoverflow.com/questions/1571265/why-is-the-java-date-api-java-util- date-calendar-such-a-mess
  4. java.util.Date • Mutuojamas - galima keisti reikšmę • Milisekundžių tikslumas

    nepakankamas • new Date().getYear() //114 • Tinkamai nepalaiko laiko juostų • Nėra patogių priemonių dirbti su intervalais • Metai prasideda nuo 1900, mėnesio diena nuo 1, visa kita nuo 0) Date date = new Date(1L); date.toString(); //Thu Jan 01 02:00:00 EET 1970
  5. Nesilaikoma equals kontrakto Date date = new Date(); Timestamp timeStamp

    = new Timestamp(date.getTime()); date.equals(timeStamp); // true timeStamp.equals(date); // false
  6. java.util.Calendar • Taip pat mutuojamas • Nepatogus API • Sunku

    pridėti naujus kalendorius • Trūksta getterių paprastai datų informacijai Calendar c = Calendar.getInstance(); int weekday = c.get(DAY_OF_WEEK); Date d = c.getTime();
  7. Joda-Time Java ekosistemoje suvokiama kaip “Kokybiška datos ir laiko biblioteka”

    Tai kodėl ne Joda-Time yra Java 8 Date/Time API? Nes nepakankamai tobula. “Well, there is one key reason - Joda-Time has design flaws.“ - Stephen Colebourne http://blog.joda.org/2009/11/why-jsr-310-isn-joda-time_4941.html
  8. JSR 310 • Nauja datos ir laiko biblioteka • Sukurta

    visiškai nuo pradžių • Įtakota Joda-Time, speco vienas iš vadovų - Stephen Colebourne • Tai atviro kodo JSR projektas (palyginimui Joda-Time - atviro kodo biblioteka) ◦ http://threeten.github.com/ ◦ Originalus kodas BSD (3 clause) licencija ◦ OpenJDK kodas GPL ◦ virš 20000 testų • Yra nuportintas JDK7
  9. Tikslai • Išsamus datos/laiko modelis • Dažnai sutinkamų kalendorių palaikymas

    • Tipizavimas ◦ vengti primityvų, kai yra prasmė ◦ save dokumentuojantis ◦ draugiškas IDE • “Sutariantis” su egzistuojančiomis JDK datų/laiko klasėmis • Atsižvelgti į XML ir SQL poreikius
  10. Projektavimo principai • Nekeičiamumas (Immutability) ◦ thread-safe, tinka kešuoti, funkcinis

    programavimas • Fluent API ◦ lengvai skaitomas, tarsi įprastas sakinys • Švarus, aiškus, tikėtinas ◦ gerai apibrėžti metodai, paprastas JavaDoc ◦ stipri būsenos kontrolė • Plečiamas ◦ JSR/JDK autoriai visko nežino
  11. Per daug savokų • Tiek daug laiko savokų • Pats

    “laikas” turi daug reikšmių • Reikia apibrėžti nuoseklią srities kalbą!
  12. Du pagrindiniai reikalavimai • Tęstinis kompiuterio laikas ◦ vienas didėjantis

    skaičius ◦ suprojektuotas kompiuteriui • Žmogaus ◦ paremtas laukeliais ◦ metai, mėnuo, diena, valanda, minutė, sekundė
  13. Momentas - Instant • Vienas momentinis taškas laiko linijoje •

    Įvykio Timestampas • Java class - Instant ◦ matuojamas nanosekundėmis nuo 1970-01-01 ◦ apima visatos amžiaus įvykius ◦ problema - nėra tinkamo primityvo - 96 bitų
  14. Trukmė - Duration • Laiko “kiekis” (amount) • Nepriklauso nuo

    laiko linijos • Java class - Duration ◦ matuojamas nanosekundėmis ◦ reikia 96 bitų
  15. // instant Instant start = Instant.ofEpochMilli(123450L); Instant end = Instant.now();

    assert start.isBefore(end); assert end.isAfter(start); // duration Duration duration = Duration.ofSeconds(12); Duration bigger = duration.multipliedBy(4); Duration biggest = bigger.plus(duration); // combination Instant later = start.plus(duration); Instant earlier = start.minus(duration);
  16. Realybė • SI sekundė fiksuoto ilgio • Diena ilgiai gali

    būti skirtingi • Astronomai sako, kad Žemė lėtėja • Diena != (24 valandos * 60 minutės * 60 SI sekundės) • Realiai dienos ilgesnės nei 86400 SI sekundės • UTC įterpia keliamąją sekundę, kad pataisytų dienos ilgį • Kompiuteriai prastai palaiko keliamąsias sekundes
  17. Java milisekundės • Java skaičiuoja milisekundes nuo 1970-01- 01 •

    Ką tai reiškia? • Apibrėžimas netikslus, nes • UTC pradėtas tik 1972-01-01 ◦ 1970-01-01 taigi, pradžios taškas labai nepatogus • Apibrėžimas - 86400 sekundės per dieną ◦ kas atsitinka toje vietoje, kur įterpiama keliamoji sekundė?
  18. Pasirinktas būdas • Apibrėžta Java laiko skalė, ir susieta su

    civiliniu laiku • Vidurdienis tiksliai atitinka civilinį laiką • Visas kitas laikas kiek įmanoma tiksliau • Diena padalinta į 86400 daleles - “sekundes” • Java epocha prasideda 1970-01-01 • Tai reiškia, kad programuotojai gali ignoruoti keliamąsias sekundes • Nieko naujo, tai UTC-SLS • Tai tik apibrėžimas, viskas priklauso nuo OS
  19. Žmogaus skalė • Žmogaus data/laikas ◦ paremtas laukeliais ◦ metai,

    mėnuo, diena, minutė, sekundė ◦ 7 dienų savaitė, diena - 24 valandos • Reikalavimai ◦ Data ir laikas ◦ Data be laiko ◦ Laikas be datos ◦ Laiko juostos skirtumas nuo UTC ◦ Laiko zona
  20. Vietos data - Local date • Data be laiko ir

    laiko zonos • pvz., gimtadienis • pvz., šventinė diena, atostogų pradžia • Java klasė - LocalDate ◦ saugomi metai-mėnuo-diena // 2014-03-12
  21. Vietos laikas - Local time • Laikas be datos ir

    laiko zonos • pvz., parduotuvės atidarymo laikas • pvz., laikas sieniniame laikrodyje • Java klasė - LocalTime ◦ saugoma valanda-minutė-sekundė-nanosekundė // 20:30
  22. Vietos data su laiku - Local date-time • Data su

    laiku be laiko zonos • pvz., vietinis laikas, kada kyla lėktuvas • Java klasė - LocalDateTime ◦ apjungia LocalDate & LocalTime // 2014-03-12T20:30
  23. Zonos poslinkis - Zone-offset • Poslinkis valandomis ir minutėmis nuo

    UTC • Lietuva priekyje UTC - teigiama reikšmė • JAV atsilieka nuo UTC - neigiama reikšmė • Java klasė - ZoneOffset ◦ saugo sekundes -18:00 to +18:00 diapazone // +01:00
  24. Data ir laikas su poslinkiu - Offset date-time • Data

    ir laikas su zonos poslinkiu • pvz., audito įrašo timestampas su laiko zona • Tas pats momentas - Instant, tik su papildoma poslinkio informacija • Java class - OffsetDateTime ◦ apjungia LocalDateTime & ZoneOffset // 2014-03-12T20:30+02:00
  25. Kitos klasės • Year • YearMonth • MonthDay • Month

    - enum • DayOfWeek - enum • OffsetTime - local time + zone-offset
  26. Užklausos • Getteriai • Metodai grąžina geriausią tipą tam laukui

    ◦ daugumai laukų - primityvai ◦ enumai savaitės dienai, mėnesiui LocalDate date = LocalDate.of(2010, 12, 3); int year = date.getYear(); Month month = date.getMonth(); int dom = date.getDayOfMonth(); boolean leap = date.isLeapYear(); int monthLen = date.lengthOfMonth();
  27. Atnaujinimai • Beveik viskas - nekintama (nemutuojama) • Naudojamas 'with'

    metodas vietoje ‘set’ ◦ nekeičiamas originalus objektas ◦ grąžinama nauja, patobulinta kopija ◦ norint toliau naudoti, reikia išsisaugoti LocalDate date = LocalDate.of(2010,Month.DECEMBER,3); LocalDate later = date.withYear(2011); LocalDate between = date.with(Month.MAY);
  28. Derintojai - Adjusters • Sudėtingesni pakeitimai gali būti atliekami Adjusterių

    pagalba, pvz.: ◦ iš turimos datos gauti paskutinę mėnesio dieną ◦ iš turimos datos gauti kitą antrą trečiadienį (kada kitas JUGas?) public interface TemporalAdjuster { Temporal adjustInto(Temporal input); } LocalDate adjusted = date.with(lastDayOfMonth()); LocalDate adjusted = date.with(next2ndWednesday());
  29. Laiko zonų apžvalga • Pasaulis padalintas įvairias laiko zonas •

    Vietovės laikas apibrėžiamas politinių taisyklių ◦ Sirija pakeičia vasaros laiką (DST) su 3 dienų perspėjimu ◦ Vakarų Australija turėjo 3 metų DST eksperimentą ◦ Brazilija keičia DST kasmet ◦ Egiptas turėjo du DST periodus 2010 • Taisyklės apibrėžia laiko zonos poslinkio pasikeitimą ◦ "Žiemą Lietuva bus 2 valandomis priekyje nuo UTC, o vasarą jau 3. Laikas persukamas viena valanda į priekį paskutinį kovo sekmadienį ir atsukamas atgal paskutinį spalio sekmadienį."
  30. Laiko zonų analizė • Keletas grupių renka ir teikia informaciją

    apie TZ taisykles ◦ IANA "Timezone database" (TZDB) ◦ Windows ◦ IATA - Tarptautinė oro transporto asociacija • Kiekviena grupė apibrėžia savo identifikatorių • TZDB naudoja 'Europe/Vilnius' Lietuvai • Taisyklės apibrėžia ilgalaikius DST pakeitimus • bei kada keičiasi poslinkis
  31. Laiko zonų dizainas • JDK javadoc: ◦ “TimeZone represents a

    time zone offset, and also figures out daylight savings” • JSR-310 atskiria atsakomybes ◦ ZoneOffset – nuo +14:00 iki -12:00 ◦ ZoneRules – poslinkių keitimo taisyklės ◦ ZoneId – zonos identifikatorius, identifikuoja vietą/valdžią • Tik viena papildoma datos/laiko klasė ◦ ZonedDateTime
  32. Zonų laiko tarpai ir persidengimai • Pavasario DST “tarpas” •

    Rudens DST “persidengimas” • JSR-310 nemeta exceptionų, o Joda-Time meta ◦ prasmingas datos interpretavimas ◦ Pagal nutylėjimą - vasaros laikas // one day before DST ends (overlap of one hour) ZoneId zone = ZoneId.of("Europe/London"); ZonedDateTime dt = ZonedDateTime.of(2010, 10, 27, 1, 30, 0, 0, zone); // dt = 2010-10-27 01:30 +01:00 dt = dt.plusDays(1); // dt = 2010-10-28 01:30 +01:00 // retains the offset where possible dt = dt.plusHours(1); // dt = 2010-10-28 01:30 +00:00 // offset changes, same time
  33. Galima pasitikrinti ZoneId zone = ZoneId.of("Europe/London"); LocalDateTime ldt = LocalDateTime.of

    (2010,10,27,1,30); ZoneOffsetTransition trn = zone.getRules(). getTransition(ldt); if (trn != null) { if (trn.isGap()) { ... } if (trn.isOverlap()) { ... } }
  34. Klasių santrauka • LocalDate 2010-12-03 • LocalTime 11:05:30 • LocalDateTime

    2010-12-03T11:05:30 • OffsetTime 11:05:30+01:00 • OffsetDateTime 2010-12-03T11:05:30+01:00 • ZonedDateTime 2010-12-03T11:05:30+01:00 Europe/Paris • Year 2010 • YearMonth 2010-12 • MonthDay 12-03 • Instant 2576458258.266 epoch- seconds
  35. Egzistuojančios JDK klasės • Date, Calendar, SQL kalsės ir TimeZone

    ◦ konvertavimo metodai į ir iš JSR-310 tipus ◦ ne deprecated :( • JSR-310 nepriklauso nuo senųjų klasių (teoriškai reiškia, kad kažkada galima bus išmesti senąsias klases)
  36. ISO-8601 • Visos klasės ir toString() paremtos ISO-8601 • Pagrįstas

    Grigaliaus kalendoriumi • Nusako, kaip skaičiais atvaizduoti datas/laikus • Kitų internetinių standartų bazė, tokių kaip XML ◦ XML schema gerai atitinka JSR-310 klases
  37. SQL duomenų bazės • Klasės atitinka SQL ◦ LocalDate DATE

    ◦ LocalTime TIME WITHOUT TIME ZONE ◦ LocalDateTime TIMESTAMP WITHOUT TIMEZONE ◦ OffsetTime TIME WITH TIME ZONE ◦ OffsetDateTime TIMESTAMP WITH TIME ZONE • JDBC atnaujintas, kad atitiktų JSR-310
  38. Calendar systems • Bazinis kalendorius paremtas ISO-8601 ◦ dabartinis civilinis

    kalendorius ◦ istoriškai netikslus ▪ katalikai - 1582 ▪ britai - 1752 ▪ rusai - 1917? • Reikalavimai ◦ JDK turi palaikyti įprastus kalendorius ◦ kalendorių palaikymas neturi pasunkinti API naudojimo ◦ nedviprasmiškas API
  39. Regioninės kalendorių sistemos • Dažniausiai naudojami ne ISO kalendoriai ◦

    Islamo, Japonų, Kinų, Budizmo ir kiti • Pasiekiami pagal pavadinimą • Pasiekiami pagal BCP-47 lokalės išplėtimą ◦ th-TH-u-ca-buddhist • Datos – era, metai, mėnuo, diena ◦ mėnesių ilgiai & tvarka skiriasi • Leidžiamos tik datos variacijos ◦ ignoruojami kalendoriai pradedantys dieną skirtingu metu (saulėtekis/saulėlydis)
  40. Kalendorių chronologijos • Chronology apibrėžia kalendoriaus sistemą ◦ factory’is kurti

    datoms • ChronoLocalDate • ChronoLocalDateTime • ChronoZonedDateTime ◦ laikas yra standartinis LocalTime
  41. Here be dragons! • Dirbti su kitais kalendoriais sudėtinga •

    Daug ISO prielaidų negalioja • Lengva suklysti • Būtina daryti kodo peržiūras! • Taigi...
  42. Here be dragons! • Naudokite ISO, ne Chrono klases •

    See ChronoLocalDate Javadoc paaiškinimą • Nesaugokite kalendoriaus visoje sistemoje • Kalendorius naudotojo profilio dalis • Konvertuokite atvaizduodami UI
  43. Elementų kombinavimas • Lengva apjungti datos ir laiko objektus •

    Naudojant ‘at’ • taip vadinamas Fluent API LocalDate date = Year.of(2013).atMonth(MARCH).atDay(27); LocalDateTime dt = date.atTime(12, 30); OffsetDateTime odt = dt.atOffset(ZoneOffset.ofHours(2)); ZoneId london = ZoneId.of("Europe/London"); ZonedDateTime zdt1 = date.atStartOfDay(london); ZonedDateTime zdt2 = Instant.now().atZone(london);
  44. Periodai - Periods • Datos pagrindo laiko “kiekis” pasiekiamas žmogiškais

    laukais ◦ 6 metai, 2 mėnesiai ir 12 dienų • Panaudojimo pvz.: ◦ konferencijos ilgis – 5 dienos ◦ nėštumas – 9 mėnesiai • Java klasė - Period ◦ laukeliai metams, mėnesiams, dienoms • galima atimti/pridėti prie datos/laiko • Duration klasė - laiko ekvivalentas
  45. Dabartinis laikas - Current time • Priklauso nuo laiko zonos

    ◦ dažnai tai pamirštame • Reikalavimai ◦ sustabdyti laiką testuose ◦ pakeisti laiką į praeitį/ateitį ◦ pristabdyti/paspartinti
  46. Dabartinis laikas - Current time • Pasiekti esamą laiką naudojant

    objektą ◦ vengti Singleton ◦ Inversion of Control • Galima implementuoti subklasę ◦ taigi - kontroliuoti laiką :) // system millis, default time zone Instant instant = Instant.now(Clock.system()); LocalDate date = LocalDate.now(Clock.system()); // system millis, specified time zone ZoneId zone = ZoneId.of("Europe/Moscow"); LocalDate date = LocalDate.now(Clock.system(zone));
  47. Dabartinis laikas - Current time • Palaiko Inversion of Control

    ◦ inject Clock ◦ galima turėti 'stop time' subklasę testavimui public class MyForm { @Inject private Clock clock; // inject with Spring/Guice public void validate(LocalDate date) { if (date.isBefore(LocalDate.now(clock))) { // error } } }
  48. Formatting and Parsing • toString() grąžina ISO-8601 • Formatavimas/parsinimas palaiko

    šablonus ◦ kaip kad SimpleDateFormat ◦ taip pat sudėtingesnius formatus • pagrindinė formatavimo klasė - DateTimeFormatter • Pateikia daug įprastų formaterių
  49. Amount between • Lengva suskaičiuoti laikotarpį tarp dviejų datų/laikų •

    Pagal vienetus • Arba visą periodą/trukmę (Period/Duration) LocalDate date1 = LocalDate.of(2012, 3, 4); LocalDate date2 = LocalDate.of(2013, 1, 26); // Duration long days = DAYS.between(date1, date2); // Period Period period = Period.between(date1, date2);
  50. Plėtimo galimybės • Plečiama 5 būdais, parašant savo ◦ datos/laiko

    klasę ◦ kiekio (amount) klasę ◦ datos/laiko laukelius ◦ datos/laiko vienetus ◦ kalendorių • Nepamiršti apie Javadoc kontraktus!
  51. Summary • JSR-310 naujas JDK8 Date/Time API • Išbandykite JDK

    8 buildą arba JDK 7 backportą • http://threeten.github.com