enum类型被intent所携带时需要注意的地方

一般我们在Activity之间传递对象时多用Parcelable。比如写一个class,在这个class上标明implements Parcelable并实现接口就可以用Intent.putExtra(String, Parcelable)了。对于传递enum对象,假设也采用此方法,即像http://stackoverflow.com/questions/2836256/passing-enum-or-object-through-an-intent-the-best-solution里面3楼说的:

[java] view plaincopy

  1. public enum MyEnum implements Parcelable {
  2. VALUE;
  3. @Override
  4. public int describeContents() {
  5. return 0;
  6. }
  7. @Override
  8. public void writeToParcel(final Parcel dest, final int flags) {
  9. dest.writeInt(ordinal());
  10. }
  11. public static final Creator<MyEnum> CREATOR = new Creator<MyEnum>() {
  12. @Override
  13. public MyEnum createFromParcel(final Parcel source) {
  14. return MyEnum.values()[source.readInt()];
  15. }
  16. @Override
  17. public MyEnum[] newArray(final int size) {
  18. return new MyEnum[size];
  19. }
  20. };
  21. }
  22. You can than use Intent.putExtra(String, Parcelable).

那么我们先定义一个MyEnum变量a,再调用intent.putExtra("name", a),会报The method putExtra(String, Parcelable) is ambiguous for the type Intent的错,为何?

因为enum自身实现了Serializable接口,Enum类的源代码里这么写的:

[java] view plaincopy

  1. public abstract class Enum<E extends Enum<E>> implements Serializable, Comparable<E> {
  2. ... ...
  3. }

之后你的MyEnum类再实现Parcelable接口,而intent里面有这样两个函数:Intent.putExtra(String, Parcelable)和Intent.putExtra(String, Serializable),你的MyEnum类实现了Serializable和Parcelable两个接口,在调用Intent.putExtra时,编译器就不知道该选Intent.putExtra(String, Parcelable)还是Intent.putExtra(String, Serializable),导致二义性。那么上面的说法是否真的无用?  其实这种写法在MyEnum对象作为类Father1的成员时还是可以这么写的,我们在Activity间传Father1,Father1是可以实现Parcelable接口的。Father1内部处理MyEnum成员时可以这样:

[java] view plaincopy

  1. private Father1(Parcel in) {
  2. mField = in.readInt();
  3. mMyEnum = MyEnum.CREATOR.createFromParcel(in);
  4. }
  5. public void writeToParcel(Parcel dest, int flags) {
  6. dest.writeInt(mField);
  7. mMyEnum.writeToParcel(dest, flags);
  8. }

经测试,也可以把enum传递出去。

既然上述MyEnum的写法可以适应enum作为Parcelable类成员来传递,而不能作为单独的对象来传递,要兼得二者该如何做?

首先,MyEnum不需实现Parcelable接口,单独传递MyEnum对象时就用Intent.putExtra(String, Serializable)。

然后把MyEnum的createFromParcel()和writeToParcel()的代码整合进Father类,代码片段(变量名有改动):

[java] view plaincopy

  1. private Father2(Parcel in) {
  2. mField = in.readInt();
  3. mAnotherEnum = AnotherEnum.values()[in.readInt()];
  4. }
  5. public static final Parcelable.Creator<Father2> CREATOR = new Parcelable.Creator<Father2>() {
  6. public Father2 createFromParcel(Parcel in) {
  7. return new Father2(in);
  8. }
  9. @Override
  10. public Father2[] newArray(int size) {
  11. return new Father2[size];
  12. }
  13. };
  14. @Override
  15. public void writeToParcel(Parcel dest, int flags) {
  16. dest.writeInt(mField);
  17. dest.writeInt(mAnotherEnum.ordinal());
  18. }

再用Intent.putExtra(String, Parcelable)来传Father2对象就可以了。

再谈一下Enum类的values()方法,这个方法是无法通过eclipse查看到的,它的定义在这里,它返回所有的定义过的枚举值,enum的底层实现就是定义从0到N的数个整数,只不过为每个整数取了个别名,一个enum变量就是这数个整数之一,这个enum变量的ordinal()方法就是返回其在这数个整数中的位置。values()静态方法返回一个包含这数个整数的数组。

给出示例代码(免积分下载),在MainActivity.java三处注释的地方,分别解注释再运行,就会认识得比较清楚了。

时间: 2024-10-05 13:48:13

enum类型被intent所携带时需要注意的地方的相关文章

enum类型的本质(转)

原地址:http://www.cppblog.com/chemz/archive/2007/06/05/25578.html 至从C语言开始enum类型就被作为用户自定义分类有限集合常量的方法被引入到了语言当中,而且一度成为C++中定义编译期常量的唯一方法(后来在类中引入了静态整型常量).    根据上面对enum类型的描述,到底enum所定义出来的类型是一个什么样的类型呢?作为一个用户自定义的类型其所占用的内存空间是多少呢?使用enum类型是否真的能够起到有限集合常量的边界约束呢?大家可能都知

C#中enum类型

最近碰到了枚举类型,就顺便整理下. 枚举的基类Enum,可以是除 Char 外的任何整型.不做显示声明的话,默认是整形(Int32). 声明一个Enum类型: /// <summary> /// 颜色 /// </summary> public enum People { [Description("婴儿")] Baby = 0, [Description("儿童")] Children = 5, [Description("青年&

Java 语言中 Enum 类型的使用介绍

Enum 类型的介绍 枚举类型(Enumerated Type) 很早就出现在编程语言中,它被用来将一组类似的值包含到一种类型当中.而这种枚举类型的名称则会被定义成独一无二的类型描述符,在这一点上和常量的定义相似.不过相比较常量类型,枚举类型可以为申明的变量提供更大的取值范围. 举个例子来说明一下,如果希望为彩虹描绘出七种颜色,你可以在 Java 程序中通过常量定义方式来实现. 清单 1. 常量定义 Public static class RainbowColor { // 红橙黄绿青蓝紫七种颜

序列化类型为&quot;xx”的对象时检测到循环引用。

错误: 序列化类型为“System.Data.Entity.DynamicProxies.PClass_F1D9DFE16F480B2B87DBB269E781139A9B4611C923176D35DD120104EDEFC17C”的对象时检测到循环引用. 背景:这是在使用MVC+EF 进行ajax post请求的时候遇到的 原因:EF  Model1和Model2进行了相互引用,就是说这两个在数据中的表有主外键关系的 解决: IList<PClass> pclassLst = PClass

Hibernate 映射枚举(Enum) 类型的属性

在数据库中我们一般用整数或字符串来表示枚举值(有些数据库(如 MySQL)本身带有枚举类型), 而在使用 Hibernate 时实体对象中也用 Integer 或 String 来表示枚举就不那么友好了.试想来我们这样定义实体对象的两个属性 @Entitypublic class User {  .... public Integer type;  //0: Individual 类型,1: Company 类型  public String gender;  //可取值 Male 和 Fema

多态:多态指的是编译时类型变化,而运行时类型不变

多态:多态指的是编译时类型变化,而运行时类型不变. 多态分两种: ① 编译时多态:编译时动态重载: ②  运行时多态:指一个对象可以具有多个类型. 对象是客观的,人对对象的认识是主观的. 例: Animal a=new Dog():查看格式名称: Dog d=(Dog)a.声明父类来引用子类. (思考上面的格式) 1 package TomText; 2 3 public class TomText_19 { 4 5 public static void link(String a) 6 { 7

JSP中使用Spring注入的Bean时需要注意的地方

遇到问题 遇到一个问题:在JSP中,使用Spring注入的Bean对象时,未能正确地获取到想要的对象. 郁闷的是,它也没报错. 研究问题 使用DEBUG功能(好久不在JSP里写Java代码了,都忘了JSP也可以打断点调试),跟踪了一下代码,发现确实有了我想使用的类的实例,不过是个代理类. 想到反射.代理相关的知识,貌似知道问题在哪了. 赶紧试了一下,果然…… 解决 在JSP里你要获得的Bean对象的类型,要定义成接口类,而不是实现类. 当然,这也视情况而定,我不确定,在JSP里使用Spring注

初次用SqlServer查看本地的Excel文件时需要注意的地方

日常用到通过SqlServer 读取Excel文件的案例 ,记录下来 文件路径 :C:\Users\Administrator\Desktop\icd10.xls 1.查询语句: SELECT  *FROM    OPENDATASOURCE('Microsoft.Jet.OLEDB.4.0',                       'Data Source="C:\Users\Administrator\Desktop\icd10.xls";User ID=Admin;Pass

qt容器在并发时需要注意的地方

最近用tbb和qt写了一个延时摄影后期控制镜头的工具,主要就是扫描目录下所有图片,按照给定参数截取图片中某区域并另存,模拟镜头摆动. 扫描后的图片路径保存在qlist内,作为只读数据,交由tbb的parellel_for处理.tbb并行对qlist每个元素内的路径对应的图片进行读取,裁剪,另存操作(磁盘是ssd,这个程序在8线程的机器上,可令cpu满负荷).理想是美好的,现实是残酷的. 前提,qlist(不光qlist了,qt很多数据结构都)实现了copy on write,qlist的索引操作