问题描述
我的目标是使用LocalDateTime
并显示一个正好3个字母的月份。
对于英语,这很容易:
val englishFormatter = DateTimeFormatter.ofPattern("MMM",Locale.ENGLISH)
for (month in 1..12) {
println(LocalDateTime.of(0,month,1,0)
.format(englishFormatter))
}
结果如预期:
Jan Feb Mar Apr May Jun Jul Aug Sep Oct Nov Dec
对于德语(如上所述,仅使用Locale.GERMAN
),结果是意外的:
Jan. Feb. März Apr. Mai Juni Juli Aug. Sep. Okt. Nov. Dez.
虽然缩写词在德语中都很常见,但是“März”,“ Juni”和“ Juli”并没有被缩短(“ Mai”不需要被缩短)。另外,大多数月份包含的字母超过3个(注意点!)
有两种方法可以缩短它们吗?
例如
梅尔兹⇒梅尔兹。
Juni⇒Jun。
Juli⇒7月。
顺便说一句:代码在Kotlin中,但是Kotlin在使用Java的LocalDateTime
。因此标记为Java
编辑:我正在Android 7.0上运行此代码
解决方法
您可以使用所需的/期望的缩写,如下所示:
import java.time.LocalDateTime
import java.time.Month
import java.time.format.TextStyle
import java.util.Locale
fun main() {
for (m in 1..12) {
val month = Month.of(m)
println(month.getDisplayName(TextStyle.SHORT,Locale.GERMAN))
}
}
输出为
Jan
Feb
Mär
Apr
Mai
Jun
Jul
Aug
Sep
Okt
Nov
Dez
LocalDateTime
的方法getMonth()
返回一个Month
对象,这意味着您可以获得LocalDateTime
的月份并构建所需的String
,也许像下面这样一个小的fun
fun getAbbreviatedMonth(localDateTime: LocalDateTime,locale: Locale): String {
return localDateTime.getMonth()
.getDisplayName(TextStyle.SHORT,locale)
}
,甚至没有Locale
作为参数和硬编码 Locale.GERMAN
。
真的不知道为什么德语缩写会被简化为4个字符,而月份名称却原样保留4个或更少的字符,而当您使用DateTimeFormatter
时,会将其余的字母缩写为三个字母,再加上一个点在您的系统上,使用Kotlin Playground中的模式"MMM"
不会!以下代码会产生两行相等的输出:
import java.time.LocalDateTime
import java.time.Month
import java.time.format.TextStyle
import java.util.Locale
import java.time.format.DateTimeFormatter
fun main() {
var localDateTime = LocalDateTime.now()
println(getAbbreviatedMonth(localDateTime,Locale.GERMAN))
println(localDateTime.format(DateTimeFormatter.ofPattern("MMM",Locale.GERMAN)))
}
fun getAbbreviatedMonth(localDateTime: LocalDateTime,locale)
}
(执行时间为2020-08-12)
Aug
Aug
,
要精确控制Java给您的月份缩写是很困难的,我认为您不会想要打扰。 Java从多达四个来源获取其语言环境数据,这些来源通常具有版本。因此,即使您设法使结果完全正确,在下一个Java版本中它们也可能有所不同。我建议您在两个选项之间进行选择:
- 与您所得到的生活在一起。它已经很好地完成了工作,而且并非不合理。任何德国人在阅读和理解您提到的缩写词时都会遇到麻烦吗?
- 如果您对缩写有非常具体的要求,例如
Mrz
而不是Mär
,请对所需的缩写进行硬编码,然后您将知道它们将保持这种方式,无论语言环境提供者和/或区域设置数据版本更改。
作为两者之间的折衷,您可以尝试通过定义系统属性java.locale.providers
选择区域设置数据提供程序。如我所说,您从提供商那里获得的语言环境数据可能会在将来的版本中更改。
如果您想对自己的首字母缩写进行硬编码,仍可以构建一个使用您的缩写的DateTimeFormatter
。对于Java的简单演示:
Map<Long,String> monthAbbreviations = Map.ofEntries(
Map.entry(1L,"Jan"),Map.entry(2L,"Feb"),Map.entry(3L,"Mrz"),Map.entry(4L,"Apr"),Map.entry(5L,"Mai"),Map.entry(6L,"Jun"),Map.entry(7L,"Jul"),Map.entry(8L,"Aug"),Map.entry(9L,"Sep"),Map.entry(10L,"Okt"),Map.entry(11L,"Nov"),Map.entry(12L,"Dez"));
DateTimeFormatter formatter = new DateTimeFormatterBuilder()
.appendText(ChronoField.MONTH_OF_YEAR,monthAbbreviations)
.toFormatter(Locale.GERMAN);
String allAbbreviations = Arrays.stream(Month.values())
.map(formatter::format)
.collect(Collectors.joining(" "));
System.out.println(allAbbreviations);
输出为:
2月1日Mrz 4月Mai Mai 6月7月8月9月Okt 11月Dez
语言环境数据提供者来自LocaleServiceProvider
的文档:
Java运行时环境提供了以下四个语言环境提供程序:
- “ CLDR”:基于Unicode联盟的CLDR项目的提供程序。
- “ COMPAT”:表示与以前的JDK发行版本(直到JDK8)兼容的语言环境敏感服务(与JDK8的相同) “ JRE”)。
- “ SPI”:表示实现此LocaleServiceProvider类的子类的对语言环境敏感的服务。
- “主机”:一种提供程序,可在基础操作系统中反映用户的自定义设置。该提供程序可能不可用, 取决于Java Runtime Environment的实现。
- “ JRE”:表示“ COMPAT”的同义词。此名称已弃用,并将在将来的JDK版本中删除。