91在线视频播放_欧美日韩精品一区二区_2020国产成人精品免费视频_国产嫩草影院

您的位置:首頁 >熱點 > 正文

解剖屎山,尋覓黃金之第二彈 世界看熱訊

大家好,我3y啊。由于去重邏輯重構了幾次,好多股東直呼看不懂,于是我今天再安排一波對代碼的解析吧。austin支持兩種去重的類型:N分鐘相同內容達到N次去重和一天內N次相同渠道頻次去重。


(相關資料圖)

在最開始,我的第一版實現是這樣的:

publicvoidduplication(TaskInfotaskInfo){//配置示例:{"contentDeduplication":{"num":1,"time":300},"frequencyDeduplication":{"num":5}}JSONObjectproperty=JSON.parseObject(config.getProperty(DEDUPLICATION_RULE_KEY,AustinConstant.APOLLO_DEFAULT_VALUE_JSON_OBJECT));JSONObjectcontentDeduplication=property.getJSONObject(CONTENT_DEDUPLICATION);JSONObjectfrequencyDeduplication=property.getJSONObject(FREQUENCY_DEDUPLICATION);//文案去重DeduplicationParamcontentParams=DeduplicationParam.builder().deduplicationTime(contentDeduplication.getLong(TIME)).countNum(contentDeduplication.getInteger(NUM)).taskInfo(taskInfo).anchorState(AnchorState.CONTENT_DEDUPLICATION).build();contentDeduplicationService.deduplication(contentParams);//運營總規則去重(一天內用戶收到最多同一個渠道的消息次數)Longseconds=(DateUtil.endOfDay(newDate()).getTime()-DateUtil.current())/1000;DeduplicationParambusinessParams=DeduplicationParam.builder().deduplicationTime(seconds).countNum(frequencyDeduplication.getInteger(NUM)).taskInfo(taskInfo).anchorState(AnchorState.RULE_DEDUPLICATION).build();frequencyDeduplicationService.deduplication(businessParams);}

那時候很簡單,基本主體邏輯都寫在這個入口上了,應該都能看得懂。后來,群里滴滴哥表示這種代碼不行,不能一眼看出來它干了什么。于是怒提了一波pull request重構了一版,入口是這樣的:

publicvoidduplication(TaskInfotaskInfo){//配置樣例:{"contentDeduplication":{"num":1,"time":300},"frequencyDeduplication":{"num":5}}Stringdeduplication=config.getProperty(DeduplicationConstants.DEDUPLICATION_RULE_KEY,AustinConstant.APOLLO_DEFAULT_VALUE_JSON_OBJECT);//去重DEDUPLICATION_LIST.forEach(key->{DeduplicationParamdeduplicationParam=builderFactory.select(key).build(deduplication,key);if(deduplicationParam!=null){deduplicationParam.setTaskInfo(taskInfo);DeduplicationServicededuplicationService=findService(key+SERVICE);deduplicationService.deduplication(deduplicationParam);}});}

我猜想他的思路就是把構建去重參數和選擇具體的去重服務給封裝起來了,在最外層的代碼看起來就很簡潔了。后來又跟他聊了下,他的設計思路是這樣的:考慮到以后會有其他規則的去重就把去重邏輯單獨封裝起來了,之后用策略模版的設計模式進行了重構,重構后的代碼 模版不變,支持各種不同策略的去重,擴展性更高更強更簡潔

確實牛逼。

我基于上面的思路微改了下入口,代碼最終演變成這樣:

publicvoidduplication(TaskInfotaskInfo){//配置樣例:{"deduplication_10":{"num":1,"time":300},"deduplication_20":{"num":5}}StringdeduplicationConfig=config.getProperty(DEDUPLICATION_RULE_KEY,CommonConstant.EMPTY_JSON_OBJECT);//去重ListdeduplicationList=DeduplicationType.getDeduplicationList();for(IntegerdeduplicationType:deduplicationList){DeduplicationParamdeduplicationParam=deduplicationHolder.selectBuilder(deduplicationType).build(deduplicationConfig,taskInfo);if(Objects.nonNull(deduplicationParam)){deduplicationHolder.selectService(deduplicationType).deduplication(deduplicationParam);}}}

到這,應該大多數人還能跟上吧?在講具體的代碼之前,我們先來簡單看看去重功能的代碼結構(這會對后面看代碼有幫助)

去重的邏輯可以統一抽象為:在X時間段內達到了Y閾值,還記得我曾經說過:「去重」的本質:「業務Key」+「存儲」。那么去重實現的步驟可以簡單分為(我這邊存儲就用的Redis):

通過Key從Redis獲取記錄判斷該Key在Redis的記錄是否符合條件符合條件的則去重,不符合條件的則重新塞進Redis更新記錄

為了方便調整去重的參數,我把X時間段和Y閾值都放到了配置里{"deduplication_10":{"num":1,"time":300},"deduplication_20":{"num":5}}。目前有兩種去重的具體實現:

1、5分鐘內相同用戶如果收到相同的內容,則應該被過濾掉

2、一天內相同的用戶如果已經收到某渠道內容5次,則應該被過濾掉

從配置中心拿到配置信息了以后,Builder就是根據這兩種類型去構建出DeduplicationParam,就是以下代碼:

DeduplicationParamdeduplicationParam=deduplicationHolder.selectBuilder(deduplicationType).build(deduplicationConfig,taskInfo);

Builder和DeduplicationService都用了類似的寫法(在子類初始化的時候指定類型,在父類統一接收,放到Map里管理)

而統一管理著這些服務有個中心的地方,我把這取名為DeduplicationHolder

/***@authorhuskey*@date2022/1/18*/@ServicepublicclassDeduplicationHolder{privatefinalMapbuilderHolder=newHashMap<>(4);privatefinalMapserviceHolder=newHashMap<>(4);publicBuilderselectBuilder(Integerkey){returnbuilderHolder.get(key);}publicDeduplicationServiceselectService(Integerkey){returnserviceHolder.get(key);}publicvoidputBuilder(Integerkey,Builderbuilder){builderHolder.put(key,builder);}publicvoidputService(Integerkey,DeduplicationServiceservice){serviceHolder.put(key,service);}}

前面提到的業務Key,是在AbstractDeduplicationService的子類下構建的:

而具體的去重邏輯實現則都在LimitService下,{一天內相同的用戶如果已經收到某渠道內容5次}是在SimpleLimitService中處理使用mget和pipelineSetEX就完成了實現。而{5分鐘內相同用戶如果收到相同的內容}是在SlideWindowLimitService中處理,使用了lua腳本完成了實現。

LimitService的代碼都來源于@caolongxiu的pull request,建議大家可以對比commit再學習一番:https://gitee.com/zhongfucheng/austin/pulls/19

1、頻次去重采用普通的計數去重方法,限制的是每天發送的條數。

2、內容去重采用的是新開發的基于redis中zset的滑動窗口去重,可以做到嚴格控制單位時間內的頻次。

3、redis使用lua腳本來保證原子性和減少網絡io的損耗

4、redis的key增加前綴做到數據隔離(后期可能有動態更換去重方法的需求)

5、把具體限流去重方法從DeduplicationService抽取出來,DeduplicationService只需設置構造器注入時注入的AbstractLimitService(具體限流去重服務)類型即可動態更換去重的方法 6、使用雪花算法生成zset的唯一value,score使用的是當前的時間戳

針對滑動窗口去重,有會引申出新的問題:limit.lua的邏輯?為什么要移除時間窗口的之前的數據?為什么ARGV[4]參數要唯一?為什么要expire?

A: 使用滑動窗口可以保證N分鐘達到N次進行去重?;瑒哟翱诳梢曰仡櫹耇CP的,也可以回顧下刷LeetCode時的一些題,那這為什么要移除,就不陌生了。

為什么ARGV[4]要唯一,具體可以看看zadd這條命令,我們只需要保證每次add進窗口內的成員是唯一的,那么就不會觸發有更新的操作(我認為這樣設計會更加簡單些),而唯一Key用雪花算法比較方便。

為什么expire?,如果這個key只被調用一次。那就很有可能在redis內存常駐了,expire能避免這種情況。

推薦項目

最后再叨叨吧,很多人可能會發一段截圖,跑來問我為什么要這樣寫,為什么要以這種方式實現,能不能以這種方式實現。這時候,我更想看到的是:你已經實現了第二種方式了,然后探討你寫的這種方案好不好,現有的代碼差在哪里。

畢竟問問題很簡單,我又不是客服,總不能沒誠意的問題我都得一一回答吧。

如果想學Java項目的,我還是強烈推薦我的開源項目消息推送平臺Austin,可以用作畢業設計,可以用作校招,可以看看生產環境是怎么推送消息的。

倉庫地址(可點擊閱讀原文跳轉):https://gitee.com/zhongfucheng/austin

我開通了股東服務內容,感興趣可以點擊下方看看,主要針對的是項目喲

VIP服務

免責聲明:本文不構成任何商業建議,投資有風險,選擇需謹慎!本站發布的圖文一切為分享交流,傳播正能量,此文不保證數據的準確性,內容僅供參考

關鍵詞:

相關內容

熱門資訊

最新圖文

91在线视频播放_欧美日韩精品一区二区_2020国产成人精品免费视频_国产嫩草影院

            国内外成人激情视频| 99福利在线观看| 成人一级生活片| 日本精品一区在线| 亚洲成熟丰满熟妇高潮xxxxx| 欧美不卡在线播放| 国产又爽又黄ai换脸| 能在线观看的av网站| 免费观看美女裸体网站| 欧美日韩dvd| 亚洲第一成肉网| 日本在线播放一区二区| 欧美少妇性生活视频| 久热精品在线播放| 四季av一区二区| 欧美性大战久久久久xxx| 分分操这里只有精品| 欧美爱爱视频免费看| 国产高清www| 欧美乱大交xxxxx潮喷l头像| 黄色a级片免费看| 91嫩草国产丨精品入口麻豆| 久久无码高潮喷水| 国产男女激情视频| 亚洲国产精品久久久久婷蜜芽| 精品久久一二三| 2022亚洲天堂| av免费中文字幕| 欧美国产日韩在线播放| 久久9精品区-无套内射无码| 男人日女人bb视频| 国产一区二区视频播放| 日韩中字在线观看| 五月丁香综合缴情六月小说| 无码播放一区二区三区| 国产麻花豆剧传媒精品mv在线| 日韩小视频在线播放| www.日本在线播放| 国产精品无码一区二区在线| 成人羞羞国产免费网站| 一区二区三区国产免费| 亚洲娇小娇小娇小| 精品国产乱码久久久久久1区二区| 天天做天天干天天操| 91xxx视频| 国产真人做爰毛片视频直播| 国产精品无码av在线播放| 国产aaa一级片| 日本久久精品一区二区| 日韩av自拍偷拍| 日本美女爱爱视频| 人妻少妇精品无码专区二区 | 男人操女人免费软件| 亚洲狼人综合干| www.亚洲自拍| 日韩在线观看a| www黄色av| 搡的我好爽在线观看免费视频| www.18av.com| 欧美两根一起进3p做受视频| 丁香六月激情婷婷| 国产精品亚洲a| 国产又大又长又粗又黄| 性一交一乱一伧国产女士spa| 在线观看免费视频污| 屁屁影院ccyy国产第一页| 色欲av无码一区二区人妻| 国产一二三四在线视频| 粉嫩av一区二区三区天美传媒 | 欧美精品久久96人妻无码| av黄色在线网站| 奇米视频888| bt天堂新版中文在线地址| 色七七在线观看| www.在线观看av| 九九九在线观看视频| 欧美精品在欧美一区二区| 天天色综合社区| 黄网站欧美内射| 91视频福利网| 成年人在线观看视频免费| av在线免费观看国产| 91精产国品一二三产区别沈先生| 91好吊色国产欧美日韩在线| 奇米777在线| 亚洲精品视频导航| 六月丁香激情网| 国产对白在线播放| 国产xxxxx视频| 99久久国产综合精品五月天喷水| 蜜臀一区二区三区精品免费视频| 2018国产在线| 国产资源第一页| 色偷偷中文字幕| 国产又大又黄又猛| 日韩精品视频一区二区在线观看| 国产高清不卡无码视频| 亚洲综合日韩欧美| 成年人观看网站| 天堂а√在线中文在线| 久久久久久久久久一区| 国产精品久久久久9999小说| 久草视频这里只有精品| 最新中文字幕久久| 国产三级国产精品国产专区50| 5月婷婷6月丁香| 免费看欧美一级片| 天堂av在线中文| 韩国黄色一级大片| 免费国产成人看片在线| 欧美大片久久久| 国产成年人视频网站| 99视频精品免费| 久久精品99国产| 国产最新免费视频| 激情五月宗合网| 日本大片免费看| 日韩第一页在线观看| 伊人国产在线视频| 我要看一级黄色大片| 黄色三级视频在线| 成人午夜激情av| caopor在线视频| 日韩黄色片视频| 国产日韩一区二区在线观看| 久久精品xxx| 国产精品久久久久7777| 国产精品av免费观看| 精品国产三级a∨在线| 一级片免费在线观看视频| www.成年人| 男女污污视频网站| 中文字幕色网站| 五月六月丁香婷婷| 午夜在线视频免费观看| 色中文字幕在线观看| 天天久久综合网| 亚洲精品偷拍视频| 大西瓜av在线| 久色视频在线播放| 97视频在线免费播放| 人妻有码中文字幕| 亚洲国产精品毛片av不卡在线| 成年人小视频网站| 亚洲欧美自偷自拍另类| 天天色天天综合网| 中国黄色录像片| avav在线播放| 国产精品无码av在线播放| 欧美日韩一区二区在线免费观看| 中文字幕无码不卡免费视频| 在线观看日本一区二区| 影音先锋男人的网站| 国产成人在线小视频| 国产精品50p| 999精品视频在线| 亚洲在线观看网站| 隔壁人妻偷人bd中字| 欧美成人精品欧美一级乱| 国产三级三级看三级| 手机福利在线视频| 精品少妇无遮挡毛片| 国产高清免费在线| 日韩成人三级视频| 欧美激情精品久久久久久小说| 中文字幕在线观看日| 国产精品自拍合集| 无人在线观看的免费高清视频 | 国产欧美日韩小视频| 色诱视频在线观看| 国产大片一区二区三区| 久艹在线免费观看| 9久久婷婷国产综合精品性色 | 欧美日韩亚洲国产成人| 9色porny| 九九热免费精品视频| 大地资源网在线观看免费官网| 黑人糟蹋人妻hd中文字幕| 182午夜在线观看| 成人短视频在线观看免费| 欧美一级片中文字幕| 久久99国产精品一区| 成年人免费大片| 日本久久高清视频| 久久精品视频91| 国产超级av在线| 欧美 另类 交| av天堂永久资源网| 四虎4hu永久免费入口| 9久久婷婷国产综合精品性色 | 毛片av免费在线观看| 国产免费一区二区三区四在线播放| 欧美牲交a欧美牲交| 日韩av福利在线观看| 在线一区二区不卡| 青娱乐自拍偷拍| 超碰免费在线公开| 亚洲精品中文字幕无码蜜桃| 大陆极品少妇内射aaaaaa| 天天干天天爽天天射|