android.text.format.DateFormat “HH” 不像 java.text.SimpleDateFormat 那样被识别

当我在 中使用 “HH” 标志时,它被解释为字面的“HH”。但是当我使用它时,它被解释为2位数字的小时。为什么它们不同?android.text.format.DateFormatjava.text.SimpleDateFormat

我不是在寻找一个可行的替代方案(我已经知道我必须使用而不是)。我只是好奇为什么“HH”不被识别。kkHH

Java 示例:

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    Calendar calendar = Calendar.getInstance();

    String dateJava = new java.text.SimpleDateFormat(
        "dd-MM-yyyy HH:mm:ss").format(calendar.getTime());

    String dateAndroid = android.text.format.DateFormat.format(
        "dd-MM-yyyy HH:mm:ss", calendar).toString();

    TextView tvAndroid = (TextView) findViewById(R.id.tvAndroid);
    TextView tvJava = (TextView) findViewById(R.id.tvJava);

    tvAndroid.setText("Android: " + dateAndroid);  //prints 26-05-2013 HH:36:34
    tvJava.setText("Java: " + dateJava);           //prints 26-05-2013 22:36:34
}

输出为:

Android: 26-05-2013 HH:36:34 
Java:    26-05-2013 22:36:34

我希望两者都是26-05-2013 22:36:34

Android的DateFormat有错误吗?

Java的SimpleDateFormat接受这些:

H   Hour in day (0-23)      Number  0
k   Hour in day (1-24)      Number  24
K   Hour in am/pm (0-11)    Number  0
h   Hour in am/pm (1-12)    Number  12

因此,Android开发人员似乎决定更改其含义,并且在他们的DateFormat函数中,它等同于SimpleDateFormat,正如他们在文档页面中明确指出的那样:kH

This constant was deprecated in API level 18. Use a literal 'H' (for 
compatibility with SimpleDateFormat and Unicode) or 'k' (for compatibility 
with Android releases up to and including Jelly Bean MR-1) instead. Note 
that the two are incompatible. 

这是什么原因?


答案 1

我知道你已经接受了一个答案,但只是为了向你充分解释这一点......

DateFormat 的源代码.java...

此类中的方法实现 Unicode UTS #35 模式的子集。此类当前支持的子集包括以下格式字符:。在 API 级别 17 之前,仅支持 adEhkMmszy。请注意,此类错误地实现了 k,就好像它是 H 以实现向后兼容性一样。formatacdEHhLKkLMmsyz

请注意我以粗体标记的部分。

我链接到的源代码已经更新,允许使用H,但它还没有正式发布(API 17是Android的当前版本,不支持H)。

在源代码的后面,在声明格式字符常量的阶段,有这个注释...

/**
 * @deprecated Use a literal {@code 'H'} (for compatibility with {@link SimpleDateFormat}
 * and Unicode) or {@code 'k'} (for compatibility with Android releases up to and including
 * Jelly Bean MR-1) instead. Note that the two are incompatible.
 */
@Deprecated
public  static final char    HOUR_OF_DAY            =    'k';

...以及稍后在字符替换期间...

case 'H': // hour in day (0-23)
case 'k': // hour in day (1-24) [but see note below]
{
    int hour = inDate.get(Calendar.HOUR_OF_DAY);
    // Historically on Android 'k' was interpreted as 'H', which wasn't
    // implemented, so pretty much all callers that want to format 24-hour
    // times are abusing 'k'. http://b/8359981.
    if (false && c == 'k' && hour == 0) {
        hour = 24;
    }
    replacement = zeroPad(hour, count);
}
break;

答案 2

因为。。。这不是一回事,它的行为与文档所述?

来自 android.text.format.DateFormat 的文档

此类仅支持完整 Unicode 规范的子集。如果您需要更多,请使用SimpleDateFormat。

但是,如果您进一步阅读文档

公共静态最终字符HOUR_OF_DAY

此指示符以 24 小时格式指示一天中的小时。下午 3 点的示例:k -> 15 午夜的示例:k -> 0 kk -> 00

所以。。。使用该类,它将是而不是kkHH