- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
如 this question 中所述,java.awt.geom.Area
的equals
方法定义为
public boolean equals(Area other)
而不是覆盖 Object
中的 equals
方法。这个问题涵盖了“为什么”,我对“如何强制 Java 使用最合适的 equals
方法”感兴趣。
考虑这个例子:
public static void main(String[] args) {
Class<?> cls = Area.class;
Area a1 = new Area(new Rectangle2D.Double(1, 2, 3, 4));
Area a2 = new Area(new Rectangle2D.Double(1, 2, 3, 4));
System.out.println("Areas equal: " + a1.equals(a2)); // true
Object o1 = (Object) a1;
Object o2 = (Object) a2;
System.out.println("Objects equal: " + o1.equals(o2)); // false
// Given only cls, o1, and o2, how can I get .equals() to return true?
System.out.println("cls.cast() approach : " + cls.cast(o1).equals(cls.cast(o2))); // false
try {
Method equalsMethod = cls.getMethod("equals", cls); // Exception thrown in most cases
System.out.println("Reflection approach: " + equalsMethod.invoke(o1, o2)); // true (when cls=Area.class)
} catch (Exception e) {
e.printStackTrace();
}
}
我的问题是:给定 o1
、o2
和 cls
,其中 o1
和 o2
保证是 cls
(或子类)的实例,我怎样才能调用最合适的 equals
方法?假设 cls
是 X.class
,我想要以下行为:
X
定义了 X.equals(X)
,这是“最合适”的选择。 (例如:X
是 Area
)X
定义了 X.equals(Object)
,这是第二合适的选择。 (例如:X
是 Rectangle2D
)Object.equals(Object)
作为回退。 (例如:X
是 Path2D
)原则上,我可以使用反射来检查上述每个方法签名,但这看起来相当笨拙。有没有更简单的方法?
为清楚起见编辑:o1
、o2
和 cls
在运行时都不同,所以我不能静态转换像 ((Area) o1).equals((Area) o2)
,因为 cls
可能始终不是 Area.class
。但是可以保证 cls.isAssignableFrom(o1.getClass())
和 cls.isAssignableFrom(o2.getClass())
都是 true
.
最佳答案
您的第二个和第三个项目符号(使用 X.equals(Object)
或回退到 Object.equals(Object)
)不需要任何努力,因为这将在调用可覆盖方法 Object.equals(Object)
时无论如何都会发生,它将使用它能找到的最具体的覆盖方法。
因此,剩下的唯一任务就是调用 X.equals(X)
方法(如果适用)。为了最小化相关成本,您可以缓存结果。从 Java 7 开始,就有了类 ClassValue
允许以线程安全、延迟评估和高效查找的方式将信息与类相关联,如果需要,仍然支持关键类的垃圾收集。
因此,Java 7 解决方案可能如下所示:
import java.lang.invoke.*;
public final class EqualsOperation extends ClassValue<MethodHandle> {
public static boolean equals(Object o, Object p) {
if(o == p) return true;
if(o == null || p == null) return false;
Class<?> t1 = o.getClass(), t2 = p.getClass();
if(t1 != t2) t1 = commonClass(t1, t2);
try {
return (boolean)OPS.get(t1).invokeExact(o, p);
} catch(RuntimeException | Error unchecked) {
throw unchecked;
} catch(Throwable ex) {
throw new IllegalStateException(ex);
}
}
private static Class<?> commonClass(Class<?> t1, Class<?> t2) {
while(t1 != Object.class && !t1.isAssignableFrom(t2)) t1 = t1.getSuperclass();
return t1;
}
static final EqualsOperation OPS = new EqualsOperation();
static final MethodHandle FALLBACK;
static {
try {
FALLBACK = MethodHandles.lookup().findVirtual(Object.class, "equals",
MethodType.methodType(boolean.class, Object.class));
} catch (ReflectiveOperationException ex) {
throw new ExceptionInInitializerError(ex);
}
}
@Override
protected MethodHandle computeValue(Class<?> type) {
try {
return MethodHandles.lookup()
.findVirtual(type, "equals", MethodType.methodType(boolean.class, type))
.asType(FALLBACK.type());
} catch(ReflectiveOperationException ex) {
return FALLBACK;
}
}
}
你可以用它来测试
Object[] examples1 = { 100, "foo",
new Area(new Rectangle(10, 20)), new Area(new Rectangle(20, 20)) };
Object[] examples2 = { new Integer(100), new String("foo"),// enforce a!=b
new Area(new Rectangle(10, 20)) };
for(Object a: examples1) {
for(Object b: examples2) {
System.out.printf("%30s %30s: %b%n", a, b, EqualsOperation.equals(a, b));
}
}
从 Java 8 开始,我们可以在运行时生成功能接口(interface)的实例,这可能会提高性能,因为那时,我们在第一次遇到类型后不再执行任何反射操作:
import java.lang.invoke.*;
import java.util.function.BiPredicate;
public final class EqualsOperation extends ClassValue<BiPredicate<Object,Object>> {
public static boolean equals(Object o, Object p) {
if(o == p) return true;
if(o == null || p == null) return false;
Class<?> t1 = o.getClass(), t2 = p.getClass();
if(t1 != t2) t1 = commonClass(t1, t2);
return OPS.get(t1).test(o, p); // test(...) is not reflective
}
private static Class<?> commonClass(Class<?> t1, Class<?> t2) {
while(t1 != Object.class && !t1.isAssignableFrom(t2)) t1 = t1.getSuperclass();
return t1;
}
static final EqualsOperation OPS = new EqualsOperation();
static final BiPredicate<Object,Object> FALLBACK = Object::equals;
@Override
protected BiPredicate<Object,Object> computeValue(Class<?> type) {
if(type == Object.class) return FALLBACK;
try {
MethodType decl = MethodType.methodType(boolean.class, type);
MethodHandles.Lookup lookup = MethodHandles.lookup();
MethodHandle mh = lookup.findVirtual(type, "equals", decl);
decl = mh.type();
BiPredicate<Object,Object> p = (BiPredicate<Object,Object>)
LambdaMetafactory.metafactory(lookup, "test",
MethodType.methodType(BiPredicate.class), decl.erase(), mh, decl)
.getTarget().invoke();
return p;
} catch(Throwable ex) {
return FALLBACK;
}
}
}
用法与其他变体一样。
这里的一个关键点是可访问性。我假设,您只想支持由 public
类声明的 public
方法。不过,如果跨越模块边界,Java 9+ 可能需要微调。为了支持在应用程序代码中声明的自定义 X.equals(X)
方法,它可能需要向您的库开放自身以进行反射访问。
相等函数与其他代码(如集合)的相等逻辑不匹配的问题已经在您的问题的评论中进行了讨论。在这里,与例如类似的问题IdentityHashMap
,可能出现;小心处理……
关于Java 反射调用重载方法 Area.equals(Area),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49918691/
为了让我的代码几乎完全用 Jquery 编写,我想用 Jquery 重写 AJAX 调用。 这是从网页到 Tomcat servlet 的调用。 我目前情况的类似代码: var http = new
我想使用 JNI 从 Java 调用 C 函数。在 C 函数中,我想创建一个 JVM 并调用一些 Java 对象。当我尝试创建 JVM 时,JNI_CreateJavaVM 返回 -1。 所以,我想知
环顾四周,我发现从 HTML 调用 Javascript 函数的最佳方法是将函数本身放在 HTML 中,而不是外部 Javascript 文件。所以我一直在网上四处寻找,找到了一些简短的教程,我可以根
我有这个组件: import {Component} from 'angular2/core'; import {UserServices} from '../services/UserService
我正在尝试用 C 实现一个简单的 OpenSSL 客户端/服务器模型,并且对 BIO_* 调用的使用感到好奇,与原始 SSL_* 调用相比,它允许一些不错的功能。 我对此比较陌生,所以我可能会完全错误
我正在处理有关异步调用的难题: 一个 JQuery 函数在用户点击时执行,然后调用一个 php 文件来检查用户输入是否与数据库中已有的信息重叠。如果是这样,则应提示用户确认是否要继续或取消,如果他单击
我有以下类(class)。 public Task { public static Task getInstance(String taskName) { return new
嘿,我正在构建一个小游戏,我正在通过制作一个数字 vector 来创建关卡,该数字 vector 通过枚举与 1-4 种颜色相关联。问题是循环(在 Simon::loadChallenge 中)我将颜
我有一个java spring boot api(数据接收器),客户端调用它来保存一些数据。一旦我完成了数据的持久化,我想进行另一个 api 调用(应该处理持久化的数据 - 数据聚合器),它应该自行异
首先,这涉及桌面应用程序而不是 ASP .Net 应用程序。 我已经为我的项目添加了一个 Web 引用,并构建了各种数据对象,例如 PayerInfo、Address 和 CreditCard。但问题
我如何告诉 FAKE 编译 .fs文件使用 fsc ? 解释如何传递参数的奖励积分,如 -a和 -target:dll . 编辑:我应该澄清一下,我正在尝试在没有 MSBuild/xbuild/.sl
我使用下划线模板配置了一个简单的主干模型和 View 。两个单独的 API 使用完全相同的配置。 API 1 按预期工作。 要重现该问题,请注释掉 API 1 的 URL,并取消注释 API 2 的
我不确定什么是更好的做法或更现实的做法。我希望从头开始创建目录系统,但不确定最佳方法是什么。 我想我在需要显示信息时使用对象,例如 info.php?id=100。有这样的代码用于显示 Game.cl
from datetime import timedelta class A: def __abs__(self): return -self class B1(A):
我在操作此生命游戏示例代码中的数组时遇到问题。 情况: “生命游戏”是约翰·康威发明的一种细胞自动化技术。它由一个细胞网格组成,这些细胞可以根据数学规则生存/死亡/繁殖。该网格中的活细胞和死细胞通过
如果我像这样调用 read() 来读取文件: unsigned char buf[512]; memset(buf, 0, sizeof(unsigned char) * 512); int fd;
我用 C 编写了一个简单的服务器,并希望调用它的功能与调用其他 C 守护程序的功能相同(例如使用 ./ftpd start 调用它并使用 ./ftpd stop 关闭该实例)。显然我遇到的问题是我不知
在 dos 中,当我粘贴此命令时它会起作用: "C:\Program Files (x86)\Google\Chrome\Application\chrome.exe" https://google.
在 dos 中,当我粘贴此命令时它会起作用: "C:\Program Files (x86)\Google\Chrome\Application\chrome.exe" https://google.
我希望能够从 cmd 在我的 Windows 10 计算机上调用 python3。 我已重新安装 Python3.7 以确保选择“添加到路径”选项,但仍无法调用 python3 并使 CMD 启动 P
我是一名优秀的程序员,十分优秀!