通过向其添加方法来扩展Java库
作者:互联网
我试图了解如何在Java中正确地“扩展”第三方库(包括类和接口)以及如何简单地将它们用作插入式替换.
我目前正在使用Twitter4J
库.现在我每次调用时都必须手动将@前缀为User
的getScreenName()方法的结果:
twitter4j.User user = getTwitterUser();
String userHandle = "@" + user.getScreenName();
我目前尝试过一些东西,但没有任何作用.这就是我想出来的:
package my.app;
public abstract class User implements twitter4j.User {
public String getHandle() {
return "@" + getScreenName();
}
}
这样称呼它:
User user = (User) getTwitterUser();
String userHandle = user.getHandle();
这与消息崩溃:
java.lang.ClassCastException: twitter4j.UserJSONImpl cannot be cast to my.app.User
解决方法:
Java对象在现实世界中的工作方式有点像.
让我们做一个汽车比喻.你的汽车旁边有一个车库.所以你可以从车库买一辆车:
Car car = garage.buyCar(money);
车库碰巧卖的车是沃尔沃.所以,你得到一辆车,而这辆车(Car是一个接口)实际上是一辆沃尔沃.如果您知道汽车是沃尔沃汽车,那么您可以这样做:
Volvo car = (Volvo) garage.buyCar(money);
这根本不会改变车库出售汽车的方式,也不会改变车库销售的汽车类型.就是这样,因为你知道它们是volvos,你可以投出结果来获得沃尔沃汽车的特性.
请注意,这现在有效.但没有什么可以阻止车库,后来决定出售大众而不是沃尔沃.如果他们这样做,演员将不再工作,这将是你的错:车库告诉你它卖汽车,但不承诺它将是沃尔沃.
现在,您希望车库生产保时捷.想要做
Porsche car = (Porsche) garage.buyCar(money);
不行.只是因为你非常希望汽车成为一辆保时捷不会神奇地改变车库生产的沃尔沃保时捷.
你无法以这种方式扩展库
您所能做的就是获取用户并调用一个实用程序方法,该方法从该用户创建句柄.或者要创建自己的UserWithHandle类来实现User,添加一个getHandle()方法,并通过委托给原始的包装用户来实现所有其他方法:
UserWithHandle u = new UserWithHandle(getTwitterUser())
如果库允许您提供UserFactory实现,那么您可以按照自己的方式替换其创建用户的方式.有点像车库允许你提供自己的画家.
做你想要的,即在Java中不可能向现有类添加方法,但在某些语言中是可能的,例如Kotlin(与Java兼容并在JVM上运行,如果你感兴趣的话),它们在哪里称为扩展方法.基本上,这些方法是静态实用程序方法,例如,将User作为参数并返回其句柄,但可以调用它们,就好像它们是User的实例方法一样.
标签:java,casting,twitter4j 来源: https://codeday.me/bug/20190623/1269239.html