Android:如何将Enum捆绑在一起?


332

如何将Enum对象添加到Android Bundle?


11
我认为Google员工的建议是不好的。枚举非常方便,值得承受上述开销。
2010年

3
您是否可以重新考虑答案并接受第二个答案,如果您认为它可能是一个更好的选择。
philipp 2012年

6
现在,在上面的链接中的“避免枚举”标题下,它表示以下内容:性能误解本文档的先前版本提出了各种具有误导性的主张。我们在这里解决其中一些问题。
StackOverflow 2012年

该部分甚至不再显示。
纳撒尼尔·瓦格纳

Answers:


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");

此方法是否有问题:保存:正在outState.putSerializable("trollData", game.getFunkyTrolls());加载:game.setFunkyTrolls((Game.FunkyTroll[]) savedInstanceState.getSerializable("trollData"));
Moberg

21
我会投票赞成您的答案,但是问题是关于将Enum添加到Bundle中,您的答复说明了如何将其添加到Intent中。授予它几乎相同的内容,但是下面的Alejandro固定了您的答案。

2
当与Bundle一起使用时,它会抛出ClassNotFoundException
显示名称

2
这可能非常慢,并且无法缩放到包含枚举等的事物数组。请参见stackoverflow.com/a/5551155/175156
yincrash 2014年

1
@yincrash枚举使用自定义序列化,这非常快。证明:docs.oracle.com/javase/1.5.0/docs/guide/serialization/spec/…–
Miha_x64

164

我知道这是一个老问题,但是我也遇到了同样的问题,我想分享一下我是如何解决的。关键是Miguel所说的:枚举是可序列化的。

鉴于以下枚举:

enum YourEnumType {
    ENUM_KEY_1, 
    ENUM_KEY_2
}

放:

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

3
基于以下内容:stackoverflow.com/questions/15521309/…,自定义枚举不可序列化。因此,枚举中的自定义字段不会被序列化。您如何处理?
CLU

好问题@clu!也许那您应该考虑将其作为字符串传递,如stackoverflow.com/questions/609860/…所述
Alejandro Colorado

@clu不期望自定义字段被序列化。如果它只是上面代码中的普通枚举,则可以正常工作。
bluehallu's

@AlejandroColorado这对miguel的答案有什么帮助?
tir38 '16

1
Miguel的答案于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

我用科特林。

companion object {

        enum class Mode {
            MODE_REFERENCE,
            MODE_DOWNLOAD
        }
}

然后放入Intent:

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

当您净赚价值时:

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

6
这是一个很好的答案,但是可以用一种扩展方法来补充,我在这里使用它:gist.github.com/Grohden/eea5ff9d5e3ba955aa2f57ff0df2683f
Gabriel De Oliveira Rohden

.name是一条非常重要的道路
Phan Van Linh

这似乎比将Enum转换为可打包的要简单得多,如果使用Android的Room数据库库,则将进一步增加复杂性。
亚当·赫维兹

@GabrielDeOliveiraRohden,我不确定是否需要扩展方法,因为它似乎仅避免使用.namein putString()。使用Kotlin的话,已经简化了.apply例如ContentFragment.newInstance(Bundle().apply { putString(FEED_TYPE_KEY, SAVED.name) })
Adam Hurwitz

@AdamHurwitz,提议的扩展功能不是Kotlins扩展功能的全部内容吗?它迫使您不要犯错误,这是完美的!@GabrielDeOliveiraRohden的链接 bundle.putEnum(key, enum) | bundle.getEnum<>(key)
Yokich

17

最好将其作为字符串从myEnumValue.name()传递,然后从YourEnums.valueOf(s)还原,否则必须保留枚举的顺序!

更长的解释:从枚举顺序转换为枚举类型


1
排序与反序列化是否在运行时立即发生(例如,从一个活动调用到另一个活动时)无关紧要。跨过程可能会出现问题,例如将Intent从一个应用程序发送到该应用程序的较早版本。
miguel 2013年

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

对于Intent,您可以使用以下方式:

目的:科特林

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来将a 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

我认为将枚举转换为int(对于普通枚举),然后在bundle上设置是最简单的方法。像这样的意图代码:

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{}
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.