#1 楼
枚举是可序列化的,因此没有问题。给出以下枚举:
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
13年5月22日在12:03
我会投票赞成您的答案,但是问题是关于将Enum添加到Bundle中,您的答复说明了如何将其添加到Intent中。授予它几乎相同的内容,但是下面的Alejandro固定了您的答案。
–
2013年9月27日下午5:34
与Bundle一起使用时,会引发ClassNotFoundException
–显示名称
2014年3月23日20:28在
这可能非常慢,并且无法缩放到包含枚举等的事物数组。请参见stackoverflow.com/a/5551155/175156
– yincrash
2014年6月18日19:30在
@yincrash枚举使用自定义序列化,这非常快。证明:docs.oracle.com/javase/1.5.0/docs/guide/serialization/spec/…
– Miha_x64
17年8月19日在10:14
#2 楼
我知道这是一个古老的问题,但是我也遇到了同样的问题,我想分享一下我是如何解决的。关键是Miguel所说的:枚举是可序列化的。给出以下枚举:
enum YourEnumType {
ENUM_KEY_1,
ENUM_KEY_2
}
放置:
Bundle args = new Bundle();
args.putSerializable("arg", YourEnumType.ENUM_KEY_1);
评论
基于此:stackoverflow.com/questions/15521309/…,自定义枚举不可序列化。因此,枚举中的自定义字段不会被序列化。您如何处理?
– LyteSpeed
15年1月8日在19:03
好问题@clu!也许然后您应该考虑将其作为字符串传递,如stackoverflow.com/questions/609860/…中所述。
–亚历杭德罗·科罗拉多
2015年1月11日在2:07
@clu不期望自定义字段被序列化。如果它只是上面代码中的普通枚举,则可以正常工作。
– bluehallu
16年1月11日,12:39
@AlejandroColorado这对miguel的答案有什么帮助?
–tir38
16年5月17日在23:28
Miguel的答案于2015年进行了编辑。最初的答案对捆绑包一无所知,仅显示了一个意图示例。
–亚历杭德罗·科罗拉多
16年5月17日在23:30
#3 楼
为了完整起见,这是如何从捆绑软件中放入和获取枚举的完整示例。给出以下枚举:
enum EnumType{
ENUM_VALUE_1,
ENUM_VALUE_2
}
您可以将枚举放入一个包中:
bundle.putSerializable("enum_key", EnumType.ENUM_VALUE_1);
并取回枚举:
EnumType enumType = (EnumType)bundle.getSerializable("enum_key");
#4 楼
我使用kotlin。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))
评论
这是一个很好的答案,但是可以用一种扩展方法来补充,我在这里使用它:gist.github.com/Grohden/eea5ff9d5e3ba955aa2f57ff0df2683f
–加布里埃尔·德·奥利维拉·罗登(Gabriel De Oliveira Rohden)
18年8月23日在3:59
.name是非常重要的路径
–潘凡林(Pan Van Linh)
18-10-16在9:15
这似乎比将Enum转换为可打包的要简单得多,如果使用Android的Room数据库库,则会进一步增加复杂性。
–亚当·赫维兹(Adam Hurwitz)
19-4-8在23:47
@GabrielDeOliveiraRohden,我不确定是否需要扩展方法,因为它似乎只是避免在putString()中使用.name。使用Kotlin时,如果使用.apply,则已经简化了。例如:ContentFragment.newInstance(Bundle()。apply {putString(FEED_TYPE_KEY,SAVED.name)})
–亚当·赫维兹(Adam Hurwitz)
19-4-8在23:57
@AdamHurwitz,提议的扩展功能不是Kotlins扩展功能的全部内容吗?它迫使您不要犯错,这是完美的! @GabrielDeOliveiraRohden的链接bundle.putEnum(key,枚举)| bundle.getEnum <>(键)
–约基希
19年8月15日在8:50
#5 楼
最好将其作为字符串从myEnumValue.name()传递,然后从YourEnums.valueOf(s)恢复,否则必须保留枚举的顺序!更长的解释:从枚举序数转换枚举类型
评论
排序与反序列化是否在运行时立即发生无关紧要,例如从一个活动调用到另一个活动时。跨过程可能会出现问题,例如将Intent从一个应用程序发送到该应用程序的较早版本。
– miguel
16 Jun 9'在5:27
#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());
}
}
评论
您可以使用putSerializable(key,value)/(Type)getSerializable(key)或putString(key,value.name())/ Type.valueOf(getString(key)),在这里可打包实现是多余的且毫无意义的。
– Miha_x64
17年8月19日在10:17
使用Parcelable是存储枚举值数组的好方法。
–RhodanV5500
19年3月21日在12:14
#7 楼
使用bundle.putSerializable(String key,Serializable s)和bundle.getSerializable(String key):enum Mode = {
BASIC, ADVANCED
}
Mode m = Mode.BASIC;
bundle.putSerializable("mode", m);
...
Mode m;
m = bundle.getSerializable("mode");
文档:http://developer.android.com/reference /android/os/Bundle.html
#8 楼
对于Intent,可以使用以下方式:Intent: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?)
}
#9 楼
要注意的一件事-如果使用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"));
#10 楼
我创建了一个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{}
评论
碎片呢?你在那里没有意图。你怎么得到的?
– aNdRO博士
20-09-29在20:32
在片段中,您可以使用参数。例如:arguments?.getString(“ YOUR_KEY”)
–库巴尼奇贝克(Kubanychbek Yakubov)
20/12/14在15:36
#11 楼
一种简单的方法,将整数值分配给枚举请参见以下示例:
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();
#12 楼
我认为将枚举转换为int(对于普通枚举),然后在bundle上进行设置是最简单的方法。像这样的意图代码:myIntent.PutExtra("Side", (int)PageType.Fornt);
然后检查状态:
int type = Intent.GetIntExtra("Side",-1);
if(type == (int)PageType.Fornt)
{
//To Do
}
,但不适用于所有人枚举类型!
评论
我认为Google员工的建议是不好的。枚举非常方便,值得承受上述开销。您是否可以重新考虑答案并接受第二个答案,如果您认为这可能是更好的选择。
现在,在上面的链接中的“避免枚举”标题下,它表示以下内容:性能误解本文档的先前版本提出了各种具有误导性的主张。我们在这里解决其中一些问题。
该部分甚至不再显示。
通过意图传递枚举或对象的可能副本(最佳解决方案)