Skip to content

Commit b96f893

Browse files
committed
update
1 parent 7ec5073 commit b96f893

File tree

1 file changed

+282
-1
lines changed

1 file changed

+282
-1
lines changed

docs/book/19-Type-Information.md

Lines changed: 282 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -1112,14 +1112,295 @@ Mouse=2}
11121112
<!-- Registered Factories -->
11131113
## 注册工厂
11141114

1115+
`Pet` 层次结构生成对象的问题是,每当向层次结构中添加一种新类型的 `Pet` 时,必须记住将其添加到 `LiteralPetCreator.java` 中的条目中。在一个定期添加更多类的系统中,这可能会成为问题。
1116+
1117+
你可能会考虑向每个子类添加静态初始值设定项,因此初始值设定项会将其类添加到某个列表中。不幸的是,静态初始值设定项仅在首次加载类时调用,因此存在鸡和蛋的问题:生成器的列表中没有类,因此它无法创建该类的对象,因此类不会被加载并放入列表中。
1118+
1119+
基本上,你必须自己手工创建列表(除非你编写了一个工具来搜索和分析源代码,然后创建和编译列表)。所以你能做的最好的事情就是把列表集中放在一个明显的地方。层次结构的基类可能是最好的地方。
1120+
1121+
我们在这里所做的另一个更改是使用*工厂方法*设计模式将对象的创建推迟到类本身。工厂方法可以以多态方式调用,并为你创建适当类型的对象。事实证明,`java.util.function.Supplier``T get()` 描述了原型工厂方法。协变返回类型允许 `get()``Supplier` 的每个子类实现返回不同的类型。
1122+
1123+
在本例中,基类 `Part` 包含一个工厂对象的静态列表,列表成员类型为 `Supplier<Part>`。对于应该由 `get()` 方法生成的类型的工厂,通过将它们添加到 `prototypes` 列表向基类“注册”。奇怪的是,这些工厂本身就是对象的实例。此列表中的每个对象都是用于创建其他对象的*原型*
1124+
1125+
```java
1126+
// typeinfo/RegisteredFactories.java
1127+
// 注册工厂到基础类
1128+
import java.util.*;
1129+
import java.util.function.*;
1130+
import java.util.stream.*;
1131+
1132+
class Part implements Supplier<Part> {
1133+
@Override
1134+
public String toString() {
1135+
return getClass().getSimpleName();
1136+
}
1137+
1138+
static List<Supplier<? extends Part>> prototypes =
1139+
Arrays.asList(
1140+
new FuelFilter(),
1141+
new AirFilter(),
1142+
new CabinAirFilter(),
1143+
new OilFilter(),
1144+
new FanBelt(),
1145+
new PowerSteeringBelt(),
1146+
new GeneratorBelt()
1147+
);
1148+
1149+
private static Random rand = new Random(47);
1150+
public Part get() {
1151+
int n = rand.nextInt(prototypes.size());
1152+
return prototypes.get(n).get();
1153+
}
1154+
}
1155+
1156+
class Filter extends Part {}
1157+
1158+
class FuelFilter extends Filter {
1159+
@Override
1160+
public FuelFilter get() { return new FuelFilter(); }
1161+
}
1162+
1163+
class AirFilter extends Filter {
1164+
@Override
1165+
public AirFilter get() { return new AirFilter(); }
1166+
}
1167+
1168+
class CabinAirFilter extends Filter {
1169+
@Override
1170+
public CabinAirFilter get() {
1171+
return new CabinAirFilter();
1172+
}
1173+
}
1174+
1175+
class OilFilter extends Filter {
1176+
@Override
1177+
public OilFilter get() { return new OilFilter(); }
1178+
}
1179+
1180+
class Belt extends Part {}
1181+
1182+
class FanBelt extends Belt {
1183+
@Override
1184+
public FanBelt get() { return new FanBelt(); }
1185+
}
1186+
1187+
class GeneratorBelt extends Belt {
1188+
@Override
1189+
public GeneratorBelt get() {
1190+
return new GeneratorBelt();
1191+
}
1192+
}
1193+
1194+
class PowerSteeringBelt extends Belt {
1195+
@Override
1196+
public PowerSteeringBelt get() {
1197+
return new PowerSteeringBelt();
1198+
}
1199+
}
1200+
1201+
public class RegisteredFactories {
1202+
public static void main(String[] args) {
1203+
Stream.generate(new Part())
1204+
.limit(10)
1205+
.forEach(System.out::println);
1206+
}
1207+
}
1208+
1209+
/* 输出:
1210+
GeneratorBelt
1211+
CabinAirFilter
1212+
GeneratorBelt
1213+
AirFilter
1214+
PowerSteeringBelt
1215+
CabinAirFilter
1216+
FuelFilter
1217+
PowerSteeringBelt
1218+
PowerSteeringBelt
1219+
FuelFilter
1220+
*/
1221+
```
1222+
1223+
并非层次结构中的所有类都应实例化;这里的 `Filter``Belt` 只是分类器,这样你就不会创建任何一个类的实例,而是只创建它们的子类(请注意,如果尝试这样做,你将获得 `Part` 基类的行为)。
1224+
1225+
因为 `Part implements Supplier<Part>``Part` 通过其 `get()` 方法供应其他 `Part`。如果为基类 `Part` 调用 `get()`(或者如果 `generate()` 调用 `get()`),它将创建随机特定的 `Part` 子类型,每个子类型最终都从 `Part` 继承,并重写相应的 `get()` 以生成它们中的一个。
11151226

11161227
<!-- Instanceof vs. Class Equivalence -->
11171228
## 类的等价比较
11181229

1230+
When you are querying for type information, there's an important difference between either form of `instanceof` (that is, `instanceof` or `isInstance()`, which produce equivalent results) and the direct comparison of the `Class` objects. Here's an example that demonstrates the difference:
1231+
1232+
```java
1233+
// typeinfo/FamilyVsExactType.java
1234+
// instanceof 与 class 的差别
1235+
// {java typeinfo.FamilyVsExactType}
1236+
package typeinfo;
1237+
1238+
class Base {}
1239+
class Derived extends Base {}
1240+
1241+
public class FamilyVsExactType {
1242+
static void test(Object x) {
1243+
System.out.println(
1244+
"Testing x of type " + x.getClass());
1245+
System.out.println(
1246+
"x instanceof Base " + (x instanceof Base));
1247+
System.out.println(
1248+
"x instanceof Derived " + (x instanceof Derived));
1249+
System.out.println(
1250+
"Base.isInstance(x) " + Base.class.isInstance(x));
1251+
System.out.println(
1252+
"Derived.isInstance(x) " +
1253+
Derived.class.isInstance(x));
1254+
System.out.println(
1255+
"x.getClass() == Base.class " +
1256+
(x.getClass() == Base.class));
1257+
System.out.println(
1258+
"x.getClass() == Derived.class " +
1259+
(x.getClass() == Derived.class));
1260+
System.out.println(
1261+
"x.getClass().equals(Base.class)) "+
1262+
(x.getClass().equals(Base.class)));
1263+
System.out.println(
1264+
"x.getClass().equals(Derived.class)) " +
1265+
(x.getClass().equals(Derived.class)));
1266+
}
1267+
public static void main(String[] args) {
1268+
test(new Base());
1269+
test(new Derived());
1270+
}
1271+
}
1272+
/* 输出:
1273+
Testing x of type class typeinfo.Base
1274+
x instanceof Base true
1275+
x instanceof Derived false
1276+
Base.isInstance(x) true
1277+
Derived.isInstance(x) false
1278+
x.getClass() == Base.class true
1279+
x.getClass() == Derived.class false
1280+
x.getClass().equals(Base.class)) true
1281+
x.getClass().equals(Derived.class)) false
1282+
Testing x of type class typeinfo.Derived
1283+
x instanceof Base true
1284+
x instanceof Derived true
1285+
Base.isInstance(x) true
1286+
Derived.isInstance(x) true
1287+
x.getClass() == Base.class false
1288+
x.getClass() == Derived.class true
1289+
x.getClass().equals(Base.class)) false
1290+
x.getClass().equals(Derived.class)) true
1291+
*/
1292+
```
1293+
1294+
`test()` 方法使用两种形式的 `instanceof` 对其参数执行类型检查。然后,它获取 `Class` 引用,并使用 `==``equals()` 测试 `Class` 对象的相等性。令人放心的是,`instanceof``isInstance()` 产生的结果与 `equals()``==` 完全相同。但测试本身得出了不同的结论。与类型的概念一致,`instanceof` 说的是“你是这个类,还是从这个类派生的类?”。另一方面,如果使用 `==` 比较实际的 `Class` 对象,则与继承无关 —— 它要么是确切的类型,要么不是。
1295+
11191296
<!-- Reflection: Runtime Class Information -->
1297+
## 反射:运行时类信息
1298+
1299+
如果你不知道对象的确切类型,RTTI 会告诉你。但是,有一个限制:必须在编译时知道类型,才能使用 RTTI 检测它,并对信息做一些有用的事情。换句话说,编译器必须知道你使用的所有类。
1300+
1301+
起初,这看起来并没有那么大的限制,但是假设你被赋予了一个对不在程序空间中的对象的引用。实际上,该对象的类在编译时甚至对程序都不可用。也许你从磁盘文件或网络连接中获得了大量的字节,并被告知这些字节代表一个类。由于这个类在编译器为你的程序生成代码后很长时间才会出现,你如何使用这样的类?
1302+
1303+
在传统编程环境中,这是一个牵强的场景。但是,当我们进入一个更大的编程世界时,会有一些重要的情况发生。第一个是基于组件的编程,你可以在应用程序构建器*集成开发环境*中使用*快速应用程序开发*(RAD)构建项目。这是一种通过将表示组件的图标移动到窗体上来创建程序的可视化方法。然后,通过在程序时设置这些组件的一些值来配置这些组件。这种设计时配置要求任何组件都是可实例化的,它公开自己的部分,并且允许读取和修改其属性。此外,处理*图形用户界面*(GUI)事件的组件必须公开有关适当方法的信息,以便 IDE 可以帮助程序员覆盖这些事件处理方法。反射提供了检测可用方法并生成方法名称的机制。
1304+
1305+
在运行时发现类信息的另一个令人信服的动机是提供跨网络在远程平台上创建和执行对象的能力。这称为*远程方法调用*(RMI),它使 Java 程序的对象分布在许多机器上。这种分布有多种原因。如果你想加速一个计算密集型的任务,你可以把它分解成小块放到空闲的机器上。或者你可以将处理特定类型任务的代码(例如,多层次客户机/服务器体系结构中的“业务规则”)放在特定的机器上,这样机器就成为描述这些操作的公共存储库,并且可以很容易地更改它以影响系统中的每个人。分布式计算还支持专门的硬件,这些硬件可能擅长于某个特定的任务——例如矩阵转换——但对于通用编程来说不合适或过于昂贵。
1306+
1307+
`Class` 支持*反射*的概念,以及 `java.lang.reflect` 库,其中包含类 `Field``Method``Constructor`(每一个都实现了 `Member` 接口)。这些类型的对象由 JVM 在运行时创建,以表示未知类中的对应成员。然后,可以使用 `Constructor` 创建新对象,`get()``set()` 方法读取和修改与 `Field` 对象关联的字段,`invoke()` 方法调用与 `Method` 对象关联的方法。此外,还可以调用便利方法 `getFields()``getMethods()``getConstructors()` 等,以返回表示字段、方法和构造函数的对象数组。(你可以通过在 JDK 文档中查找类 `Class` 来了解更多信息。)因此,匿名对象的类信息可以在运行时完全确定,编译时不需要知道任何信息。
1308+
1309+
重要的是要意识到反射没有什么魔力。当你使用反射与未知类型的对象交互时,JVM 将查看该对象,并看到它属于特定的类(就像普通的 RTTI)。在对其执行任何操作之前,必须加载 `Class` 对象。因此,该特定类型的 `.class` 文件必须在本地计算机上或通过网络对 JVM 仍然可用。因此,RTTI 和反射的真正区别在于,使用 RTTI 时,编译器在编译时会打开并检查 `.class` 文件。换句话说,你可以用“正常”的方式调用一个对象的所有方法。通过反射,`.class` 文件在编译时不可用;它由运行时环境打开并检查。
1310+
1311+
### 类方法提取器
1312+
1313+
通常,你不会直接使用反射工具,但它们可以帮助你创建更多的动态代码。反射是用来支持其他 Java 特性的,例如对象序列化(参见[附录:对象序列化](#ch040.xhtml#appendix-object-serialization))。但是,有时动态提取有关类的信息很有用。
1314+
1315+
考虑一个类方法提取器。查看类定义的源代码或 JDK 文档,只显示*在该类定义中*定义或重写的方法。但是,可能还有几十个来自基类的可用方法。找到它们既单调又费时。
1316+
1317+
```java
1318+
// typeinfo/ShowMethods.java
1319+
// 使用反射展示一个类的所有方法,甚至包括定义在基类中方法
1320+
// {java ShowMethods ShowMethods}
1321+
import java.lang.reflect.*;
1322+
import java.util.regex.*;
1323+
1324+
public class ShowMethods {
1325+
private static String usage =
1326+
"usage:\n" +
1327+
"ShowMethods qualified.class.name\n" +
1328+
"To show all methods in class or:\n" +
1329+
"ShowMethods qualified.class.name word\n" +
1330+
"To search for methods involving 'word'";
1331+
1332+
private static Pattern p = Pattern.compile("\\w+\\.");
1333+
public static void main(String[] args) {
1334+
1335+
if(args.length < 1) {
1336+
System.out.println(usage);
1337+
System.exit(0);
1338+
}
1339+
int lines = 0;
1340+
try {
1341+
Class<?> c = Class.forName(args[0]);
1342+
Method[] methods = c.getMethods();
1343+
Constructor[] ctors = c.getConstructors();
1344+
if(args.length == 1) {
1345+
for(Method method : methods)
1346+
System.out.println(
1347+
p.matcher(
1348+
method.toString()).replaceAll(""));
1349+
for(Constructor ctor : ctors)
1350+
System.out.println(
1351+
p.matcher(ctor.toString()).replaceAll(""));
1352+
lines = methods.length + ctors.length;
1353+
} else {
1354+
for(Method method : methods)
1355+
if(method.toString().contains(args[1])) {
1356+
System.out.println(p.matcher(
1357+
method.toString()).replaceAll(""));
1358+
lines++;
1359+
}
1360+
for(Constructor ctor : ctors)
1361+
if(ctor.toString().contains(args[1])) {
1362+
System.out.println(p.matcher(
1363+
ctor.toString()).replaceAll(""));
1364+
lines++;
1365+
}
1366+
}
1367+
} catch(ClassNotFoundException e) {
1368+
System.out.println("No such class: " + e);
1369+
}
1370+
}
1371+
}
1372+
/* 输出:
1373+
public static void main(String[])
1374+
public final void wait() throws InterruptedException
1375+
public final void wait(long,int) throws
1376+
InterruptedException
1377+
public final native void wait(long) throws
1378+
InterruptedException
1379+
public boolean equals(Object)
1380+
public String toString()
1381+
public native int hashCode()
1382+
public final native Class getClass()
1383+
public final native void notify()
1384+
public final native void notifyAll()
1385+
public ShowMethods()
1386+
*/
1387+
```
1388+
1389+
`Class` 方法 `getmethods()` 和'getconstructors()` 分别返回 `Method` 数组和 `Constructor` 数组。这些类中的每一个都有进一步的方法来解析它们所表示的方法的名称、参数和返回值。但你也可以像这里所做的那样,使用 `toString()`,生成带有整个方法签名的 `String`。代码的其余部分提取命令行信息,确定特定签名是否与目标 `String`(使用 `indexOf()`)匹配,并使用正则表达式(在 [Strings](#ch021.xhtml#strings) 一章中介绍)删除名称限定符。
1390+
1391+
编译时无法知道 `Class.forName()` 生成的结果,因此所有方法签名信息都是在运行时提取的。如果你研究 JDK 反射文档,你将看到有足够的支持来实际设置和对编译时完全未知的对象进行方法调用(本书后面有这样的例子)。虽然最初你可能认为你永远都不需要这样做,但是反射的全部价值可能会令人惊讶。
1392+
1393+
上面的输出来自命令行:
1394+
1395+
```java
1396+
java ShowMethods ShowMethods
1397+
```
1398+
1399+
输出包含一个 `public` 无参数构造函数,即使未定义构造函数。你看到的构造函数是由编译器自动合成的。如果将 `ShowMethods` 设置为非 `public` 类(即只有包级访问权),则合成的无参数构造函数将不再显示在输出中。自动为合成的无参数构造函数授予与类相同的访问权。
11201400

1121-
## 内省:反射运行时类信息
1401+
尝试运行 `java ShowMethods java.lang.String`,并附加一个 `char``int``String` 等参数。
11221402

1403+
编程时,当你不记得某个类是否有特定的方法,并且不想在 JDK 文档中搜索索引或类层次结构时,或者如果你不知道该类是否可以对 `Color` 对象执行任何操作时,该工具能节省不少时间。
11231404

11241405
<!-- Dynamic Proxies -->
11251406
## 动态代理

0 commit comments

Comments
 (0)