我一直在寻找这个答案,但我可能会误认为如何实现这一点。我有一个 ZonedDateTime 变量,目前如果我要打印它,它将打印例如 2017-12-03T10:15:30+01:00。有没有办法打印已经添加或减去偏移量的时间?例如,我想看看上面 16:30 的例子。谢谢您的帮助!
3 回答
看看这个:
public Date parseDateTime(String input) throws java.text.ParseException {
//NOTE: SimpleDateFormat uses GMT[-+]hh:mm for the TZ which breaks
//things a bit. Before we go on we have to repair this.
SimpleDateFormat df = new SimpleDateFormat( "yyyy-MM-dd'T'HH:mm:ssz" );
//this is zero time so we need to add that TZ indicator for
if ( input.endsWith( "Z" ) ) {
input = input.substring( 0, input.length() - 1) + "GMT-00:00";
} else {
int inset = 6;
String s0 = input.substring( 0, input.length() - inset );
String s1 = input.substring( input.length() - inset, input.length() );
input = s0 + "GMT" + s1;
}
return df.parse( input );
}
几乎没有什么改变,所以它适合您的需要。你应该可以毫不费力地做到这一点。然后在您拥有 Date 对象后,添加所需的偏移量:
int myNumHours = 4; //Any number of hours you want
myDateObject.add(Calendar.HOUR,myNumHours);
//Adds myNumHours to the Hour slot and processes all carrys if needed be.
现在打印它:
SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
String myDateString = df.format(myDateObject);
//Now just use your myDateString wherever you want.
tl;博士
yourZonedDateTime.atZone(ZoneOffset.UTC).withZoneSameInstant(ZoneId.systemDefault())
或者在我的情况下:
serverSeconds.toLocalDateTime(ZonedDateTime.now().offset) // extension below
fun Long.toLocalDateTime(zoneOffset: ZoneOffset = ZoneOffset.UTC): LocalDateTime {
return LocalDateTime.ofEpochSecond(this / 1000L, 0, zoneOffset)
}
历史
当我尝试正确格式化从服务器接收到的日期时间时,我偶然发现了这个问题。我做了这些简单的步骤:
- 服务器给了我这个:
1597752329121
,这是一个以秒为单位的日期时间。 - 将其转换为 LocalDateTime 给了我这个:
2020-08-18T12:05:29
. - 将其转换为 ZonedDateTime 给了我这个:
2020-08-18T12:05:29+03:00[Europe/Kiev]
这很完美,因为我需要像这样显示它15:05, 18 August 2020
,简单吧?但是以任何方式格式化它都是在嘲笑我12:05, 18 August 2020
。这么笨。我开始寻找我做错了什么,只找到“聪明”的评论,比如“时间已经在给定的时区”。你应该只得到它。
但是,如果 ZonedDateTime 以任何方式给您 12:05 而不是 15:05,那怎么可能?!
无论如何,我终于找到了这个答案(https://stackoverflow.com/a/35689566/3569545),实际上,它并没有解决我的问题,而是对如何解决它提出了一些想法。所以最后我尝试了这个:
val localDate = serverSeconds.toLocalDateTime() //an extension, see below
val zonedUTC = localDate.atZone(ZoneOffset.UTC)
val zoned = zonedUTC.withZoneSameInstant(ZoneId.systemDefault())
fun Long.toLocalDateTime(): LocalDateTime {
return LocalDateTime.ofEpochSecond(this / 1000L, 0, ZoneOffset.UTC)
}
最后格式化zoned
给了我要求15:05, 18 August 2020
的。我将其改进为您可以在tl;dr中看到的代码。
ZonedDateTime#withZoneSameInstant
import java.time.OffsetDateTime;
import java.time.ZoneId;
import java.time.ZoneOffset;
import java.time.ZonedDateTime;
import java.time.format.DateTimeFormatter;
public class Main {
public static void main(String[] args) {
String dateTimeStr = "2017-12-03T10:15:30+01:00";
ZonedDateTime zdtGiven = ZonedDateTime.parse(dateTimeStr);
System.out.println(zdtGiven);
// Date-time adjusted with Zone-Offset (i.e. date-time at UTC)
ZonedDateTime zdtAtUTC = zdtGiven.withZoneSameInstant(ZoneId.of("Etc/UTC"));
System.out.println(zdtAtUTC);
// Alternatively,
OffsetDateTime odtAtUTC = zdtGiven.toOffsetDateTime().withOffsetSameInstant(ZoneOffset.UTC);
System.out.println(odtAtUTC);
// Custom format
String formattedDateTimeStr = zdtAtUTC.format(DateTimeFormatter.ofPattern("uuuu-MM-dd'T'HH:mm:ss"));
System.out.println(formattedDateTimeStr);
}
}
输出:
2017-12-03T10:15:30+01:00
2017-12-03T09:15:30Z[Etc/UTC]
2017-12-03T09:15:30Z
2017-12-03T09:15:30
你理解的差距:
+01:00
日期时间字符串中的 ,表示字符串2017-12-03T10:15:30+01:00
中的日期和时间已经通过+01:00
在日期和时间中添加小时来调整UTC
。这意味着如果您想从中删除+01:00
,则+01:00
需要从中减去小时,2017-12-03T10:15:30
即它将2017-12-03T09:15:30
代表日期和时间UTC
。