相关文章
【合集】Java设计模式
概述
访问者模式用到了一种双分派的技术。
Map map = new HashMap() ,map变量的静态类型(又叫明显类型)是 Map,实际类型(又叫真实类型)是 HashMap 。
根据对象的类型而对方法进行的选择,就是分派。
分派又分为两种,即静态分派和动态分派。
动态分派(通过方法重写支持)
发生在运行时期,动态分派动态地置换掉某个方法。
Java编译器在编译时期并不总是知道哪些代码会被执行,因为编译器仅仅知道对象的静态类型,而不知道对象的真实类型;
而方法的调用则是根据对象的真实类型,而不是静态类型。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
|
class DynamicDispatch {
public static void main(String[] args) {
Animal dog = new Dog();
// 方法的调用则是根据对象的真实类型,而不是静态类型。
dog.execute(); // dog
Animal cat = new Cat();
// 方法的调用则是根据对象的真实类型,而不是静态类型。
cat.execute(); // cat
}
public static class Animal {
public void execute() {
System.out.println("Animal");
}
}
public static class Dog extends Animal {
@Override
public void execute() {
System.out.println("dog");
}
}
public static class Cat extends Animal {
@Override
public void execute() {
System.out.println("cat");
}
}
}
|
静态分派(通过方法重载支持)
发生在编译时期,分派根据静态类型信息发生。如:方法重载
这个结果可能出乎一些人的意料了,为什么呢?
重载方法的分派是根据静态类型进行的,这个分派过程在编译时期就完成了。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
|
class StaticDispatch {
public static void main(String[] args) {
Animal animal = new Animal();
Animal dog = new Dog();
Animal cat = new Cat();
Execute execute = new Execute();
// 静态分派根据静态类型决定
execute.execute(animal); // Animal
execute.execute(dog); // Animal
execute.execute(cat); // Animal
}
public static class Animal {
}
public static class Dog extends Animal {
}
public static class Cat extends Animal {
}
public static class Execute {
public void execute(Animal a) {
System.out.println("Animal");
}
public void execute(Dog d) {
System.out.println("dog");
}
public void execute(Cat c) {
System.out.println("cat");
}
}
}
|
双分派(方法重写+方法重载)
是指在选择一个方法时,既考虑调用者的运行时区别,又要考虑参数的静态类型。
其实现方法其实就是在各个类中依赖了 Execute,再将自己 this 传递过去,从而实现在 Execute 操作的目的。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
|
class DoubleDispatch {
public static void main(String[] args) {
Animal animal = new Animal();
Animal dog = new Dog();
Animal cat = new Cat();
Execute execute = new Execute();
// 第1次分派:accept方法重写,根据真实类型,实现动态分派
animal.accept(execute); // animal
dog.accept(execute); // dog
cat.accept(execute); // cat
}
public static class Animal {
public void accept(Execute exe) {
// 第2次分派:execute方法重载,根据静态类型,实现静态分派
exe.execute(this);
}
}
public static class Dog extends Animal {
@Override
public void accept(Execute exe) {
exe.execute(this);
}
}
public static class Cat extends Animal {
@Override
public void accept(Execute exe) {
exe.execute(this);
}
}
public static class Execute {
public void execute(Animal a) {
System.out.println("animal");
}
public void execute(Dog d) {
System.out.println("dog");
}
public void execute(Cat c) {
System.out.println("cat");
}
}
}
|