Android: วิธีใส่ Enum ใน Bundle อย่างไร


332

คุณจะเพิ่มอ็อบเจกต์ Enum ลงใน Android Bundle ได้อย่างไร


11
ในความเห็นของฉันคำแนะนำจากพนักงานของ Google นั้นไม่ดี Enums สะดวกมากและต้องทนทุกข์ทรมานจากค่าใช้จ่ายที่อธิบายไว้
ognian

3
คุณสามารถทบทวนคำตอบอีกครั้งและยอมรับคำตอบที่สองหากคุณคิดว่าอาจเป็นทางเลือกที่ดีกว่า
philipp

6
ภายใต้หัวข้อ "การหลีกเลี่ยง Enums" ในลิงค์ด้านบนตอนนี้มันบอกว่า: ตำนานประสิทธิภาพรุ่นก่อนหน้าของเอกสารนี้ทำให้การเรียกร้องที่ทำให้เข้าใจผิดต่าง ๆ เราพูดถึงบางส่วนของพวกเขาที่นี่
StackOverflow

ส่วนนั้นจะไม่ปรากฏอีกต่อไป
นาธาเนียล D. Wagoner

คำตอบ:


726

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

21
ฉันจะโหวตให้กับคำตอบของคุณ แต่คำถามเกี่ยวกับการเพิ่ม Enum ใน Bundle และคำตอบของคุณจะอธิบายวิธีเพิ่มลงใน Intent ... ได้รับมันเกือบจะเหมือนกัน แต่ Alejandro ด้านล่างแก้ไขคำตอบของคุณ
Pooks

2
เมื่อใช้กับ Bundle จะมีการClassNotFoundException
แสดง

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 เป็นอนุกรม

รับ enum ต่อไปนี้:

enum YourEnumType {
    ENUM_KEY_1, 
    ENUM_KEY_2
}

ใส่:

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

3
ขึ้นอยู่กับสิ่งนี้: stackoverflow.com/questions/15521309/… Enums ที่กำหนดเองนั้นไม่สามารถต่ออนุกรมได้ ดังนั้นฟิลด์ที่กำหนดเองใน Enum จะไม่ถูกทำให้เป็นอนุกรม คุณจัดการกับสิ่งนี้ได้อย่างไร
clu

เป็นคำถามที่ดี @clu! บางทีคุณควรคิดว่าผ่านเป็นสตริงตามที่ระบุไว้ในstackoverflow.com/questions/609860/…
Alejandro Colorado

@clu โดยไม่ต้องการให้ฟิลด์ที่กำหนดเองถูกทำให้เป็นอนุกรม มันใช้งานได้ดีถ้ามันเป็น enum ปกติเหมือนในโค้ดด้านบน
bluehallu

@AlejandroColorado สิ่งนี้เพิ่มไปยังคำตอบของ miguel?
tir38

1
คำตอบของมิเกลได้รับการแก้ไขเมื่อปี 2558 คำตอบเดิมไม่ได้เกี่ยวกับการรวมกลุ่มมันเป็นเพียงตัวอย่างของเจตนา
Alejandro Colorado

41

เพื่อความสมบูรณ์นี่คือตัวอย่างเต็มรูปแบบของวิธีการใส่และรับคืน 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");

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
Gabriel De Oliveira Rohden

.nameเป็นเส้นทางที่สำคัญมาก
Phan Van Linh

ดูเหมือนว่าจะง่ายกว่าการเปลี่ยนEnumเป็นพัสดุซึ่งจะสร้างความซับซ้อนเพิ่มเติมหากทำงานกับห้องสมุดฐานข้อมูลห้องพักของ Android
Adam Hurwitz

@GabrielDeOliveiraRohden ฉันไม่แน่ใจว่าจำเป็นต้องใช้วิธีการขยายแบบใดเนื่องจากดูเหมือนว่าจะหลีกเลี่ยงการใช้งาน.nameในputString()เท่านั้น ด้วย Kotlin .applyมันคล่องตัวแล้วถ้าใช้ ตัวอย่างเช่น :ContentFragment.newInstance(Bundle().apply { putString(FEED_TYPE_KEY, SAVED.name) })
Adam Hurwitz

@ AdamHurwitz ไม่ได้มีฟังก์ชั่นการขยายที่เสนอให้กับฟังก์ชั่นการขยาย Kotlins ทั้งหมดหรือ มันบังคับให้คุณไม่ทำผิดพลาดมันสมบูรณ์แบบ! ลิงก์ของ bundle.putEnum(key, enum) | bundle.getEnum<>(key)
@GabrielDeOliveiraRohden

17

มันอาจจะดีกว่าที่จะผ่านมันเป็นสตริงจาก myEnumValue.name () และเรียกคืนจาก YourEnums.valueOf (s) มิฉะนั้นคำสั่งของ enum จะต้องได้รับการเก็บรักษาไว้!

คำอธิบายอีกต่อไป: แปลงจาก enum ordinal เป็น enum type


1
การจัดลำดับไม่สำคัญว่าซีเรียลไลเซชั่น -> ดีซีเรียลไลเซชั่นเกิดขึ้นทันทีที่รันไทม์เช่นเมื่อโทรจากกิจกรรมหนึ่งไปยังอีกกิจกรรมหนึ่ง อาจเป็นปัญหาระหว่างกระบวนการเช่นการส่ง Intents จากแอปหนึ่งไปยังแอปที่เก่ากว่า
miguel

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

กิจกรรมแรก:

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

วิธีง่ายๆกำหนดค่าจำนวนเต็มให้กับ 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 ปกติ) จากนั้นการตั้งค่าบนบันเดิลเป็นวิธีที่ง่ายที่สุด ชอบรหัสนี้สำหรับความตั้งใจ:

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{}
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.