如何强制LocalDateTime Month设为3个字母长

问题描述

我的目标是使用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版本中它们也可能有所不同。我建议您在两个选项之间进行选择:

  1. 与您所得到的生活在一起。它已经很好地完成了工作,而且并非不合理。任何德国人在阅读和理解您提到的缩写词时都会遇到麻烦吗?
  2. 如果您对缩写有非常具体的要求,例如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版本中删除。

链接