如何在共享首选项中保存和检索日期tl;博士java.time关于 java.time

我需要在Android中保存一些日期并检索它。我正在构建提醒应用程序,我需要保存未来日期的列表。它必须能够以毫秒为单位进行检索。首先,我想计算今天现在和未来时间之间的时间,并以共享首选项存储。但是该方法不起作用,因为我需要将其用于.SharedPreferencesAlarmManagerAlarmManager


答案 1

要保存和加载准确的日期,可以使用对象的(数字)表示形式。longDate

例:

//getting the current time in milliseconds, and creating a Date object from it:
Date date = new Date(System.currentTimeMillis()); //or simply new Date();

//converting it back to a milliseconds representation:
long millis = date.getTime();

您可以使用它来保存或检索/数据,如下所示DateTimeSharedPreferences

救:

SharedPreferences prefs = ...;
prefs.edit().putLong("time", date.getTime()).apply();

读回去:

Date myDate = new Date(prefs.getLong("time", 0));

编辑

如果你想存储附加的,你可以为此目的编写一些帮助器方法,如下所示(我还没有测试它们,如果有什么问题,请随时纠正它):TimeZone

public static Date getDate(final SharedPreferences prefs, final String key, final Date defValue) {
    if (!prefs.contains(key + "_value") || !prefs.contains(key + "_zone")) {
        return defValue;
    }
    Calendar calendar = Calendar.getInstance();
    calendar.setTimeInMillis(prefs.getLong(key + "_value", 0));
    calendar.setTimeZone(TimeZone.getTimeZone(prefs.getString(key + "_zone", TimeZone.getDefault().getID())));
    return calendar.getTime();
}

public static void putDate(final SharedPreferences prefs, final String key, final Date date, final TimeZone zone) {
    prefs.edit().putLong(key + "_value", date.getTime()).apply();
    prefs.edit().putString(key + "_zone", zone.getID()).apply();
}

答案 2

tl;博士

现代方法使用 java.time 类和 ISO 8601 字符串。

读数。

Instant                             // Represent a moment in UTC with a resolution of nanoseconds.
.ofEpochMilli( 
    Long.getLong( incomingText )  
)                                   // Returns a `Instant` object.
.atZone(                            // Adjust from UTC to some time zone. Same moment, same point on the timeline, different wall-clock time.
    ZoneId.of( "Europe/Paris" ) 
)                                   // Returns a `ZonedDateTime` object.

写作。

ZonedDateTime
.of(
    LocalDate.of( 2018 , Month.JANUARY , 23 ) ,
    LocalTime.of( 15 , 35 ) ,
    ZoneId.of( "Europe/Paris" ) 
)                                   // Returns a `ZonedDateTime` object.
.toInstant()                        // Returns an `Instant`. Adjust from a time zone to UTC. Same moment, same point on the timeline, different wall-clock time.
.toEpochMilli()                     // Returns a `long` integer number primitive. Any microseconds or nanoseconds are ignored, of course.

如果您的警报管理器尚未现代化以处理 java.time 对象,请使用添加到旧类的新方法在旧类和现代类之间进行转换。

java.util.Date d = java.util.Date.from( instant ) ;

...和。。。

Instant instant = d.toInstant() ;

java.time

麻烦的旧日期时间类被java.time类所取代。

在 UTC 中,分辨率为纳秒的片刻,请使用 。Instant

Instant instant = Instant.now() ;  // Capture the current moment in UTC.

您只需要毫秒来满足您的需求,因此请截断任何微秒和纳秒。

Instant instant = Instant.now().truncatedTo( ChronoUnit.MILLIS ) ;

要按日期和时间确定时刻,需要一个时区。时区对于确定日期至关重要。在任何给定的时刻,日期在全球范围内因区域而异。例如,法国巴黎午夜后几分钟是新的一天,而在魁北克蒙特利尔仍然是“昨天”。

如果未指定时区,则 JVM 将隐式应用其当前缺省时区。该默认值可能会在运行时(!)的任何时刻更改,因此您的结果可能会有所不同。最好将所需/预期的时区显式指定为参数。

以 的格式指定适当的时区名称,例如 美国/蒙特利尔非洲/卡萨布兰卡或 。切勿使用3-4个字母的缩写,例如或因为它们不是真正的时区,不是标准化的,甚至不是唯一的(!)。continent/regionPacific/AucklandESTIST

ZoneId z = ZoneId.of( "America/Montreal" ) ;  
LocalDate today = LocalDate.now( z ) ;

如果要使用 JVM 的当前缺省时区,请请求它并作为参数传递。如果省略,则隐式应用 JVM 的当前缺省值。最好是显式的,因为在运行时的任何时刻,JVM 中任何应用程序的任何线程中的任何代码都可能更改默认值。

ZoneId z = ZoneId.systemDefault() ;  // Get JVM’s current default time zone.

或者指定一个日期。您可以按数字设置月份,1 月至 12 月的合理编号为 1-12。

LocalDate ld = LocalDate.of( 1986 , 2 , 23 ) ;  // Years use sane direct numbering (1986 means year 1986). Months use sane numbering, 1-12 for January-December.

或者,更好的是,使用预定义的 Month 枚举对象,一年中的每个月都有一个对象。提示: 在整个代码库中使用这些对象,而不仅仅是整数,以使代码更具自我记录性,确保有效值并提供类型安全Month

LocalDate ld = LocalDate.of( 1986 , Month.FEBRUARY , 23 ) ;

与一天中的时间组合,.LocalTime

LocalTime lt = LocalTime.of( 14 , 0 ) ;

将它们作为一个对象包装在一起。ZonedDateTime

ZonedDateTime zdt = ZonedDateTime.of( ld , lt , z ) ;

通过提取 .Instant

Instant instant = zdt.toInstant() ;

提取自 UTC 中 1970 年第一个时刻的纪元参考以来所需的毫秒数。同样,请注意,在提取毫秒时,您的任何微米/纳米都将被忽略。Instant

long milliseconds = instant.toEpochMilli() ;  // Be aware of potential data loss, ignoring any microseconds or nanoseconds. 

使用 Long 类将这些毫秒作为文本从存储中读回。

long milliseconds = Long.getLong( incomingText ) ;
Instant instant = Instant.ofEpochMilli( milliseconds ) ;

要通过特定地区(时区)的人们使用的挂钟时间的镜头来查看该时刻,请应用a以获得.ZoneIdZonedDateTime

ZoneId z = ZoneId.of( "Africa/Tunis" ) ;
ZonedDateTime zdt = instant.atZone( z ) ;

若要生成表示该值的文本,请使用 自动本地化。DateTimeFormatter.ofLocalizedDateTime

提示: 请考虑以标准 ISO 8601 格式将日期时间值写入存储,而不是以毫秒数的形式写入存储。毫秒无法被人类有意义地读取,这使得调试和监控变得棘手。

String output = instant.toString() ; 

2018-10-03T10:48:48.584Z

Instant instant = Instant.parse( 2018-10-05T20:28:48.584Z ) ;

关于 java.time

java.time 框架内置于 Java 8 及更高版本中。这些类取代了麻烦的旧日期时间类,如java.util.DateCalendarSimpleDateFormat

Joda-Time 项目现在处于维护模式,建议迁移到 java.time 类。

要了解更多信息,请参阅 Oracle 教程。搜索 Stack Overflow 以获取许多示例和解释。规格是JSR 310

您可以直接与数据库交换 java.time 对象。使用符合 JDBC 4.2 或更高版本的 JDBC 驱动程序。不需要字符串,不需要类。java.sql.*

从哪里获取 java.time 类?

ThreeTen-Extra 项目通过其他类扩展了 java.time。这个项目是未来可能添加到java.time的试验场。您可以在此处找到一些有用的课程,例如IntervalYearWeekYearQuarter