Android: Як помістити Enum в пакет?


332

Як додати об’єкт Enum до пакету Android?


11
На мою думку, поради від співробітників Google погані. Енуми дуже зручні і страждають, що описані накладні покриваються.
ognian

3
ви можете переглянути відповіді та прийняти другий, якщо вважаєте, що це може бути кращим вибором.
philipp

6
Під заголовком "Уникнення енумів" у вищенаведеному посиланні тепер йдеться про це: Міфи щодо ефективності Попередні версії цього документа висували різні оманливі претензії. Деякі з них ми звертаємося тут.
StackOverflowed

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

Відповіді:


726

Енуми є серіалізаційними, тому проблем не виникає.

З огляду на наступний перелік:

enum YourEnum {
  TYPE1,
  TYPE2
}

Комплект:

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

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

Намір:

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

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

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

21
Я би проголосував за вашу відповідь, але питання полягало в тому, щоб додати Енума до пачки, і ваша відповідь пояснює, як додати його до наміру ... Звичайно, це майже те ж саме, але Алехандро нижче виправив вашу відповідь.
Pooks

2
при використанні його з Bundle він кидаєClassNotFoundException
Відображати ім'я

2
це може бути дуже повільним і не масштабується до масивів речей , які містять перерахування і т.д. Див stackoverflow.com/a/5551155/175156
yincrash

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

164

Я знаю, що це давнє питання, але я прийшов із тією ж проблемою, і хотів би поділитися тим, як я його вирішив. Ключовим є те, що сказав Мігель: «Енуми» є послідовними.

З огляду на наступний перелік:

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
}

Ви можете помістити перерахунок у пучок:

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

І поверніть перерахунок:

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

32

Я використовую kotlin.

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дуже важливий шлях
Фан Ван Лін

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

@GabrielDeOliveiraRohden, я не впевнений , що метод розширення, має бути , як це здається тільки використання уникнути ефекту .nameв putString(). З Котліном це вже впорядковано, якщо використовується .apply. Наприклад :ContentFragment.newInstance(Bundle().apply { putString(FEED_TYPE_KEY, SAVED.name) })
Адам Гурвіц

@AdamHurwitz, чи не пропонована функція розширення є всією точкою функцій розширення Котлінса? Це змушує вас не помилятися, це ідеально! Посилання @GabrielDeOliveiraRohden «сек bundle.putEnum(key, enum) | bundle.getEnum<>(key)
Yokich

17

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

Більш тривале пояснення: Перетворення від enum порядкового до enum типу


1
Упорядкування не має значення, чи відбувається серіалізація-> десеріалізація негайно під час виконання, наприклад, при переході з однієї активності в іншу. Це може бути проблемою в таких процесах, як пересилання Інтенту з одного додатка на старіші версії програми.
Мігель

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-значень.
РоданV5500


2

Для наміру можна скористатися таким чином:

Намір: котлін

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

Простий спосіб присвоїти цілому значенню перерахунок

Дивіться наступний приклад:

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), а потім встановити на пакет було найпростішим способом. як цей код з наміром:

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

потім для стану перевірки:

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

але не працює для всіх типів перерахунків!


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.