Як я можу перетворити цілий номер у локалізоване ім'я місяця на Java?


99

Я отримую ціле число, і мені потрібно перетворити на місяць імена в різних мовах:

Приклад для локальної мови en-us:
1 ->
2 січня -> лютого

Приклад локальної програми es-mx:
1 -> Enero
2 -> Febrero


5
Слідкуйте, місяці на Java складаються з нуля, тому 0 = січень, 1 = лютий тощо
Нік Холт

ви маєте рацію, тож якщо вам потрібно змінити мову, просто потрібно змінити локаль. Спасибі
atomfat

2
@NickHolt ОНОВЛЕННЯ Сучасний java.timeMonthперелік є одноосновним : 1-12 за січень-грудень. У Ditto для [ java.time.DayOfWeek](https://docs.oracle.com/javase/9/docs/api/java/time/DayOfWeek.html): 1-7 for Monday-Sunday per ISO 8601 standard. Only the troublesome old legacy date-time classes such as Календар` є шалені схеми нумерації. Однією з багатьох причин уникати застарілих класів, які тепер повністю витіснені класами java.time .
Василь Бурк

Відповіді:


211
import java.text.DateFormatSymbols;
public String getMonth(int month) {
    return new DateFormatSymbols().getMonths()[month-1];
}

12
Вам не потрібен "місяць-1", оскільки масив заснований на нулі? atomfat хоче 1 -> січня і т. д.
Брайан Агнеу

7
Йому потрібен місяць-1, тому що місяць - це число, засноване на 1-му місяці, яке потрібно перетворити на нульове положення масиву
Сем Барнум,

5
public String getMonth (int month, locale locale) {return DateFormatSymbols.getInstance (locale) .getMonths () [month-1]; }
atomfat

4
ВІН потребує month-1. Усі, хто користується Calendar.get(Calendar.MONTH), просто знадоблятьсяmonth
Ron

1
Реалізація DateFormatSymbols була змінена в JDK 8, тому метод getMonths більше не повертає правильних значень для всіх Locale: oracle.com/technetwork/java/javase/…
ahaaman

33

Вам потрібно використовувати LLLL для самостійних імен місяця. це зафіксовано в SimpleDateFormatдокументації, наприклад:

SimpleDateFormat dateFormat = new SimpleDateFormat( "LLLL", Locale.getDefault() );
dateFormat.format( date );

JDK 1.7 /IllegalArgumentException : Illegal pattern character 'L'
AntJavaDev

26

тл; д-р

Month                             // Enum class, predefining and naming a dozen objects, one for each month of the year. 
.of( 12 )                         // Retrieving one of the enum objects by number, 1-12. 
.getDisplayName(
    TextStyle.FULL_STANDALONE , 
    Locale.CANADA_FRENCH          // Locale determines the human language and cultural norms used in localizing. 
)

java.time

Оскільки Java 1.8 (або 1.7 та 1.6 з ThreeTen-Backport ), ви можете використовувати це:

Month.of(integerMonth).getDisplayName(TextStyle.FULL_STANDALONE, locale);

Зауважимо, що integerMonthна основі 1, тобто 1 на січень. Діапазон завжди від 1 до 12 для січня-грудня (тобто лише григоріанський календар).


скажімо, у вас є строковий місяць травня на французькій мові, використовуючи метод, який ви опублікували (травень по-французьки - це Май), як я можу отримати число 5 з цієї рядки ??
usertest

@usertest Я написав клас грубої чернетки MonthDelocalizerу своїй відповіді, щоб отримати Monthоб’єкт із переданого локалізованого рядка імені місяця: mai→ Month.MAY. Зауважте, що питання чутливості до справ: Французька мова Maiє недійсною та має бути такою mai.
Василь Бурк

Це 2019. Як це не найкраща відповідь?
anothernode

16

Я б використав SimpleDateFormat. Хтось виправить мене, якщо є простіший спосіб скласти календар з печивом, я це роблю в коді зараз, і я не такий впевнений.

import java.text.DateFormat;
import java.text.SimpleDateFormat;
import java.util.Calendar;
import java.util.GregorianCalendar;


public String formatMonth(int month, Locale locale) {
    DateFormat formatter = new SimpleDateFormat("MMMM", locale);
    GregorianCalendar calendar = new GregorianCalendar();
    calendar.set(Calendar.DAY_OF_MONTH, 1);
    calendar.set(Calendar.MONTH, month-1);
    return formatter.format(calendar.getTime());
}

Ці страшні класи тепер успадковані, повністю витіснені сучасними класами java.time, визначеними в JSR 310.
Василь Бурк

14

Ось як я це зробив би. Я залишу перевірку дальності на int monthвас.

import java.text.DateFormatSymbols;

public String formatMonth(int month, Locale locale) {
    DateFormatSymbols symbols = new DateFormatSymbols(locale);
    String[] monthNames = symbols.getMonths();
    return monthNames[month - 1];
}

12

Використання SimpleDateFormat.

import java.text.SimpleDateFormat;

public String formatMonth(String month) {
    SimpleDateFormat monthParse = new SimpleDateFormat("MM");
    SimpleDateFormat monthDisplay = new SimpleDateFormat("MMMM");
    return monthDisplay.format(monthParse.parse(month));
}


formatMonth("2"); 

Результат: лютий


7

Мабуть, в Android 2.2 є помилка з SimpleDateFormat.

Для того, щоб використовувати імена місяця, ви повинні визначити їх самостійно у своїх ресурсах:

<string-array name="month_names">
    <item>January</item>
    <item>February</item>
    <item>March</item>
    <item>April</item>
    <item>May</item>
    <item>June</item>
    <item>July</item>
    <item>August</item>
    <item>September</item>
    <item>October</item>
    <item>November</item>
    <item>December</item>
</string-array>

А потім використовуйте їх у своєму коді так:

/**
 * Get the month name of a Date. e.g. January for the Date 2011-01-01
 * 
 * @param date
 * @return e.g. "January"
 */
public static String getMonthName(Context context, Date date) {

    /*
     * Android 2.2 has a bug in SimpleDateFormat. Can't use "MMMM" for
     * getting the Month name for the given Locale. Thus relying on own
     * values from string resources
     */

    String result = "";

    Calendar cal = Calendar.getInstance();
    cal.setTime(date);
    int month = cal.get(Calendar.MONTH);

    try {
        result = context.getResources().getStringArray(R.array.month_names)[month];
    } catch (ArrayIndexOutOfBoundsException e) {
        result = Integer.toString(month);
    }

    return result;
}

"Мабуть, в Android 2.2 є помилка" - Було б корисно, якби ви могли посилання на те, де відслідковується помилка.
Пітер Хол

6

тл; д-р

Month.of( yourMonthNumber )           // Represent a month by its number, 1-12 for January-December. 
  .getDisplayName(                    // Generate text of the name of the month automatically localized. 
      TextStyle.SHORT_STANDALONE ,    // Specify how long or abbreviated the name of month should be.
      new Locale( "es" , "MX" )       // Locale determines (a) the human language used in translation, and (b) the cultural norms used in deciding issues of abbreviation, capitalization, punctuation, and so on.
  )                                   // Returns a String.

java.time.Month

Набагато простіше це зробити зараз у класах java.time, які витісняють ці клопітні старі застарілі класи.

The MonthПерерахування визначає дюжину об'єктів, по одному на кожен місяць.

Місяці нумеруються 1-12 за січень-грудень.

Month month = Month.of( 2 );  // 2 → February.

Попросіть об'єкт створити рядок імені місяця, автоматично локалізованого .

Налаштуйте значення, TextStyleщоб вказати, наскільки довге або скорочене ви хочете назвати. Зауважте, що в деяких мовах (не англійській) назва місяця змінюється, якщо вона використовується окремо або як частина повної дати. Отже, кожен стиль тексту має a…_STANDALONE варіант.

Вкажіть a, Localeщоб визначити:

  • Яку людську мову слід використовувати в перекладі.
  • Які культурні норми повинні вирішувати такі питання, як абревіатура, пунктуація та велика літера.

Приклад:

Locale l = new Locale( "es" , "MX" );
String output = Month.FEBRUARY.getDisplayName( TextStyle.SHORT_STANDALONE , l );  // Or Locale.US, Locale.CANADA_FRENCH. 

Назва → Month об’єкт

FYI, в іншому напрямку (розбір рядка імені місяця для отримання Monthоб'єкта enum) не вбудований. Ви можете написати власний клас для цього. Ось моя швидка спроба такого заняття. Використовуйте на свій страх і ризик . Я не давав цього коду жодної серйозної думки, ані серйозних випробувань.

Використання.

Month m = MonthDelocalizer.of( Locale.CANADA_FRENCH ).parse( "janvier" ) ;  // Month.JANUARY

Код.

package com.basilbourque.example;

import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;

import java.time.Month;
import java.time.format.TextStyle;
import java.util.ArrayList;
import java.util.List;
import java.util.Locale;

// For a given name of month in some language, determine the matching `java.time.Month` enum object.
// This class is the opposite of `Month.getDisplayName` which generates a localized string for a given `Month` object.
// Usage… MonthDelocalizer.of( Locale.CANADA_FRENCH ).parse( "janvier" ) → Month.JANUARY
// Assumes `FormatStyle.FULL`, for names without abbreviation.
// About `java.time.Month` enum: https://docs.oracle.com/javase/9/docs/api/java/time/Month.html
// USE AT YOUR OWN RISK. Provided without guarantee or warranty. No serious testing or code review was performed.
public class MonthDelocalizer
{
    @NotNull
    private Locale locale;

    @NotNull
    private List < String > monthNames, monthNamesStandalone; // Some languages use an alternate spelling for a “standalone” month name used without the context of a date.

    // Constructor. Private, for static factory method.
    private MonthDelocalizer ( @NotNull Locale locale )
    {
        this.locale = locale;

        // Populate the pair of arrays, each having the translated month names.
        int countMonthsInYear = 12; // Twelve months in the year.
        this.monthNames = new ArrayList <>( countMonthsInYear );
        this.monthNamesStandalone = new ArrayList <>( countMonthsInYear );

        for ( int i = 1 ; i <= countMonthsInYear ; i++ )
        {
            this.monthNames.add( Month.of( i ).getDisplayName( TextStyle.FULL , this.locale ) );
            this.monthNamesStandalone.add( Month.of( i ).getDisplayName( TextStyle.FULL_STANDALONE , this.locale ) );
        }
//        System.out.println( this.monthNames );
//        System.out.println( this.monthNamesStandalone );
    }

    // Constructor. Private, for static factory method.
    // Personally, I think it unwise to default implicitly to a `Locale`. But I included this in case you disagree with me, and to follow the lead of the *java.time* classes. --Basil Bourque
    private MonthDelocalizer ( )
    {
        this( Locale.getDefault() );
    }

    // static factory method, instead of  constructors.
    // See article by Dr. Joshua Bloch. http://www.informit.com/articles/article.aspx?p=1216151
    // The `Locale` argument determines the human language and cultural norms used in de-localizing input strings.
    synchronized static public MonthDelocalizer of ( @NotNull Locale localeArg )
    {
        MonthDelocalizer x = new MonthDelocalizer( localeArg ); // This class could be optimized by caching this object.
        return x;
    }

    // Attempt to translate the name of a month to look-up a matching `Month` enum object.
    // Returns NULL if the passed String value is not found to be a valid name of month for the human language and cultural norms of the `Locale` specified when constructing this parent object, `MonthDelocalizer`.
    @Nullable
    public Month parse ( @NotNull String input )
    {
        int index = this.monthNames.indexOf( input );
        if ( - 1 == index )
        { // If no hit in the contextual names, try the standalone names.
            index = this.monthNamesStandalone.indexOf( input );
        }
        int ordinal = ( index + 1 );
        Month m = ( ordinal > 0 ) ? Month.of( ordinal ) : null;  // If we have a hit, determine the `Month` enum object. Else return null.
        if ( null == m )
        {
            throw new java.lang.IllegalArgumentException( "The passed month name: ‘" + input + "’ is not valid for locale: " + this.locale.toString() );
        }
        return m;
    }

    // `Object` class overrides.

    @Override
    public boolean equals ( Object o )
    {
        if ( this == o ) return true;
        if ( o == null || getClass() != o.getClass() ) return false;

        MonthDelocalizer that = ( MonthDelocalizer ) o;

        return locale.equals( that.locale );
    }

    @Override
    public int hashCode ( )
    {
        return locale.hashCode();
    }

    public static void main ( String[] args )
    {
        // Usage example:
        MonthDelocalizer monthDelocJapan = MonthDelocalizer.of( Locale.JAPAN );
        try
        {
            Month m = monthDelocJapan.parse( "pink elephant" ); // Invalid input.
        } catch ( IllegalArgumentException e )
        {
            // … handle error
            System.out.println( "ERROR: " + e.getLocalizedMessage() );
        }

        // Ignore exception. (not recommended)
        if ( MonthDelocalizer.of( Locale.CANADA_FRENCH ).parse( "janvier" ).equals( Month.JANUARY ) )
        {
            System.out.println( "GOOD - In locale "+Locale.CANADA_FRENCH+", the input ‘janvier’ parses to Month.JANUARY." );
        }
    }
}

Про java.time

Java.time каркас вбудований в Java 8 і пізніших версій. Ці класи витісняти неприємні старі застарілі класи дати і часу , такі як java.util.Date, Calendar, іSimpleDateFormat .

Проект Joda-Time , який зараз знаходиться в режимі обслуговування , радить перейти на java.time класів .

Щоб дізнатися більше, дивіться навчальний посібник Oracle . І шукайте переповнення стека за багатьма прикладами та поясненнями. Специфікація - JSR 310 .

Ви можете обмінюватися об'єктами java.time безпосередньо зі своєю базою даних. Використовуйте драйвер JDBC, сумісний з JDBC 4.2 або пізнішої версії. Немає необхідності в струнах, немає необхідності в java.sql.*заняттях.

Де отримати класи java.time?

  • Java SE 8 , Java SE 9 та новіших версій
    • Вбудований.
    • Частина стандартного Java API з пакетною реалізацією.
    • Java 9 додає деякі незначні функції та виправлення.
  • Java SE 6 та Java SE 7
    • Значна частина функцій java.time повертається до Java 6 і 7 у ThreeTen-Backport .
  • Android
    • Пізніші версії реалізації пакетів Android для класів java.time.
    • Для більш ранньої Android (<26), то ThreeTenABP проект адаптує ThreeTen-Backport (згаданий вище). Див. Як користуватися ThreeTenABP… .

Проект ThreeTen-Extra розширює java.time додатковими класами. Цей проект є передумовою для можливих майбутніх доповнень до java.time. Ви можете знайти деякі корисні класи тут , такі як Interval, YearWeek, YearQuarter, і більш .


1

Виникає проблема, коли ви використовуєте клас DateFormatSymbols для свого методу getMonthName, щоб отримати місяць за іменем, який показуватиме місяць за номером на деяких пристроях Android. Я вирішив цю проблему таким чином:

У String_array.xml

<string-array name="year_month_name">
    <item>January</item>
    <item>February</item>
    <item>March</item>
    <item>April</item>
    <item>May</item>
    <item>June</item>
    <item>July</item>
    <item>August</item>
    <item>September</item>
    <item>October</item>
    <item>November</item>
    <item>December</item>
    </string-array>

У класі Java просто викликайте цей масив так:

public String[] getYearMonthName() {
        return getResources().getStringArray(R.array.year_month_names);
        //or like 
       //return cntx.getResources().getStringArray(R.array.month_names);
    } 

      String[] months = getYearMonthName(); 
           if (i < months.length) {
            monthShow.setMonthName(months[i] + " " + year);

            }

Щасливе кодування :)


1

Розширення Kotlin

fun Int.toMonthName(): String {
    return DateFormatSymbols().months[this]
}

Використання

calendar.get(Calendar.MONTH).toMonthName()

Страшний Calendarклас витіснили роки тому класи java.time, визначені в JSR 310.
Василь Бурк

0
    public static void main(String[] args) {
    SimpleDateFormat format = new SimpleDateFormat("MMMMM", new Locale("en", "US"));
    System.out.println(format.format(new Date()));
}

це здається правильною відповіддю, але чи можете ви пояснити, що ви робите, і чому це робите саме так?
Мартін Франк

Я роблю так, бо думаю, що це просто і не складно!
Діого Олівейра

SimpleDateFormatОднак він використовує горезвісний клопіт і довго застарілий клас.
Оле В. В.

Ці страшні заняття з датою часу були витіснені роками тому класами java.time, визначеними в JSR 310.
Василь Бурк

0

Просто вставляючи рядок

DateFormatSymbols.getInstance().getMonths()[view.getMonth()] 

зробить трюк.


2
DateFormatSymbolsє частиною жахливих класів дати-часу, які зараз перейшли у спадщину, з моменту прийняття JSR 310 . Тепер витіснені класами java.time . Запропонувати їх використання у 2019 році - це погана порада.
Василь Бурк

Ця відповідь дублює зміст прийнятої відповіді .
Василь Бурк

0

Спробуйте скористатися цим дуже простим способом і назвіть це як власний функціонал

public static String convertnumtocharmonths(int m){
         String charname=null;
         if(m==1){
             charname="Jan";
         }
         if(m==2){
             charname="Fev";
         }
         if(m==3){
             charname="Mar";
         }
         if(m==4){
             charname="Avr";
         }
         if(m==5){
             charname="Mai";
         }
         if(m==6){
             charname="Jun";
         }
         if(m==7){
             charname="Jul";
         }
         if(m==8){
             charname="Aou";
         }
         if(m==9){
             charname="Sep";
         }
         if(m==10){
             charname="Oct";
         }
         if(m==11){
             charname="Nov";
         }
         if(m==12){
             charname="Dec";
         }
         return charname;
     }

1
Не потрібно писати такий тип коду. Java має вбудований Month::getDisplayName.
Василь Бурк

Не потрібно писати цей код котла. Перевірте мою відповідь, розміщену вище.
Садда
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.