java策略模式

如果都使用if-else的話,就會使用代碼變的臃腫,而且難以複用。那我們就可以根據不同的情況,將不同的方式封裝成不同的策略,將策略與它的使用對象分離開來。

案例:

  • 定義註解,標註範圍
/**
 * @author Gjing
 * 價格範圍註解
 **/
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface PriceRegion {
    int min() default 0;
    int max() default Integer.MAX_VALUE;
}
  • 具體策略
/**
 * @author Gjing
 * 抽象策略
 **/
public interface Price {
    BigDecimal getPrice(BigDecimal price);
}

/**
 * @author Gjing
 * 會員,六折
 **/
@PriceRegion(min = 10000,max = 20000)
class Member implements Price {
    @Override
    public BigDecimal getPrice(BigDecimal price) {
        return price.multiply(new BigDecimal(0.6 + ""));
    }
}

/**
 * @author Gjing
 * 普通客戶
 **/
@PriceRegion(max = 10000)
class Ordinary implements Price {
    @Override
    public BigDecimal getPrice(BigDecimal price) {
        return price;
    }
}

/**
 * @author Gjing
 * 超級會員,4折
 **/ 
@PriceRegion(min = 20000)
class SuperMember implements Price{
    @Override
    public BigDecimal getPrice(BigDecimal price) {
        return price.multiply(new BigDecimal(0.4+""));
    }
}
  • 策略上下文和策略工廠
/**
 * @author Gjing
 * 上下文
 **/
class PriceContext {

    BigDecimal getPrice(BigDecimal costPrice) throws Exception {
        Price price = PriceFactory.getInstance().getPrice(costPrice);
        return price.getPrice(costPrice);
    }
}
/**
 * @author Gjing
 * 策略工廠
 **/
class PriceFactory {

    private static final PriceFactory FACTORY = new PriceFactory();
    /**
     * 策略類集合
     */
    private List<Class<? extends Price>> priceList = new ArrayList<>();


    private PriceFactory() {
        //這邊是可以改造成自動獲取文件路徑的哈
        priceList.add(Member.class);
        priceList.add(Ordinary.class);
        priceList.add(SuperMember.class);
    }

    /**
     * 獲取對應金額的策略類
     * @param price 金額
     * @return Price
     * @throws Exception classNotFound
     */
    Price getPrice(BigDecimal price) throws Exception{
        for (Class<? extends Price> clazz : priceList) {
            PriceRegion priceRegion = clazz.getAnnotation(PriceRegion.class);
            if (price.compareTo(new BigDecimal(priceRegion.max())) < 0 && price.compareTo(new BigDecimal(priceRegion.min())) > 0) {
                return clazz.newInstance();
            }
        }
        return null;
    }
    
    static PriceFactory getInstance() {
        return FACTORY;
    }
}
  • 調用:
/**
 * @author Gjing
 **/
public class Test {
    public static void main(String[] args) throws Exception {
        PriceContext priceContext = new PriceContext();
        System.out.println(priceContext.getPrice(new BigDecimal(200)));
        System.out.println(priceContext.getPrice(new BigDecimal(11100)));
        System.out.println(priceContext.getPrice(new BigDecimal(30000)));
    }
}

以上爲個人理解,如有誤歡迎各位指正

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章