`
sammor
  • 浏览: 413284 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

是谁破坏了枚举类型?

    博客分类:
  • J2SE
阅读更多

引言:

     举类型给我们的开发带来了很多的好处,它是个非常典型的一个单例模式,但这样做优越的功能,给我们带来的不全是好处。

     当我们开发的接口给别人调用时,往往因为业务需要,添加一个字段类型等等,就也就需要调用端一起更新api包,这里想方设法的去解决这个问题,但这里深深自问题,这是否算是在破坏着这个枚举类型 了呢?

 

问题:

        为了不去经常性的去增加枚举类型的字段,我选择了加方法给出设置值的接口来做个测试,但这样的 测试 出现了一个问题。

 

示例:

   枚举类如下:

 

/**
 * @desc 配送物品类型枚举类
 */
public enum GoodsType {

	/** 卡 */
	Card("0"),

	/** 礼品 */
	Gift("1"),

	/** 奖品 */
	Prize("2");

	public GoodsType setValue(String value) {
		this.value = value;
		return this;
	}

	GoodsType(String value) {
		this.value = value;
	}

	private String value;

	public String getValue() {
		return value;
	}
}

 

    !!! 发现问题:   在这里面出现了一个问题,当这个枚举类在序列化和反序列化之后,对应的值全为默认的参数值了。不管你怎么改变它的值。

 

 

    以下就做个实验:

     服务端:

public class Server {
	public static void main(String[] args) throws IOException,
			ClassNotFoundException {

		ServerSocket serversocket = new ServerSocket(1111);
		System.out.println("端口已启动...");
		Socket socket = serversocket.accept();
		InputStream is = socket.getInputStream();
		ObjectInputStream ois = new ObjectInputStream(is);
		GoodsType goodsType = (GoodsType) ois.readObject();
		System.out.println(goodsType.getValue());
		System.out.println("测试完成");
	}
}

 

    客户端:

public class Client {

	public static void main(String[] args) throws UnknownHostException,
			IOException {
		Socket socket = new Socket("127.0.0.1", 1111);
		ObjectOutputStream oos = new ObjectOutputStream(socket
				.getOutputStream());

		GoodsType goodsType = GoodsType.Gift.setValue("22");
		System.out
				.println("before write : goodsType = " + goodsType.getValue());
		oos.writeObject(goodsType);
		System.out.println("after write : goodsType = " + goodsType.getValue());
		System.out.println("客户端测试完成");

	}
}

 

   服务端控制台显示:

端口已启动...
1
服务端测试完成

    p.s:服务端接到的值为默认值1.


    客户端控制台显示:

before write : goodsType = 22
after write : goodsType = 22
客户端测试完成

     p.s:客户端接到的值为修改后的值22.

 

 

 

问题原因:

      很感谢在这里面,很多“同学”的留言,给的原因和建议都很好,找到了这个问题的原因了。

      在传递枚举类型时,其实io中,有专门对它的传输方式,源码如下:

 

private void writeEnum(Enum en,    
           ObjectStreamClass desc,    
           boolean unshared)    
throws IOException    
{    
     bout.writeByte(TC_ENUM);    
     ObjectStreamClass sdesc = desc.getSuperDesc();    
     writeClassDesc((sdesc.forClass() == Enum.class) ? desc : sdesc, false);    
     handles.assign(unshared ? null : en);    
     writeString(en.name(), false);    
}   

    在这里面,枚举传递过去的只是类路径、一个类型名和一个实例名,也就如这里面的Gift这个名字,

    而不会把Gift这个值给传递过去,当到传服务器端的时候,服务器端转换过来的也是该枚举类,对应的服务器端Gift类的值而已。

 

结论:

     其实这样的变动想法是在场景应用上而引发了一次思考,这样的做法,其实已经在破坏了枚举类的初衷了。罪魁祸首的人是 ,但经历这一次的风暴,让我从中对枚举类了解更深刻了。呵呵。

 

 

 

 

分享到:
评论
31 楼 hoton 2011-05-23  
sammor 写道
hukai9200 写道
强迫症 强迫症~~

呵呵。。。


有点意思
30 楼 sammor 2011-05-23  
hukai9200 写道
强迫症 强迫症~~

呵呵。。。
29 楼 hukai9200 2011-05-23  
强迫症 强迫症~~
28 楼 sammor 2011-05-23  
Angi 写道
最近,我也遇到了类似的疑惑。
之前的接口也是用到了枚举,结果发现,一旦修改枚举,客户端就得替换接口包,这样对客户来说是不必要的,于是改为不用枚举了。
楼主的这个测试个人觉得还是很有意义的,谢谢!


你的应用场景和我的就用场景是相似的。只有真正碰到问题时,才会思考如何去优化它或是改造它。
27 楼 Angi 2011-05-23  
最近,我也遇到了类似的疑惑。
之前的接口也是用到了枚举,结果发现,一旦修改枚举,客户端就得替换接口包,这样对客户来说是不必要的,于是改为不用枚举了。
楼主的这个测试个人觉得还是很有意义的,谢谢!
26 楼 yizhilong28 2011-05-23  
典型的乱用枚举,枚举初衷就是是替换以前的static final XXX用的,属于严格的vo。
25 楼 riching 2011-05-23  
产品信息存DB,觉得麻烦存配置文件,XML挺好
24 楼 gu__sofia 2011-05-22  
很好    顶
23 楼 sammor 2011-05-22  
luntan08 写道
我觉得如果基础数据经常变化的,这种需求最好还是建表
建表可以公用啊.不一定就存这个卡类型

比如建表字段
table_name ,col_name,value,value_desc
前台读取的时候通过table_name和col_name来读就行
这样很多基础数据都可以放一个表里维护


嗯,对于经常性的变化,确实用这样的方式好点。
22 楼 sammor 2011-05-22  
liu.anxin 写道
sammor 写道

1、只是为了这样一个需求而去建一张专门的表,好像有点儿大材小用了。
2、编译会确实会转为static final这样的修饰进去,只是如果这样的话,加的setValue()方法里面就不应该可以把value值改变才是呀,但是为什么只有服务端没有被改变值,客户端却被修改掉了呢?


在前期需求不大的时候, 直接写入应用程序, 后面类型很多的时候使用持久化表.
客户端被修改, 在我看来, 是因为序列化. 具体还得深入再看看, 只是我的一点看法...

你这个设计有很大的问题, 序列化本身就要保证对象是可被更改的, 你却用在了不可被更改的枚举上


是的,这样破坏性代码确实是写不得的。

  private void writeEnum(Enum en,  
            ObjectStreamClass desc,  
            boolean unshared)  
 throws IOException  
    {  
 bout.writeByte(TC_ENUM);  
 ObjectStreamClass sdesc = desc.getSuperDesc();  
 writeClassDesc((sdesc.forClass() == Enum.class) ? desc : sdesc, false);  
 handles.assign(unshared ? null : en);  
 writeString(en.name(), false);  
    }  


我刚才这边做了个测试,把传输的内容显示出来,发现枚举类有自己的传输方式,传输的是这个枚举类的路径,还有这个枚举的实例名,并不会把value的值也传过去。
21 楼 luntan08 2011-05-22  
我觉得如果基础数据经常变化的,这种需求最好还是建表
建表可以公用啊.不一定就存这个卡类型

比如建表字段
table_name ,col_name,value,value_desc
前台读取的时候通过table_name和col_name来读就行
这样很多基础数据都可以放一个表里维护
20 楼 liu.anxin 2011-05-22  
sammor 写道

1、只是为了这样一个需求而去建一张专门的表,好像有点儿大材小用了。
2、编译会确实会转为static final这样的修饰进去,只是如果这样的话,加的setValue()方法里面就不应该可以把value值改变才是呀,但是为什么只有服务端没有被改变值,客户端却被修改掉了呢?


在前期需求不大的时候, 直接写入应用程序, 后面类型很多的时候使用持久化表.
客户端被修改, 在我看来, 是因为序列化. 具体还得深入再看看, 只是我的一点看法...
19 楼 sammor 2011-05-22  
qianhd 写道
sammor 写道
qianhd 写道
你这标题应该改成 是谁设计了这种垃圾代码


哈哈,是我故意这么做的,不是为了设计,只是一个实验。



那直接告诉你 这条路走不通的
在ObjectOutputStream中,所有Enum都使用下面这个方法write,


    private void writeEnum(Enum en,
			   ObjectStreamClass desc,
			   boolean unshared)
	throws IOException
    {
	bout.writeByte(TC_ENUM);
	ObjectStreamClass sdesc = desc.getSuperDesc();
	writeClassDesc((sdesc.forClass() == Enum.class) ? desc : sdesc, false);
	handles.assign(unshared ? null : en);
	writeString(en.name(), false);
    }




嗯,非常感谢,枚举类在进行传输的时候,传过去的是它的实例名,具体传输过去之后的值,是看接收端的枚举类下的实例名对应的值了。这个源码很好的说明了这个问题的原因。
18 楼 sammor 2011-05-22  
liu.anxin 写道
sammor 写道
liu.anxin 写道
枚举被拿来改值? 有这么用的吗. 拜托别滥用! 你直接用 class 不结了吗


枚举类是本来就在应用中的,只是我的新需求,要求需要支持更多的类型,以后可能还有未知的类型出来。而我又不能经常的去更新这个枚举类(因为在一个引用包中)。所以头脑风暴似的YY的试了一下。


如果是这样, 我建议你建议一个类型表, 把控制的权力交给用户和应用, 而不是应用程序.

枚举在编译时会自动将字段加上 static final 修饰.

GoodsType 编译后会就这样:
public final class GoodsType extends java.lang.Enum{
public static final GoodsType Card;

public static final GoodsType Gift;

public static final GoodsType Prize;
...
}


告诉我, 你想怎么 setter ?


嗯,你的建议很不错,
1、只是为了这样一个需求而去建一张专门的表,好像有点儿大材小用了。
2、编译会确实会转为static final这样的修饰进去,只是如果这样的话,加的setValue()方法里面就不应该可以把value值改变才是呀,但是为什么只有服务端没有被改变值,客户端却被修改掉了呢?

17 楼 liu.anxin 2011-05-22  
sammor 写道
liu.anxin 写道
枚举被拿来改值? 有这么用的吗. 拜托别滥用! 你直接用 class 不结了吗


枚举类是本来就在应用中的,只是我的新需求,要求需要支持更多的类型,以后可能还有未知的类型出来。而我又不能经常的去更新这个枚举类(因为在一个引用包中)。所以头脑风暴似的YY的试了一下。


如果是这样, 我建议你建议一个类型表, 把控制的权力交给用户和应用, 而不是应用程序.

枚举在编译时会自动将字段加上 static final 修饰.

GoodsType 编译后会就这样:
public final class GoodsType extends java.lang.Enum{
public static final GoodsType Card;

public static final GoodsType Gift;

public static final GoodsType Prize;
...
}


告诉我, 你想怎么 setter ?
16 楼 qianhd 2011-05-22  
sammor 写道
qianhd 写道
你这标题应该改成 是谁设计了这种垃圾代码


哈哈,是我故意这么做的,不是为了设计,只是一个实验。



那直接告诉你 这条路走不通的
在ObjectOutputStream中,所有Enum都使用下面这个方法write,


    private void writeEnum(Enum en,
			   ObjectStreamClass desc,
			   boolean unshared)
	throws IOException
    {
	bout.writeByte(TC_ENUM);
	ObjectStreamClass sdesc = desc.getSuperDesc();
	writeClassDesc((sdesc.forClass() == Enum.class) ? desc : sdesc, false);
	handles.assign(unshared ? null : en);
	writeString(en.name(), false);
    }
15 楼 sammor 2011-05-22  
liu.anxin 写道
枚举被拿来改值? 有这么用的吗. 拜托别滥用! 你直接用 class 不结了吗


枚举类是本来就在应用中的,只是我的新需求,要求需要支持更多的类型,以后可能还有未知的类型出来。而我又不能经常的去更新这个枚举类(因为在一个引用包中)。所以头脑风暴似的YY的试了一下。
14 楼 sammor 2011-05-22  
liu.anxin 写道
枚举被拿来改值? 有这么用的吗. 拜托别滥用! 你直接用 class 不结了吗


     很多人可能会有疑问,为什么好端端的枚举类,不只加字段而去想方设法破坏它呢。我想说的是,应用的场景各种各样,有时要需求变更过程中,而去想一些解决的方式,当然这个过程中会有很多的问题,但这样的问题,碰到了,解决了,对我而言,对是一种成长。
13 楼 liu.anxin 2011-05-22  
枚举被拿来改值? 有这么用的吗. 拜托别滥用! 你直接用 class 不结了吗
12 楼 sammor 2011-05-22  
mercyblitz 写道
sammor 写道
devworks 写道
enum是声明不变量的,你的 setValue是干什么用的?

    为了不去因为要加一种子类型而去再添加一个字段,而是提供可以改变其值,只不过字段里面设好的是默认值而已。
    卡下面,又分很多种卡类型,这样的子类型。



晕倒,那就不要用枚举了,用个类不就完了!场景使用的问题,枚举类本身及其状态应该是固定(不变的),好比远程调用需要借口,当然你要用类来专递也可以的。但是总会有不同的看法。


是的,你说的没错,场景上的问题,但往往一些已经设计好的东西,在不断的需求变更中,需要灵活的变动,而引发的对枚举类的思考。

相关推荐

    枚举大小sizeof中枚举的大小详解.pdf

    此外,由于枚举类型可能不与`int`等基本类型具有相同的大小,直接将枚举类型视为字节进行处理可能导致意外的结果,如堆栈破坏。 总之,枚举在C++中提供了命名整数常量的便利,但它并不保证严格的类型安全。了解枚举...

    C语言详解[收集].pdf

    - 枚举类型本质上是一个整数类型,可以参与整数运算,但通常不推荐这样做,因为这可能破坏枚举的语义。 - 枚举成员在内存中占用的字节数取决于实现,通常与所在系统的`int`类型相同。 - 枚举成员的值可以是任何...

    com_枚举方式实现单例模式_代码详解.rar

    为了解决这个问题,Java引入了枚举类型来实现单例模式,这是一种简洁且线程安全的方法。 枚举在Java中是特殊的类,由JVM自动管理,保证了线程安全。当枚举类被加载时,JVM会自动初始化所有的枚举实例,因此在多线程...

    完整版枚举注册表.e.rar

    4. **HKEY_CLASSES_ROOT (HKCR)**:定义了文件类型和应用程序关联,以及对象类ID(CLSID)。 5. **HKEY_CURRENT_CONFIG (HKCC)**:存储与当前硬件配置相关的设置,如屏幕分辨率和打印机设置。 6. **HKEY_DYN_DATA ...

    ATL中的集合和枚举--一.docx

    开发者只需传入枚举器类型和容器类型,以及相关的接口指针和释放对象,即可得到一个初始化好的枚举器实例。 通过理解和应用ATL的集合和枚举机制,开发者可以更高效地构建COM组件,同时保持数据安全性和访问控制。...

    er

    同样,如果您更改类型 巨集 宏会得到一个astd-ast和键入环境。 所有术语从上到下都被删除,对于同时声明的术语只有明确的循环依赖关系。 我认为包括准报价。 什么是宏? 是match宏吗? 宏总是得到continuation吗?...

    在XML模式中扩展枚举列表

    1. 使用子类型扩展:创建一个基类型,列出初始枚举值,然后定义一个子类型,增加新的枚举值。这样,旧的实现继续使用基类型,新的实现使用子类型,同时保持兼容性。 2. 使用外部枚举值源:枚举值可以从外部文件或者...

    JavaNew1-8.rar_Java 8

    最后,Java 8对枚举类型也进行了扩展,引入了接口和静态方法的概念,使得枚举类型更加灵活和强大。同时,枚举类型可以与新的Stream API结合,进行枚举常量的集合操作。 总结来说,“JavaNew1-8.pdf”这份文档很可能...

    易语言枚举系统所有ApiHook源码.zip

    1. **API导入**:易语言通过“API声明”来导入Windows API,例如`声明函数`关键字用于定义外部函数,指定函数名、参数类型和返回值类型。如`声明函数 "kernel32" "LoadLibraryA" (库名 : 字符型) 返回 (句柄型)`。 ...

    个人认为是Java方面最好的书籍

    - 枚举类型比常量类更安全,能避免实例化控制的漏洞。 - 枚举类型可以定义方法和字段,实现更丰富的功能。 - 使用`enum`关键字替代传统的`final class`模拟枚举,利用其内置的枚举方法,如`values()`和`valueOf()...

    google protobuf C++文档

    例如,`Person`消息类型包含了必填的`name`和`id`字段,选填的`email`字段,以及一个枚举类型`PhoneType`和一个嵌套的`PhoneNumber`消息类型。 2. **字段编号和类型**:每个字段都有一个唯一的整数编号,如`name = ...

    Java开发工具-jdk8-64x

    枚举类型现在可以拥有自己的方法和字段,使得枚举更加灵活。 在JDBC方面,JDK8对SQL的处理也有所改进,新增了PreparedStatement的批处理支持,提高了数据库操作的效率。 除此之外,JDK8还包括了对反射API的优化,...

    可读的枚举元数据

    枚举类型使得代码更具可读性,易于维护,并且能够强制类型安全。标题“可读的枚举元数据”指向了一个关于如何提升枚举成员可读性的方法,这涉及到元数据的使用,以及可能的本地化支持。描述中提到的“非侵入,可靠,...

    java单例模式详解Java系列2021.pdf

    另外,还有基于枚举的单例模式,Java枚举类型是线程安全的,并且它们是天生的单例。枚举单例写法简洁,能够避免反射破坏单例的问题,并且它还能够防止反序列化重新创建新的实例。使用枚举实现单例模式时,Java虚拟...

    C++代码优化方法解析

    3. **枚举类型优化**:枚举类型(`enum`)可用于优化字符串数组的处理,通过将字符串映射到整数值,利用整数运算的高速特性,同时实现自动排序和递增,极大地提升了字符串处理的效率。 #### 结论 在嵌入式系统开发中...

    jdk-api-1.8-英文-原版官方完整版版本-Java8帮助文档-CHM

    另外,Java 8还改进了枚举类型,允许枚举类型声明方法,并引入了并行数组排序。 总的来说,这个CHM文档将详细解释上述所有特性,以及Java 8 API中的其他重要变化。对于Java开发者来说,熟悉并掌握这些内容是提升...

    C# SpecialFolder

    开发者可以通过遍历指定目录(如`SpecialFolder.LocalApplicationData`,存储应用本地数据的文件夹)下的文件,根据文件类型或时间戳来判断并删除无用的临时文件或日志。这有助于节省磁盘空间,提高系统性能。 4. *...

    易语言-易语言文件枚举W 修复各种问题

    在遍历过程中,可以执行各种操作,如检查文件类型、读取文件内容或移动文件。 4. **文件属性查询**:在枚举过程中,可以获取每个文件的属性,如文件大小、创建日期、最后修改日期等,这有助于根据特定条件筛选文件...

    C++程序设计练习题

    根据提供的文件信息,我们可以归纳出一系列与C++相关的知识点,这些知识点主要集中在基本概念、枚举类型、类型转换、常量引用以及类的设计等方面。下面将详细解释这些知识点。 ### C++基本知识点详解 #### 1. 构造...

    Java语言规范(JavaSE8)英文版

    9. **新的枚举类型和遍历**:JavaSE8增加了对枚举类型的一些新特性,比如EnumSet和EnumMap,以及枚举类型的便捷遍历方法,提高了枚举的使用效率和灵活性。 以上内容只是JavaSE8中部分重要更新和新特性的概述,实际...

Global site tag (gtag.js) - Google Analytics