我正在尝试以"mm/dd/yyyy hh:mm a"格式创建一个时间戳。我的应用程序最初崩溃了,但我尝试了另一种方法,它不再崩溃,但在聊天屏幕上它显示"java.text.SimpleDateFormat@4f47c0fb"而不是时间戳。
这是聊天屏幕时间戳的屏幕截图
这是来自我的适配器聊天.java文件
@Override
public void onBindViewHolder(@NonNull MyHolder myHolder, int i) {
//Get data
String message = chatList.get(i).getMessage();
String timeStamp = chatList.get(i).getTimestamp();
//Convert time stamp to mm/dd/yyyy hh:mm am/pm
//Calendar cal = Calendar.getInstance(Locale.ENGLISH);
//cal.setTimeInMillis(Long.parseLong(timeStamp));
String dateTime = new SimpleDateFormat("MM/dd/yyyy hh:mm a", Locale.getDefault()).toString();
//Set data
myHolder.messageTv.setText(message);
myHolder.timeTv.setText(dateTime);
try {
Picasso.get().load(imageUrl).into(myHolder.profileIv);
} catch (Exception e) {
}
//Set seen/delivered status of message
if (i == chatList.size() - 1) {
if (chatList.get(i).isSeen()) {
myHolder.isSeenTv.setText("Seen");
}
else {
myHolder.isSeenTv.setText("Delivered");
}
}
else {
myHolder.isSeenTv.setVisibility(View.GONE);
}
}
我注释掉了导致它崩溃的 2 行代码,并尝试了它下面的新方法。有人可以引导我朝着正确的方向前进吗?我不确定我做错了什么。任何帮助将不胜感激。谢谢!
java.time 和 ThreeTenABP
关于你的问题,有不止一个不清楚的地方,但我正在尝试一个建议。对于我的方式,我需要两个格式化程序:
private static DateTimeFormatter timestampFormatter = new DateTimeFormatterBuilder()
.appendValue(ChronoField.INSTANT_SECONDS)
.appendValue(ChronoField.MILLI_OF_SECOND, 3)
.toFormatter();
private static DateTimeFormatter dtf = DateTimeFormatter.ofPattern("MM/dd/yyyy hh:mm a");
我知道chatList.get(i).getTimestamp()
可能会返回null
,但在这种情况下,您没有任何聊天消息要显示。我们需要考虑到这种可能性。下面的代码将为您提供字符串NA
。如果您想要其他内容,请修改。接下来,从您注释掉的代码(在null
上崩溃的代码(来看,它也可能返回一个包含自纪元以来的长毫秒计数的String
。所以我也在处理这种情况。
int i = 0;
String timeStamp = chatList.get(i).getTimestamp();
String dateTime;
if (timeStamp == null) {
dateTime = "NA";
} else {
Instant inst = timestampFormatter.parse(timeStamp, Instant.FROM);
dateTime = inst.atZone(ZoneId.systemDefault()).format(dtf);
}
System.out.println(dateTime);
返回1567890123456
时我的时区(欧洲/哥本哈根(chatList.get(i).getTimestamp()
输出示例:
2019/09/07 11:02 下午
您的代码中出了什么问题?
第一种情况,当timeStamp
null
并且您尝试Long.parseLong(timeStamp)
时,您将获得java.lang.NumberFormatException: null
。如果代码未捕获此异常,则会崩溃。
第二种情况,您调用了SimpleDateFormat
的toString
方法。SimpleDateFormat
本身没有定义toString
方法,因此调用Object.toStrng()
。这将返回一个字符串,如java.text.SimpleDateFormat@4f47c0fb
.类名和一个有趣的十六进制值,由@
符号分隔。
但是,您可能根本不想使用SimpleDateFormat
。这个阶级是一个臭名昭著的麻烦制造者,早已过时。相反,在我的代码中,我使用的是来自java.time(现代Java日期和时间API(的DateTimeFormatter
和其他类。
问:java.time 不需要 Android API 级别 26 吗?
java.time在较旧和较新的Android设备上都能很好地工作。它只需要至少Java6。
- 在 Java 8 及更高版本以及较新的 Android 设备(从 API 级别 26(中,内置了现代 API。在这种情况下,您需要将上述代码中的常量
Instant.FROM
替换为方法引用Instant::from
。 - 在非Android Java 6和7中,获得ThreeTen Backport,现代类的反向移植(JSR 310的ThreeTen;请参阅底部的链接(。
- 在(较旧的(Android上使用Android版本的ThreeTen Backport。它被称为ThreeTenABP。并确保从带有子包的
org.threeten.bp
导入日期和时间类。
链接
- Oracle 教程:日期时间解释如何使用 java.time。
- Java 规范请求 (JSR( 310,其中首次描述了
java.time
。 - ThreeTen Backport 项目,将
java.time
向 Java 6 和 7 的反向移植(ThreeTen 用于 JSR-310(。 - ThreeTenABP,ThreeTen Backport 的安卓版
- 问:如何在Android项目中使用ThreeTenABP,并进行了非常详尽的解释。
你必须使用toLocalizedPattern((代替 toString((
String dateTime = new SimpleDateFormat("mm/dd/yyyy hh:mm a", Locale.getDefault()).toString();
即你必须像这样使用它
String dateTime = new SimpleDateFormat("mm/dd/yyyy hh:mm a", Locale.getDefault()).toLocalizedPattern();