3dTiles 最後一塊拼圖:幾何誤差與螢幕空間誤差
目錄
基本介紹
策略模式的策略,該怎麼理解呢?在我看來,就是在面對不同的場景,採取不一樣的處理方式
策略模式屬於行為型模式,大多應用於動態在一個物件的多種行為進行切換的場景。
意圖:定義一系列的演算法,把它們一個個封裝起來,並且使它們可相互替換,以避免在多種演算法相似的場景下,使用過多的 if...else 所帶來的複雜和難以維護
注意:當一個系統的策略多於四個,就需要考慮使用混合模式,解決策略類膨脹的問題
假設有一家遊戲公司,委託我們開發一款模擬鴨子池塘的遊戲,該公司的主要產品是一種可以模擬展示多種會游泳和嘎嘎叫的鴨子的遊戲,它們要求在設計鴨子時儘量貼近生活,保持真實性。
他們的最初始的設計如下:
鴨子的抽象類
public abstract class Duck { public Duck() {} public abstract void display();// 顯示鴨子資訊 public void quack() { System.out.println("鴨子嘎嘎叫"); } public void swim() { System.out.println("鴨子會游泳"); } public void fly() { System.out.println("鴨子會飛翔"); } }
北京鴨
public class PekingDuck extends Duck {
@Override
public void display() {
System.out.println("北京鴨");
}
/**
* 由於北京鴨不會飛翔,因此需要重寫fly
*/
@Override
public void fly(){
System.out.println("北京鴨不會飛翔");
}
}
玩具鴨
public class ToyDuck extends Duck { @Override public void display() { System.out.println("玩具鴨"); } @Override public void quack() { System.out.println("玩具鴨不能叫~~~"); } @Override public void fly() { System.out.println("玩具鴨不會飛翔~~"); } }
野鴨
public class WildDuck extends Duck {
@Override
public void display() {
System.out.println("這是野鴨");
}
}
利用繼承來複用鴨子的各個行為,初衷是好的,但是沒有真正將鴨子的各個行為拆分,只是簡單的複用,就會有很多問題。
當我們需要新新增一類鴨子,若其父類的抽象方法不滿足當前物件時,我們就需要重寫以覆蓋父類的抽象方法,這就違背了裡式替換原則且擴充套件性欠缺。
策略模式
剛剛我們也說了,策略模式主要應用 動態在一個物件的多種行為進行切換的場景
那我們開始重新構建模擬鴨子的這款遊戲吧,我們想一想能不能把鴨子的各種行為抽出來再跟鴨子進行組合, 這樣在面對不同的鴨子時進行組裝即可
鴨子的抽象類
public abstract class Duck {
//屬性,策略介面
FlyBehavior flyBehavior;
public Duck() {}
public abstract void display();// 顯示鴨子資訊
public void quack() {
System.out.println("鴨子嘎嘎叫");
}
public void swim() {
System.out.println("鴨子會游泳");
}
public void fly() {
if(flyBehavior != null){
flyBehavior.fly();
}
}
}
飛行行為
public interface FlyBehavior {
void fly();
}
行為-飛的好
public class GoodFlyBehavior implements FlyBehavior {
@Override
public void fly() {
System.out.println("飛翔技術高超");
}
}
行為-飛的一般
public class NormalFlyBehavior implements FlyBehavior {
@Override
public void fly() {
System.out.println("飛翔技術一般");
}
}
行為-不會飛
public class NoFlyBehavior implements FlyBehavior {
@Override
public void fly() {
System.out.println("不會飛翔");
}
}
北京鴨
public class PekingDuck extends Duck {
public PekingDuck(){
//給鴨子賦予飛的一般的行為
flyBehavior = new NormalFlyBehavior();
}
@Override
public void display() {
System.out.println("北京鴨");
}
}
野鴨
public class WildDuck extends Duck {
public WildDuck() {
//給鴨子賦予飛的好的行為
flyBehavior = new GoodFlyBehavior();
}
@Override
public void display() {
System.out.println("這是野鴨");
}
}
玩具鴨
public class ToyDuck extends Duck {
public ToyDuck(){
//給鴨子賦予不會飛的行為
flyBehavior = new NoFlyBehavior();
}
@Override
public void display() {
System.out.println("玩具鴨");
}
@Override
public void quack() {
System.out.println("玩具鴨不能叫~~~");
}
}
從上述的例子我們看出,將鴨子的各個行為組合到不同的鴨子中去,既增加了系統的彈性,也滿足了每個鴨子的特點。
鴨子會不會飛,飛的好不好根據不同的策略制定。
綜上: 策略模式提供了對“開閉原則”的完美支援,使用者可以在不修改原有系統的基礎上選擇演算法或行為,也可以靈活地增加新的演算法或行為
JDK-Arrays的應用
在jdk中, Arrays 的Comparator 就實現了策略模式
public class Arrays {
public static <T> void sort(T[] a, Comparator<? super T> c) {
if (c == null) {
sort(a);
} else {
if (LegacyMergeSort.userRequested)
legacyMergeSort(a, c);
else
TimSort.sort(a, 0, a.length, c, null, 0, 0);
}
}
}
public interface Comparator<T> {
int compare(T o1, T o2);
}
我們測試一下:
public class TestStrategy {
public static void main(String[] args) {
// 陣列
Integer[] data = { 9, 1, 2, 8, 4, 3 };
// 實現降序排序,返回-1放左邊,1放右邊,0保持不變
// 說明
// 1. 實現了 Comparator 介面(策略介面) , 匿名類 物件 new Comparator<Integer>(){..}
// 2. 物件 new Comparator<Integer>(){..} 就是實現了 策略介面 的物件
// 3. public int compare(Integer o1, Integer o2){} 指定具體的處理方式
Comparator<Integer> comparator = new Comparator<Integer>() {
public int compare(Integer o1, Integer o2) {
if (o1 > o2) {
return -1;
} else {
return 1;
}
};
};
// 說明
/*
* public static <T> void sort(T[] a, Comparator<? super T> c) { if (c == null)
* { sort(a); //預設方法 } else { if (LegacyMergeSort.userRequested)
* legacyMergeSort(a, c); //使用策略物件c else // 使用策略物件c TimSort.sort(a, 0, a.length,
* c, null, 0, 0); } }
*/
// 方式1
Arrays.sort(data, comparator);
System.out.println(Arrays.toString(data)); // 降序排序
// 方式2- 同時lambda 表示式實現 策略模式
Integer[] data2 = { 19, 11, 12, 18, 14, 13 };
Arrays.sort(data2, (var1, var2) -> {
if (var1.compareTo(var2) > 0) {
return -1;
} else {
return 1;
}
});
System.out.println("data2=" + Arrays.toString(data2));
}
}