- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
在这个 bug report ,Doug Lea 写道(指的是 JDK 5.0 的预发布版本):
While
CopyOnWriteArraySet
is declaredCloneable
, it fails to define public clone method.
CopyOnWriteArraySet
没有实现
Cloneable
完全没有界面! (这在 Java SE 6、7 和 8 中是正确的。)
CopyOnWriteArraySet
怎么样不同于
CopyOnWriteArrayList
关于克隆?没有人想要克隆它是有充分理由的吗?
clone()
不推荐,
CopyOnWriteArraySet
基于
CopyOnWriteArrayList
内部。
最佳答案
在 secret 数据库中有一些关于此错误 (JDK-5055732) 的重要信息。我已在有关该错误的公开评论中发布了此信息,我将在此处复制它以回答此问题。
Problem
As explained in Josh Bloch's Effective Java, the Cloneable mechanism is not terribly well designed. In particular, it is impossible for a non-final class with a final reference field that needs to be unique for each object to satisfy the requirement that
x.clone().getClass() == x.getClass()
(when the class is subclassed)
CopyOnWriteArraySet, ConcurrentHashMap currently are specified to implement Cloneable. CopyOnWriteArraySet erroneously did not implement a public clone() method, while ConcurrentHashMap implemented a clone() method using a constructor, thereby not fulfilling the above requirement.
Doug Lea writes:
"Martin and Josh convinced me that we can't just add the one-line public Object clone() { return new CopyOnWriteArraySet(al); } because, as noted by Josh in Effective Java book, clone methods should not invoke constructors:
In practice, programmers assume that if they extend a class and call super.clone from within the subclass, the returned object will be an instance of the subclass. The only way that a superclass can provide this functionality is to return an object obtained by calling super.clone. If a clone method returns an object created by a normal constructor, it will not have the correct class. Therefore, if you override the clone method in a non-final class, you should always return an object obtained by invoking super.clone().
In general this means that any class with a blank final field will encounter problems because it needs to set the field inside clone. This is now possible inside JDK classes using the setAccessible loophole (see JMM list) but is ugly and slow. It seems like a better idea just to remove "implements Cloneable".
The ConcurrentHashMap class has exactly the same problem, and the same solution."
Solution
Remove "implements Cloneable" from the specification for CopyOnWriteArraySet, ConcurrentHashMap. Delete ConcurrentHashMap.clone()
Cloneable
接口(interface)public clone()
方法clone()
方法,必须super.clone()
进行实际克隆 CopyOnWriteArraySet
和
ConcurrentHashMap
两者都实现了
Cloneable
界面。但是
CopyOnWriteArraySet
没有实现
public clone()
方法,而
ConcurrentHashMap
确实实现了
public clone()
方法,它通过返回一个新构造的
ConcurrentHashMap
的实例来错误地执行此操作。 .这两个都是错误,是本错误报告的主题。
CopyOnWriteArraySet
也不是
ConcurrentHashMap
可以履行支持克隆的所有义务。因此,该错误的“修复”是让他们退出
Cloneable
。契约(Contract)。
CopyOnWriteArraySet
不能克隆的是它有一个final字段
al
指向
CopyOnWriteArrayList
存储实际元素。克隆不能与原始共享此状态,因此
clone()
方法将需要复制(或克隆)支持列表并将其存储到字段中。但是 final 字段只能存储在构造函数中,而
clone()
不是构造函数。实现者考虑并拒绝了英雄的努力,例如使用反射来编写最终字段。
public clone() { return new CopyOnWriteArraySet(al); }
这里的问题是它违反了克隆契约(Contract)。如果
CopyOnWriteArraySet
的子类支持克隆,调用
clone()
在那个子类上应该返回那个子类的一个实例。
clone()
子类的方法会正确调用
super.clone()
创建克隆。如果按照上面的方式实现,那将返回
CopyOnWriteArraySet
的实例。而不是子类的实例。因此,这将防止子类能够克隆自己。
ConcurrentHashMap
呢? ?它没有任何最终字段。嗯,当时确实如此,所以它正好遇到了从
clone()
中更新最终字段的问题。方法。
ConcurrentHashMap
的最新版本不再有最终字段。复制构造函数只需调用
putAll
在 map 参数上,它懒惰地初始化所有字段。不能
clone()
方法只需克隆、清空所有字段,然后调用
putAll()
?
关于java - 为什么 CopyOnWriteArraySet 不实现 Cloneable 接口(interface),而 CopyOnWriteArrayList 实现?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45212827/
编写一个仅用于集中其他接口(interface)的接口(interface)是好的做法还是坏的做法? interface InterfaceA : InterfaceB, InterfaceC { }
有没有一种方法可以确定具体类型从任意接口(interface)列表?我知道类型转换,但我想知道所有满意的接口(interface)。 例如,给定: type Mover interface { Mo
我正在尝试制作斐波那契堆。 (在我正在上的算法课中多次提到它们,我想检查一下。)我希望堆使用任何类型的节点,所以我定义了一个 Node 接口(interface): package node type
这是我的代码: type IA interface { FB() IB } type IB interface { Bar() string } type A struct {
示例 A: // pseudo code interface IFoo { void bar(); } class FooPlatformA : IFoo { void bar() {
合并它编译的 leppies 反馈 - 但 IMO 有一些缺点,我希望编译器强制每个子类定义它们自己的 Uri 属性。现在的代码: [] type UriUserControl() = inh
我正在构建一个项目,该项目从用户那里获取一个术语,然后执行谷歌搜索并返回一个 json 格式的标题列表。 我正在使用 serpwow API 来执行谷歌搜索并试图解析响应。 但是我收到的错误是: pa
我只想在其他接口(interface)中实现某些接口(interface),我不希望它们能够被类直接继承。 提前致谢! 最佳答案 您不能在 C# 中执行此操作 - 任何类都可以实现它有权访问的任何接口
我是 Go 的新手,还有一些我还没有掌握的技巧 例如,我有一个可以这样调用的函数: myVar.InitOperation("foo",Operator.EQUAL,"bar") myVar.Init
我有一个通用接口(interface)来描述对输出流的访问,如下所示: interface IOutput { function writeInteger(aValue:Int):Void;
我正在做一个项目,我想通过某种接口(interface)(最好是 USB)将光电探测器电路安装到计算机上。但是,由于我是新手,所以我不知道应该朝哪个方向处理这个问题。假设我有一个带有 USB 连接的光
背景 我正在尝试创建一个简单的应用程序,以真正理解DDD + TDD + etc的整个堆栈。我的目标是在运行时动态注入DAL存储库类。这让我 域和应用程序服务层可测试。我打算用“穷人的DI”来完成 现
在 Java 中,接口(interface)扩展接口(interface)是完全合法的。 UML 中的这种关系看起来像“扩展”关系(实线、闭合、未填充的箭头)还是“实现”关系(虚线、闭合、未填充的箭头
我想创建一个具有相等和比较函数默认实现的接口(interface)。 如果我从类型 IKeyable 中删除所有内容除了Key成员,只要我不添加默认实现,它就是一个有效的接口(interface)。从
COM 中的双接口(interface)是能够通过 DispInterface 或 VTable 方法访问的接口(interface)。 现在有人可以告诉我这两种方法之间到底有什么区别吗? 我认为 V
我有一个类方法,它返回一个可以迭代的员工列表。返回列表的最佳方式是什么?通常我只返回一个 ArrayList。然而,据我了解,界面更适合这种类型的操作。哪个是最好使用的界面?另外,为什么返回接口(in
我想从包装类外部实例化一个内部非静态接口(interface)。 这可能吗? 考虑以下代码: shared class AOuterClass() { Integer val = 3; shared
我为一个类编写了一个接口(interface),如下所示: public interface IGenericMultipleRepository { Lazy> addresses { ge
我是 UML 的初学者,现在我正在创建一个序列图,问题是我想根据用户输入实现 DAO 接口(interface)。如何在时序图中正确绘制以实现接口(interface)。 最佳答案 您不会在 SD 上
要使用 jsr 303 验证创建有条件验证的组,请将接口(interface)类传递给注释,如下所示: @NotEmpty (groups={UpdateValue.class}) 我有很多不同的接口
我是一名优秀的程序员,十分优秀!