Android: как поместить Enum в комплект?


332

Как добавить объект Enum в комплект Android?


11
На мой взгляд, такой совет от сотрудников Google плох. Перечисления очень удобны и страдают описанными накладными расходами.
Огнян

3
Можете ли вы вернуться к ответам и принять второй, если считаете, что это может быть лучшим выбором?
Филипп

6
Под заголовком «Избегание перечислений» в приведенной выше ссылке теперь говорится следующее: Мифы о производительности Предыдущие версии этого документа выдвигали различные вводящие в заблуждение утверждения. Мы рассмотрим некоторые из них здесь.
Переполнение

этот раздел даже больше не присутствует.
Натаниэль Д. Ваггонер

Ответы:


726

Перечисления являются сериализуемыми, поэтому проблем не возникает.

Учитывая следующее перечисление:

enum YourEnum {
  TYPE1,
  TYPE2
}

Bundle:

// put
bundle.putSerializable("key", YourEnum.TYPE1);

// get 
YourEnum yourenum = (YourEnum) bundle.get("key");

Намерение:

// put
intent.putExtra("key", yourEnum);

// get
yourEnum = (YourEnum) intent.getSerializableExtra("key");

Что-то не так с этим методом: save: outState.putSerializable("trollData", game.getFunkyTrolls());loading game.setFunkyTrolls((Game.FunkyTroll[]) savedInstanceState.getSerializable("trollData"));:?
Моберг

21
Я бы проголосовал за ваш ответ, но вопрос был о добавлении Enum в Bundle, и ваш ответ объясняет, как добавить его в Intent ... Конечно, это почти то же самое, но Алехандро ниже исправил ваш ответ.
Pooks

2
при использовании его с Bundle, он выбрасываетClassNotFoundException
Display Name

2
это может быть очень медленным и не масштабироваться до массивов вещей, которые содержат enum и т. д. См. stackoverflow.com/a/5551155/175156
yincrash

1
@yincrash enum использует пользовательскую сериализацию, которая довольно быстрая. Доказательство: docs.oracle.com/javase/1.5.0/docs/guide/serialization/spec/…
Miha_x64

164

Я знаю, что это старый вопрос, но у меня возникла та же проблема, и я хотел бы поделиться тем, как я ее решил. Ключ к тому, что сказал Мигель: Enums являются Serializable.

Учитывая следующее перечисление:

enum YourEnumType {
    ENUM_KEY_1, 
    ENUM_KEY_2
}

Ставить:

Bundle args = new Bundle();
args.putSerializable("arg", YourEnumType.ENUM_KEY_1);

3
Исходя из этого: stackoverflow.com/questions/15521309/… , пользовательские перечисления не сериализуются. Поэтому пользовательские поля в Enum не будут сериализованы. Как вы справляетесь с этим?
CLU

Хороший вопрос @clu! Может быть, тогда вам следует подумать о передаче его в виде строки, как указано в stackoverflow.com/questions/609860/…
Алехандро Колорадо

@clu Не ожидая сериализации пользовательских полей. Это работает нормально, если это просто обычное перечисление, как в коде выше.
bluehallu

@AlejandroColorado, что это добавляет к ответу Мигеля?
tir38

1
Ответ Мигеля был отредактирован в 2015 году. В первоначальном ответе ничего не говорилось о связках, он лишь показал пример намерения.
Алехандро Колорадо

41

Для полноты картины это полный пример того, как вставить и получить перечисление из пакета.

Учитывая следующее перечисление:

enum EnumType{
    ENUM_VALUE_1,
    ENUM_VALUE_2
}

Вы можете поместить enum в пакет:

bundle.putSerializable("enum_key", EnumType.ENUM_VALUE_1);

И вернуть перечисление:

EnumType enumType = (EnumType)bundle.getSerializable("enum_key");

32

Я использую котлин.

companion object {

        enum class Mode {
            MODE_REFERENCE,
            MODE_DOWNLOAD
        }
}

затем введите в намерение:

intent.putExtra(KEY_MODE, Mode.MODE_DOWNLOAD.name)

когда вы чистые, чтобы получить значение:

mode = Mode.valueOf(intent.getStringExtra(KEY_MODE))

6
Это хороший ответ, но он может быть дополнен методом расширения, я использую его здесь: gist.github.com/Grohden/eea5ff9d5e3ba955aa2f57ff0df2683f
Габриэль Де Оливейра Роден

.nameочень важный путь
Phan Van Linh

Это кажется намного проще, чем превращение Enum в пакет, который может создать дополнительную сложность при работе с библиотекой базы данных Android Room .
Адам Гурвиц

@GabrielDeOliveiraRohden, я не уверен, что нужен метод расширения, поскольку кажется, что он только избегает использования .namein putString(). С Kotlin это уже упрощено при использовании .apply. Например :ContentFragment.newInstance(Bundle().apply { putString(FEED_TYPE_KEY, SAVED.name) })
Адам Гурвиц

@ AdamHurwitz, не является ли предложенная функция расширения целым пунктом функций расширения Котлина? Это заставляет вас не делать ошибок, это прекрасно! Ссылка @GabrielDeOliveiraRohden bundle.putEnum(key, enum) | bundle.getEnum<>(key)
Йокич

17

Может быть, лучше передать его как строку из myEnumValue.name () и восстановить его из YourEnums.valueOf (s), так как в противном случае порядок перечисления должен быть сохранен!

Более длинное объяснение: Преобразовать из порядкового номера перечисления в тип перечисления


1
Порядок не имеет значения, если сериализация-> десериализация происходит сразу во время выполнения, например, при вызове одного действия другому. Это может быть проблемой для разных процессов, таких как отправка Intents из одного приложения в старые версии приложения.
Мигель

6

Другой вариант:

public enum DataType implements Parcleable {
    SIMPLE, COMPLEX;

    public static final Parcelable.Creator<DataType> CREATOR = new Creator<DataType>() {

        @Override
        public DataType[] newArray(int size) {
            return new DataType[size];
        }

        @Override
        public DataType createFromParcel(Parcel source) {
            return DataType.values()[source.readInt()];
        }
    };

    @Override
    public int describeContents() {
        return 0;
    }

    @Override
    public void writeToParcel(Parcel dest, int flags) {
        dest.writeInt(this.ordinal());
    }
}

1
Вы можете использовать putSerializable(key, value)/ (Type) getSerializable(key)или putString(key, value.name())/ Type.valueOf(getString(key)), реализация Parcelable здесь избыточна и бессмысленна.
Miha_x64

1
Использование Parcelable- хорошее решение для хранения массивов значений Enum.
RhodanV5500


2

Для намерения вы можете использовать этот способ:

Намерение: kotlin

FirstActivity:

val intent = Intent(context, SecondActivity::class.java)
intent.putExtra("type", typeEnum.A)
startActivity(intent)

SecondActivity:

override fun onCreate(savedInstanceState: Bundle?) {
     super.onCreate(savedInstanceState) 
     //...
     val type = (intent.extras?.get("type") as? typeEnum.Type?)
}

1

Следует помнить одну вещь - если вы используете bundle.putSerializableдля Bundleдобавления в уведомление, вы можете столкнуться со следующей проблемой:

*** Uncaught remote exception!  (Exceptions are not yet supported across processes.)
    java.lang.RuntimeException: Parcelable encountered ClassNotFoundException reading a Serializable object.

...

Чтобы обойти это, вы можете сделать следующее:

public enum MyEnum {
    TYPE_0(0),
    TYPE_1(1),
    TYPE_2(2);

    private final int code;

    private MyEnum(int code) {
        this.code = navigationOptionLabelResId;
    }

    public int getCode() {
        return code;
    }

    public static MyEnum fromCode(int code) {
        switch(code) {
            case 0:
                return TYPE_0;
            case 1:
                return TYPE_1;
            case 2:
                return TYPE_2;
            default:
                throw new RuntimeException(
                    "Illegal TYPE_0: " + code);
        }
    }
}

Который затем можно использовать так:

// Put
Bundle bundle = new Bundle();
bundle.putInt("key", MyEnum.TYPE_0.getCode());

// Get 
MyEnum myEnum = MyEnum.fromCode(bundle.getInt("key"));

0

Простой способ, присвоить целочисленное значение для enum

Смотрите следующий пример:

public enum MyEnum {

    TYPE_ONE(1), TYPE_TWO(2), TYPE_THREE(3);

    private int value;

    MyEnum(int value) {
        this.value = value;
    }

    public int getValue() {
        return value;
    }

}

Сторона отправителя:

Intent nextIntent = new Intent(CurrentActivity.this, NextActivity.class);
nextIntent.putExtra("key_type", MyEnum.TYPE_ONE.getValue());
startActivity(nextIntent);

Сторона получателя:

Bundle mExtras = getIntent().getExtras();
int mType = 0;
if (mExtras != null) {
    mType = mExtras.getInt("key_type", 0);
}

/* OR
    Intent mIntent = getIntent();
    int mType = mIntent.getIntExtra("key_type", 0);
*/

if(mType == MyEnum.TYPE_ONE.getValue())
    Toast.makeText(NextActivity.this, "TypeOne", Toast.LENGTH_SHORT).show();
else if(mType == MyEnum.TYPE_TWO.getValue())
    Toast.makeText(NextActivity.this, "TypeTwo", Toast.LENGTH_SHORT).show();
else if(mType == MyEnum.TYPE_THREE.getValue())
    Toast.makeText(NextActivity.this, "TypeThree", Toast.LENGTH_SHORT).show();
else
    Toast.makeText(NextActivity.this, "Wrong Key", Toast.LENGTH_SHORT).show();

0

Я думаю, преобразовать enum в int (для обычного enum) и затем установить на bundle было проще всего. как этот код для намерения:

myIntent.PutExtra("Side", (int)PageType.Fornt);

тогда для проверки состояния:

int type = Intent.GetIntExtra("Side",-1);
if(type == (int)PageType.Fornt)
{
    //To Do
}

но не работает для всех типов enum!


0

Я создал расширение Koltin:

fun Bundle.putEnum(key: String, enum: Enum<*>) {
    this.putString( key , enum.name )
}

inline fun <reified T: Enum<T>> Intent.getEnumExtra(key:String) : T {
    return enumValueOf( getStringExtra(key) )
}

Создайте пакет и добавьте:

Bundle().also {
   it.putEnum( "KEY" , ENUM_CLAS.ITEM )
}

и получить:

intent?.getEnumExtra< ENUM_CLAS >( "KEY" )?.let{}
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.