Skip to content

Commit e846a60

Browse files
committed
翻译范型到潜在类型
1 parent 8e08794 commit e846a60

File tree

1 file changed

+372
-2
lines changed

1 file changed

+372
-2
lines changed

docs/book/20-Generics.md

Lines changed: 372 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -3946,12 +3946,382 @@ with element type class typeinfo.pets.Dog
39463946

39473947
## 泛型异常
39483948

3949-
39503949
<!-- Mixins -->
3951-
## 混入
39523950

3951+
由于擦除的原因,将泛型应用于异常是非常受限的。**catch** 语句不能捕获泛型类型的异常,因为在编译期和运行时都必须知道异常的确切类型。泛型类也不能直接或间接继承自 **Throwable**(这将进一步阻止你去定义不能捕获的泛型异常)。
3952+
但是,类型参数可能会在一个方法的 **throws** 子句中用到。这使得你可以编写随检查型异常的类型而发生变化的泛型代码:
3953+
3954+
```java
3955+
// generics/ThrowGenericException.java
3956+
3957+
import java.util.*;
3958+
3959+
interface Processor<T, E extends Exception> {
3960+
void process(List<T> resultCollector) throws E;
3961+
}
3962+
3963+
class ProcessRunner<T, E extends Exception>
3964+
extends ArrayList<Processor<T, E>> {
3965+
List<T> processAll() throws E {
3966+
List<T> resultCollector = new ArrayList<>();
3967+
for(Processor<T, E> processor : this)
3968+
processor.process(resultCollector);
3969+
return resultCollector;
3970+
}
3971+
}
3972+
3973+
class Failure1 extends Exception {}
3974+
3975+
class Processor1
3976+
implements Processor<String, Failure1> {
3977+
static int count = 3;
3978+
@Override
3979+
public void process(List<String> resultCollector)
3980+
throws Failure1 {
3981+
if(count-- > 1)
3982+
resultCollector.add("Hep!");
3983+
else
3984+
resultCollector.add("Ho!");
3985+
if(count < 0)
3986+
throw new Failure1();
3987+
}
3988+
}
3989+
3990+
class Failure2 extends Exception {}
3991+
3992+
class Processor2
3993+
implements Processor<Integer, Failure2> {
3994+
static int count = 2;
3995+
@Override
3996+
public void process(List<Integer> resultCollector)
3997+
throws Failure2 {
3998+
if(count-- == 0)
3999+
resultCollector.add(47);
4000+
else {
4001+
resultCollector.add(11);
4002+
}
4003+
if(count < 0)
4004+
throw new Failure2();
4005+
}
4006+
}
4007+
4008+
public class ThrowGenericException {
4009+
public static void main(String[] args) {
4010+
ProcessRunner<String, Failure1> runner =
4011+
new ProcessRunner<>();
4012+
for(int i = 0; i < 3; i++)
4013+
runner.add(new Processor1());
4014+
try {
4015+
System.out.println(runner.processAll());
4016+
} catch(Failure1 e) {
4017+
System.out.println(e);
4018+
}
4019+
4020+
ProcessRunner<Integer, Failure2> runner2 =
4021+
new ProcessRunner<>();
4022+
for(int i = 0; i < 3; i++)
4023+
runner2.add(new Processor2());
4024+
try {
4025+
System.out.println(runner2.processAll());
4026+
} catch(Failure2 e) {
4027+
System.out.println(e);
4028+
}
4029+
}
4030+
}
4031+
/* Output:
4032+
[Hep!, Hep!, Ho!]
4033+
Failure2
4034+
*/
4035+
```
4036+
4037+
**Processor** 执行 `process()`,并且可能会抛出具有类型 **E** 的异常。`process()` 的结果存储在 `List<T>resultCollector` 中(这被称为*收集参数*)。**ProcessRunner** 有一个 `processAll()` 方法,它将执行所持有的每个 **Process** 对象,并返回 **resultCollector** 。
4038+
如果不能参数化所抛出的异常,那么由于检查型异常的缘故,将不能编写出这种泛化的代码。
4039+
4040+
## 混型
4041+
4042+
术语*混型*随时间的推移好像拥有了无数的含义,但是其最基本的概念是混合多个类的能力,以产生一个可以表示混型中所有类型的类。这往往是你最后的手段,它将使组装多个类变得简单易行。
4043+
混型的价值之一是它们可以将特性和行为一致地应用于多个类之上。如果想在混型类中修改某些东西,作为一种意外的好处,这些修改将会应用于混型所应用的所有类型之上。正由于此,混型有一点*面向方面编程* (AOP) 的味道,而方面经常被建议用来解决混型问题。
4044+
4045+
### C++中的混型
4046+
4047+
在 C++ 中,使用多重继承的最大理由,就是为了使用混型。但是,对于混型来说,更有趣、更优雅的方式是使用参数化类型,因为混型就是继承自其类型参数的类。在 C++ 中,可以很容易的创建混型,因为 C++ 能够记住其模版参数的类型。
4048+
下面是一个 C++ 示例,它有两个混型类型:一个使得你可以在每个对象中混入拥有一个时间戳这样的属性,而另一个可以混入一个序列号。
4049+
4050+
```cpp
4051+
// generics/Mixins.cpp
4052+
4053+
#include <string>
4054+
#include <ctime>
4055+
#include <iostream>
4056+
using namespace std;
4057+
4058+
template<class T> class TimeStamped : public T {
4059+
long timeStamp;
4060+
public:
4061+
TimeStamped() { timeStamp = time(0); }
4062+
long getStamp() { return timeStamp; }
4063+
};
4064+
4065+
template<class T> class SerialNumbered : public T {
4066+
long serialNumber;
4067+
static long counter;
4068+
public:
4069+
SerialNumbered() { serialNumber = counter++; }
4070+
long getSerialNumber() { return serialNumber; }
4071+
};
4072+
4073+
// Define and initialize the static storage:
4074+
template<class T> long SerialNumbered<T>::counter = 1;
4075+
4076+
class Basic {
4077+
string value;
4078+
public:
4079+
void set(string val) { value = val; }
4080+
string get() { return value; }
4081+
};
4082+
4083+
int main() {
4084+
TimeStamped<SerialNumbered<Basic>> mixin1, mixin2;
4085+
mixin1.set("test string 1");
4086+
mixin2.set("test string 2");
4087+
cout << mixin1.get() << " " << mixin1.getStamp() <<
4088+
" " << mixin1.getSerialNumber() << endl;
4089+
cout << mixin2.get() << " " << mixin2.getStamp() <<
4090+
" " << mixin2.getSerialNumber() << endl;
4091+
}
4092+
/* Output:
4093+
test string 1 1452987605 1
4094+
test string 2 1452987605 2
4095+
*/
4096+
```
4097+
4098+
在 `main()` 中, **mixin1** 和 **mixin2** 所产生的类型拥有所混入类型的所有方法。可以将混型看作是一种功能,它可以将现有类映射到新的子类上。注意,使用这种技术来创建一个混型是多么地轻而易举。基本上,只需要声明“这就是我想要的”,紧跟着它就发生了:
4099+
4100+
```
4101+
TimeStamped<SerialNumbered<Basic>> mixin1,mixin2;
4102+
```
4103+
4104+
遗憾的是,Java泛型不允许这样。擦除会忘记基类类型,因此
4105+
4106+
> 泛型类不能直接继承自一个泛型参数
4107+
4108+
这突显了许多我在Java语言设计决策(以及与这些功能一起发布)中遇到的一大问题:处理一件事很有希望,但是当您实际尝试做一些有趣的事情时,您发现自己做不到。
4109+
4110+
### 与接口混合
4111+
4112+
一种更常见的推荐解决方案是使用接口来产生混型效果,就像下面这样:
4113+
4114+
```java
4115+
// generics/Mixins.java
4116+
4117+
import java.util.*;
4118+
4119+
interface TimeStamped { long getStamp(); }
4120+
4121+
class TimeStampedImp implements TimeStamped {
4122+
private final long timeStamp;
4123+
TimeStampedImp() {
4124+
timeStamp = new Date().getTime();
4125+
}
4126+
@Override
4127+
public long getStamp() { return timeStamp; }
4128+
}
4129+
4130+
interface SerialNumbered { long getSerialNumber(); }
4131+
4132+
class SerialNumberedImp implements SerialNumbered {
4133+
private static long counter = 1;
4134+
private final long serialNumber = counter++;
4135+
@Override
4136+
public long getSerialNumber() { return serialNumber; }
4137+
}
4138+
4139+
interface Basic {
4140+
void set(String val);
4141+
String get();
4142+
}
4143+
4144+
class BasicImp implements Basic {
4145+
private String value;
4146+
@Override
4147+
public void set(String val) { value = val; }
4148+
@Override
4149+
public String get() { return value; }
4150+
}
4151+
4152+
class Mixin extends BasicImp
4153+
implements TimeStamped, SerialNumbered {
4154+
private TimeStamped timeStamp = new TimeStampedImp();
4155+
private SerialNumbered serialNumber =
4156+
new SerialNumberedImp();
4157+
@Override
4158+
public long getStamp() {
4159+
return timeStamp.getStamp();
4160+
}
4161+
@Override
4162+
public long getSerialNumber() {
4163+
return serialNumber.getSerialNumber();
4164+
}
4165+
}
4166+
4167+
public class Mixins {
4168+
public static void main(String[] args) {
4169+
Mixin mixin1 = new Mixin(), mixin2 = new Mixin();
4170+
mixin1.set("test string 1");
4171+
mixin2.set("test string 2");
4172+
System.out.println(mixin1.get() + " " +
4173+
mixin1.getStamp() + " " +
4174+
mixin1.getSerialNumber());
4175+
System.out.println(mixin2.get() + " " +
4176+
mixin2.getStamp() + " " +
4177+
mixin2.getSerialNumber());
4178+
}
4179+
}
4180+
/* Output:
4181+
test string 1 1494331663026 1
4182+
test string 2 1494331663027 2
4183+
*/
4184+
```
4185+
4186+
**Mixin** 类基本上是在使用*代理*,因此每个混入类型都要求在 **Mixin** 中有一个相应的域,而你必须在 **Mixin** 中编写所有必需的方法,将方法调用转发给恰当的对象。这个示例使用了非常简单的类,但是当使用更复杂的混型时,代码数量会急速增加。
4187+
4188+
### 使用装饰器模式
4189+
4190+
当你观察混型的使用方式时,就会发现混型概念好像与*装饰器*设计模式关系很近。装饰器经常用于满足各种可能的组合,而直接子类化会产生过多的类,因此是不实际的。
4191+
装饰器模式使用分层对象来动态透明地向单个对象中添加责任。装饰器指定包装在最初的对象周围的所有对象都具有相同的基本接口。某些事物是可装饰的,可以通过将其他类包装在这个可装饰对象的四周,来将功能分层。这使得对装饰器的使用是透明的一—无论对象是否被装饰,你都拥有一个可以向对象发送的公共消息集。装饰类也可以添加新方法,但是正如你所见,这将是受限的。
4192+
装饰器是通过使用组合和形式化结构(可装饰物/装饰器层次结构)来实现的,而混型是基于继承的。因此可以将基于参数化类型的混型当作是一种泛型装饰器机制,这种机制不需要装饰器设计模式的继承结构。
4193+
前面的示例可以被改写为使用装饰器:
4194+
4195+
```java
4196+
// generics/decorator/Decoration.java
4197+
4198+
// {java generics.decorator.Decoration}
4199+
package generics.decorator;
4200+
import java.util.*;
4201+
4202+
class Basic {
4203+
private String value;
4204+
public void set(String val) { value = val; }
4205+
public String get() { return value; }
4206+
}
4207+
4208+
class Decorator extends Basic {
4209+
protected Basic basic;
4210+
Decorator(Basic basic) { this.basic = basic; }
4211+
@Override
4212+
public void set(String val) { basic.set(val); }
4213+
@Override
4214+
public String get() { return basic.get(); }
4215+
}
4216+
4217+
class TimeStamped extends Decorator {
4218+
private final long timeStamp;
4219+
TimeStamped(Basic basic) {
4220+
super(basic);
4221+
timeStamp = new Date().getTime();
4222+
}
4223+
public long getStamp() { return timeStamp; }
4224+
}
4225+
4226+
class SerialNumbered extends Decorator {
4227+
private static long counter = 1;
4228+
private final long serialNumber = counter++;
4229+
SerialNumbered(Basic basic) { super(basic); }
4230+
public long getSerialNumber() { return serialNumber; }
4231+
}
4232+
4233+
public class Decoration {
4234+
public static void main(String[] args) {
4235+
TimeStamped t = new TimeStamped(new Basic());
4236+
TimeStamped t2 = new TimeStamped(
4237+
new SerialNumbered(new Basic()));
4238+
//- t2.getSerialNumber(); // Not available
4239+
SerialNumbered s = new SerialNumbered(new Basic());
4240+
SerialNumbered s2 = new SerialNumbered(
4241+
new TimeStamped(new Basic()));
4242+
//- s2.getStamp(); // Not available
4243+
}
4244+
}
4245+
```
4246+
4247+
产生自泛型的类包含所有感兴趣的方法,但是由使用装饰器所产生的对象类型是最后被装饰的类型。也就是说,尽管可以添加多个层,但是最后一层才是实际的类型,因此只有最后一层的方法是可视的,而混型的类型是所有被混合到一起的类型。因此对于装饰器来说,其明显的缺陷是它只能有效地工作于装饰中的一层(最后一层),而混型方法显然会更自然一些。因此,装饰器只是对由混型提出的问题的一种局限的解决方案。
4248+
4249+
### 与动态代理混合
4250+
4251+
可以使用动态代理来创建一种比装饰器更贴近混型模型的机制(查看类型信息(Type Information) 章中关于 Java 的动态代理如何工作的解释)。通过使用动态代理,所产生的类的动态类型将会是已经混入的组合类型。
4252+
由于动态代理的限制,每个被混入的类都必须是某个接口的实现:
4253+
4254+
```java
4255+
// generics/DynamicProxyMixin.java
4256+
4257+
import java.lang.reflect.*;
4258+
import java.util.*;
4259+
import onjava.*;
4260+
import static onjava.Tuple.*;
4261+
4262+
class MixinProxy implements InvocationHandler {
4263+
Map<String, Object> delegatesByMethod;
4264+
@SuppressWarnings("unchecked")
4265+
MixinProxy(Tuple2<Object, Class<?>>... pairs) {
4266+
delegatesByMethod = new HashMap<>();
4267+
for(Tuple2<Object, Class<?>> pair : pairs) {
4268+
for(Method method : pair.a2.getMethods()) {
4269+
String methodName = method.getName();
4270+
// The first interface in the map
4271+
// implements the method.
4272+
if(!delegatesByMethod.containsKey(methodName))
4273+
delegatesByMethod.put(methodName, pair.a1);
4274+
}
4275+
}
4276+
}
4277+
@Override
4278+
public Object invoke(Object proxy, Method method,
4279+
Object[] args) throws Throwable {
4280+
String methodName = method.getName();
4281+
Object delegate = delegatesByMethod.get(methodName);
4282+
return method.invoke(delegate, args);
4283+
}
4284+
@SuppressWarnings("unchecked")
4285+
public static Object newInstance(Tuple2... pairs) {
4286+
Class[] interfaces = new Class[pairs.length];
4287+
for(int i = 0; i < pairs.length; i++) {
4288+
interfaces[i] = (Class)pairs[i].a2;
4289+
}
4290+
ClassLoader cl =
4291+
pairs[0].a1.getClass().getClassLoader();
4292+
return Proxy.newProxyInstance(
4293+
cl, interfaces, new MixinProxy(pairs));
4294+
}
4295+
}
4296+
4297+
public class DynamicProxyMixin {
4298+
public static void main(String[] args) {
4299+
Object mixin = MixinProxy.newInstance(
4300+
tuple(new BasicImp(), Basic.class),
4301+
tuple(new TimeStampedImp(), TimeStamped.class),
4302+
tuple(new SerialNumberedImp(),
4303+
SerialNumbered.class));
4304+
Basic b = (Basic)mixin;
4305+
TimeStamped t = (TimeStamped)mixin;
4306+
SerialNumbered s = (SerialNumbered)mixin;
4307+
b.set("Hello");
4308+
System.out.println(b.get());
4309+
System.out.println(t.getStamp());
4310+
System.out.println(s.getSerialNumber());
4311+
}
4312+
}
4313+
/* Output:
4314+
Hello
4315+
1494331653339
4316+
1
4317+
*/
4318+
```
4319+
4320+
因为只有动态类型而不是非静态类型才包含所有的混入类型,因此这仍旧不如 C++ 的方式好,因为可以在具有这些类型的对象上调用方法之前,你被强制要求必须先将这些对象向下转型到恰当的类型。但是,它明显地更接近于真正的混型。
4321+
为了让 Java 支持混型,人们已经做了大量的工作朝着这个目标努力,包括创建了至少一种附加语言( Jam 语言),它是专门用来支持混型的。
39534322

39544323
<!-- Latent Typing -->
4324+
39554325
## 潜在类型
39564326

39574327

0 commit comments

Comments
 (0)