概述
大多數好的設計者象躲避瘟疫一樣來避免使用實現繼承(extends 關系)
Extends是有害的
好的設計者在他的代碼中
Interface和Class
一次
失去了靈活性
為什麼你應該避免實現繼承呢?第一個問題是明確的使用具體類名將你固定到特定的實現
在當前的敏捷編程方法中
勝於實現你也許需要的特征
對於Inteface的編程是靈活結構的核心
[/代碼]
f()
{ LinkedList list = new LinkedList();
//
g( list );
}
g( LinkedList list )
{
list
g
}
[/代碼]
現在
象下面這樣重寫代碼:
[/代碼]
f()
{ Collection list = new LinkedList();
//
g( list );
}
g( Collection list )
{
list
g
}
[/代碼]
這樣修改Linked list成hash
作為另一個例子
[/代碼]
f()
{ Collection c = new HashSet();
//
g( c );
}
g( Collection c )
{
for( Iterator i = erator(); i
do_something_with( i
}
[/代碼]
和
[/代碼]
f
{ Collection c = new HashSet();
//
g
}
g
{ while( i
do_something_with( i
}
[/代碼]
g
耦合
對於實現繼承
作為一個設計者
這裡
脆弱的基類問題
現在
讓我們一起檢查脆弱的基類和基類耦合的問題
[/代碼]
class Stack extends ArrayList
{ private int stack_pointer =
public void push( Object article )
{ add( stack_pointer++
}
public Object pop()
{ return remove(
}
public void push_many( Object[] articles )
{ for( int i =
push( articles[i] );
}
}
[/代碼]
甚至一個象這樣簡單的類也有問題。Tw.wiNGwIT.Com思考當一個用戶平衡繼承和用ArrayList的clear()方法去彈出堆棧時:
[/代碼]
Stack a_stack = new Stack();
a_stack.push("1");
a_stack.push("2");
a_stack.clear();
[/代碼]
這個代碼成功編譯,但是因為基類不知道關於stack指針堆棧的情況,這個stack對象當前在一個未定義的狀態。下一個對於push()調用把新的項放入索引2的位置。(stack_pointer的當前值),所以stack有效地有三個元素-下邊兩個是垃圾。(Java的stack類正是有這個問題,不要用它).
對這個令人討厭的繼承的方法問題的解決辦法是為Stack覆蓋所有的ArrayList方法,那能夠修改數組的狀態,所以覆蓋正確的操作Stack指針或者拋出一個例外。(removeRange()方法對於拋出一個例外一個好的候選方法)。
這個方法有兩個缺點。第一,如果你覆蓋了所有的東西,這個基類應該真正的是一個interface,而不是一個class。如果你不用任何繼承方法,在實現繼承中就沒有這一點。第二,更重要的是,你不能夠讓一個stack支持所有的ArrayList方法。例如,令人煩惱的removeRange()沒有什麼作用。唯一實現無用方法的合理的途徑是使它拋出一個例外,因為它應該永遠不被調用。這個方法有效的把編譯錯誤成為運行錯誤。不好的方法是,如果方法只是不被定義,編譯器會輸出一個方法未找到的錯誤。如果方法存在,但是拋出一個例外,你只有在程序真正的運行時,你才能夠發現調用錯誤。
對於這個基類問題的一個更好的解決辦法是封裝數據結構代替用繼承。這是新的和改進的Stack的版本:
[/代碼]
class Stack
{
private int stack_pointer = 0;
private ArrayList the_data = new ArrayList();
public void push( Object article )
{
the_data.add( stack_poniter++, article );
}
public Object pop()
{
return the_data.remove( --stack_pointer );
}
public void push_many( Object[] articles )
{
for( int i = 0; i < o.length; ++i )
push( articles[i] );
}
}
From:http://tw.wingwit.com/Article/program/Java/JSP/201311/19788.html