คุณจะเพิ่มอ็อบเจกต์ Enum ลงใน Android Bundle ได้อย่างไร
คุณจะเพิ่มอ็อบเจกต์ Enum ลงใน Android Bundle ได้อย่างไร
คำตอบ:
Enums เป็น Serializable ดังนั้นจึงไม่มีปัญหา
รับ enum ต่อไปนี้:
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");
outState.putSerializable("trollData", game.getFunkyTrolls());
โหลด: game.setFunkyTrolls((Game.FunkyTroll[]) savedInstanceState.getSerializable("trollData"));
?
ClassNotFoundException
ฉันรู้ว่านี่เป็นคำถามเก่า แต่ฉันมาพร้อมกับปัญหาเดียวกันและฉันอยากจะแบ่งปันวิธีที่ฉันแก้ไขมัน กุญแจสำคัญคือสิ่งที่มิเกลพูดว่า: Enums เป็นอนุกรม
รับ enum ต่อไปนี้:
enum YourEnumType {
ENUM_KEY_1,
ENUM_KEY_2
}
ใส่:
Bundle args = new Bundle();
args.putSerializable("arg", YourEnumType.ENUM_KEY_1);
เพื่อความสมบูรณ์นี่คือตัวอย่างเต็มรูปแบบของวิธีการใส่และรับคืน enum จากชุดข้อมูล
รับ enum ต่อไปนี้:
enum EnumType{
ENUM_VALUE_1,
ENUM_VALUE_2
}
คุณสามารถใส่ Enum เป็นมัด:
bundle.putSerializable("enum_key", EnumType.ENUM_VALUE_1);
และรับ Enum กลับมา:
EnumType enumType = (EnumType)bundle.getSerializable("enum_key");
ฉันใช้ 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))
.name
เป็นเส้นทางที่สำคัญมาก
.name
ในputString()
เท่านั้น ด้วย Kotlin .apply
มันคล่องตัวแล้วถ้าใช้ ตัวอย่างเช่น :ContentFragment.newInstance(Bundle().apply { putString(FEED_TYPE_KEY, SAVED.name) })
bundle.putEnum(key, enum) | bundle.getEnum<>(key)
มันอาจจะดีกว่าที่จะผ่านมันเป็นสตริงจาก myEnumValue.name () และเรียกคืนจาก YourEnums.valueOf (s) มิฉะนั้นคำสั่งของ enum จะต้องได้รับการเก็บรักษาไว้!
คำอธิบายอีกต่อไป: แปลงจาก enum ordinal เป็น enum type
ตัวเลือกอื่น:
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))
, การใช้งาน Parcelable ที่นี่ซ้ำซ้อนและไร้สาระ
Parcelable
เป็นวิธีแก้ปัญหาที่ดีในการจัดเก็บอาร์เรย์ของค่า Enum
ใช้ bundle.putSerializable (คีย์ String, Serializable s) และ bundle.getSerializable (คีย์ String):
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
สำหรับเจตนาคุณสามารถใช้วิธีนี้:
เจตนา: kotlin
กิจกรรมแรก:
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?)
}
สิ่งหนึ่งที่ต้องระวัง - หากคุณใช้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"));
วิธีง่ายๆกำหนดค่าจำนวนเต็มให้กับ 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();
ฉันคิดว่าเปลี่ยน enum เป็น int (สำหรับ enum ปกติ) จากนั้นการตั้งค่าบนบันเดิลเป็นวิธีที่ง่ายที่สุด ชอบรหัสนี้สำหรับความตั้งใจ:
myIntent.PutExtra("Side", (int)PageType.Fornt);
จากนั้นตรวจสอบสถานะ:
int type = Intent.GetIntExtra("Side",-1);
if(type == (int)PageType.Fornt)
{
//To Do
}
แต่ใช้ไม่ได้กับ enum ทุกประเภท!
ฉันสร้างส่วนขยาย 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{}