如何從代碼設計角度應對產品經理的復雜篩選邏輯

背景

有這樣一種需求,要在一個列表的數據中進行篩選.篩選規則和篩選優先級時常改變(不得不吐槽產品同學才華橫溢,富有創意).如果不采取合適的方案去應對,而且你的代碼功力又不是很強,那很可能每周加班是你的宿命了.

場景

假設現在有一個產品類,屬性如下,有類型,有單價,有讓利三個字段.

import lombok.AllArgsConstructor;
import lombok.Getter;
import lombok.Setter;

@Getter
@Setter
@AllArgsConstructor
@ToString
public class Product {
    private String type;
    private String price;
    private String benefit;
}

產品類型有AB兩種,產品A的優先級大于B,當產品的優先級一致時,價格低的優先級更高,當價格一致時,讓利大的優先級更高.現在給你一個列表的數據,你要篩選出優先級最高的數據,如果有多個,就取第一個.我們看看有幾種解決方案.

方案一:硬擼

public class Demo01 {
    public static void main(String[] args) {
        Product p1 = new Product("A", 10, 5);
        Product p2 = new Product("A", 10, 6);
        Product p3 = new Product("B", 10, 6);
        Product p4 = new Product("B", 9, 6);
        Product p5 = new Product("A", 10, 6);
        List<Product> productList = Lists.newArrayList(p1, p2, p3, p4, p5);

        Product resultProduct = new Product("B", Integer.MAX_VALUE, Integer.MAX_VALUE);

        for (Product product : productList) {
            // A 和 B 比較.A更小
            if (product.getType().compareTo(resultProduct.getType()) < 0) {
                resultProduct = product;
                continue;
            }
            if (product.getType().equals(resultProduct.getType())) {
                // 選擇價格低的
                if (product.getPrice() < resultProduct.getPrice()) {
                    resultProduct = product;
                    continue;
                }
                if (product.getPrice().intValue() == resultProduct.getPrice().intValue()) {
                    // 選擇讓利大的
                    if (product.getBenefit() > resultProduct.getBenefit()) {
                        resultProduct = product;
                    }
                }
            }
        }
        System.out.println(resultProduct);
    }
}

在這種僅僅只有三個篩選條件的情況,這種方式的代碼已經十分臃腫,而且,如果維護這份代碼,在需求頻繁更改的情況下是極其容易出錯的.更何況生產環境只會更復雜.當然你可以選擇把他們抽成一個方法.然后寫單元測試.但是后期維護工作依然巨大.

方案二: 使用List的sort方法進行排序

使用List.sort方法.傳入Compartor對象,隨后取出排序后的第一個元素.代碼如下:

public class Demo02 {
    public static void main(String[] args) {
        Product p1 = new Product("A", 10, 5);
        Product p2 = new Product("A", 10, 6);
        Product p3 = new Product("B", 10, 6);
        Product p4 = new Product("B", 9, 6);
        Product p5 = new Product("A", 10, 6);
        List<Product> productList = Lists.newArrayList(p1, p2, p3, p4, p5);
        productList.sort((pro1, pro2) -> {
            if (pro1.getType().compareTo(pro2.getType()) < 0) {
                return -1;
            }
            if (pro1.getType().equals(pro2.getType())) {
                if (pro1.getPrice() < pro2.getPrice()) {
                    return -1;
                }
                if (pro1.getPrice().intValue() == pro2.getPrice().intValue()) {
                    return pro2.getBenefit() - pro1.getBenefit();
                }
                return 1;
            }
            return 1;
        });
        System.out.println(productList.get(0));
    }
}

這種方式本質上和上面的方案一差不多,應對變化的能力差.但也是這種情況下常見的解決方案.

方案三:責任鏈

上面的兩種方案,本質上來說都是面向過程的開發方式.在這個場景下,實際上我們就是在進行一個操作:過濾.所以我們可以給過濾這個動作建模,然后針對各種不同的過濾條件進行實現.通過組裝各種過濾實現達到擴展和調整的目標.實現如下:

給過濾動作建模:

public interface IFilter<T> {
    List<T> filter(List<T> data);
}

實現按類型篩選的過濾器:

public class TypeFilter implements IFilter<Product> {

    private Ordering<Product> ordering = new Ordering<Product>() {
        @Override
        public int compare(Product product, Product t1) {
            return product.getType().compareTo(t1.getType());
        }
    };

    @Override
    public List<Product> filter(List<Product> data) {
        if (CollectionUtils.isEmpty(data)) return data;
        if (data.size() == 1) return data;

        List<Product> result = Lists.newArrayList();
        List<Product> sortResult = ordering.sortedCopy(data);

        String type = sortResult.get(0).getType();

        for (Product product : sortResult) {
            if (type.equals(product.getType())) {
                result.add(product);
            } else {
                break;
            }
        }
        return result;
    }
}

按價格篩選的過濾器

public class PriceFilter implements IFilter<Product> {

    private Ordering<Product> ordering = new Ordering<Product>() {
        @Override
        public int compare(Product product, Product t1) {
            return product.getPrice() - t1.getPrice();
        }
    };

    @Override
    public List<Product> filter(List<Product> data) {
        if (CollectionUtils.isEmpty(data)) return data;
        if (data.size() == 1) return data;

        List<Product> result = Lists.newArrayList();
        List<Product> sortResult = ordering.sortedCopy(data);

        int price = sortResult.get(0).getPrice();

        for (Product product : sortResult) {
            if (price == product.getPrice()) {
                result.add(product);
            } else {
                break;
            }
        }
        return result;
    }
}

按讓利篩選的過濾器

public class BenefitFilter implements IFilter<Product> {

    private Ordering<Product> ordering = new Ordering<Product>() {
        @Override
        public int compare(Product product, Product t1) {
            return t1.getBenefit() - product.getBenefit();
        }
    };

    @Override
    public List<Product> filter(List<Product> data) {
        if (CollectionUtils.isEmpty(data)) return data;
        if (data.size() == 1) return data;

        List<Product> result = Lists.newArrayList();
        List<Product> sortResult = ordering.sortedCopy(data);

        int benefit = sortResult.get(0).getBenefit();

        for (Product product : sortResult) {
            if (benefit == product.getBenefit()) {
                result.add(product);
            } else {
                break;
            }
        }
        return result;
    }
}

客戶端代碼

public class Demo03 {

    public static void main(String[] args) {

        Product p1 = new Product("A", 10, 5);
        Product p2 = new Product("A", 10, 6);
        Product p3 = new Product("B", 10, 6);
        Product p4 = new Product("B", 9, 6);
        Product p5 = new Product("A", 10, 6);
        List<Product> productList = Lists.newArrayList(p1, p2, p3, p4, p5);

        List<IFilter<Product>> filterChain = Lists.newArrayList(
                new TypeFilter(), new PriceFilter(), new BenefitFilter()
        );

        for(IFilter<Product> filter : filterChain){
            productList = filter.filter(productList);
        }

        System.out.println(productList.get(0));
    }
}

使用這種方式似乎代碼量更多,但是擴展性和維護性上都有了質的提升.

加規則

只需要寫一個IFilter的實現類,并在FilterChain上放到一個合適的位置即可.

不加規則,調整權重

只需要修改FilterChain上的位置即可.

還能做的更好嗎?

可以針對產品的需求將所有的Filter實現類都實例化,做成配置,接入到公司的配置管理系統.產品在已有規則下的權重調整只需要在配置系統中修改即可,不需要重新修改代碼,編譯,部署.這樣下來,你還需要加班嗎?

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,763評論 6 539
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,238評論 3 428
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 177,823評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,604評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,339評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,713評論 1 328
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,712評論 3 445
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,893評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,448評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,201評論 3 357
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,397評論 1 372
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,944評論 5 363
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,631評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,033評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,321評論 1 293
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,128評論 3 398
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,347評論 2 377

推薦閱讀更多精彩內容

  • Spring Cloud為開發人員提供了快速構建分布式系統中一些常見模式的工具(例如配置管理,服務發現,斷路器,智...
    卡卡羅2017閱讀 134,818評論 18 139
  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,711評論 25 708
  • #每天一幅禪繞畫#Day 13 黃色水晶種子年,接收夢想的清晰指引,種下美好的種子,讓自己蛻變、凈化,看見更完整的自己。
    Cindy留聲機閱讀 247評論 0 0
  • 20170601 星期四 晴 《正面管教》就是讓孩子發揮內因作用的教育方法。 《十幾歲正面管教》與《教室里的正面管...
    晴媽愛上閱讀閱讀 212評論 0 0
  • 最難忘, 你一眼望過來時的脈脈含情…… 我便嬌羞的低下頭, 任憑火熱的心情把雙頰燙紅! 最難忘, 你偷偷遞過來的紙...
    夢艷兒閱讀 181評論 0 3