在JAVA中除了合成與繼承兩種關係外,還有第三種關係稱爲代理,JAVA並沒有提供對它的直接支持,這是合成與繼承的中庸之道。如果我們要將一個成員對象置於要構造的類中(就像組合),但與此同時我們在新類中暴露了該成員對象的所有方法(就像繼承)。看如下代碼段:
package access;
class SpaceShipControls{
void up(int velocity){
}
void down(int velocity){
}
void left(int velocity){
}
void right(int velocity){
}
void forward(int velocity){
System.out.println(velocity);
}
void back(int velocity){
}
void turboBoost(){
}
}
public class SpaceShip extends SpaceShipControls {
private String name;
public SpaceShip(String name){
this.name = name;
}
public static void main(String[] args) {
// TODO Auto-generated method stub
SpaceShip protector =
new SpaceShip("NSEA Protector");
protector.forward(100);
}
}
SpaceShip中包含了SpaceShipControls,SpaceShipControls中的所有方法都在SpaceShip中暴露了出來。如何解決該問題,看修改後的代碼段:
package access;
import java.util.*;
class SpaceShipControls{
void up(int velocity){
}
void down(int velocity){
}
void left(int velocity){
}
void right(int velocity){
}
void forward(int velocity){
System.out.println(velocity);
}
void back(int velocity){
}
void turboBoost(){
}
}
public class SpaceShipDelegation {
private String name;
private SpaceShipControls controls = new SpaceShipControls();
public SpaceShipDelegation(String name){
this.name = name;
}
public void back(int velocity){
controls.back(velocity);
}
public void down(int velocity){
controls.down(velocity);
}
public void forward(int velocity){
controls.forward(velocity);
}
public void left(int velocity){
controls.left(velocity);
}
public void right(int velocity){
controls.right(velocity);
}
public void turboBoost(){
controls.turboBoost();
}
public void up(int velocity){
controls.up(velocity);
}
public static void main(String[] args) {
// TODO Auto-generated method stub
SpaceShipDelegation protector =
new SpaceShipDelegation("NSEA Protector");
protector.forward(100);
}
}
可以看到,上述的方法轉遞給了controls對象,而其接口由此也就與使用繼承得到的接口相同了,但在我們使用代理的時候,可以擁有更多的控制力,我們可以選擇只提供在成員對象中的方法的某個子集。