日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術(shù)文章
文章詳情頁

Android scheme 跳轉(zhuǎn)的設(shè)計與實現(xiàn)詳解

瀏覽:159日期:2022-09-23 17:22:54

緣起

隨著 App 的成長,我們難免會遇到以下這些需求:

H5 跳原生界面 Notification 點擊調(diào)相關(guān)界面 根據(jù)后臺返回數(shù)據(jù)跳轉(zhuǎn)界面,例如登錄成功后跳不同界面或者根據(jù)運營需求跳不同界面 實現(xiàn) AppLink 的跳轉(zhuǎn)

為了解決這些問題,App 一般都會自定義一個 scheme 跳轉(zhuǎn)協(xié)議,多端都實現(xiàn)這個協(xié)議,以此來解決各種運營需求。今天就來解析下QMUI最新版QMUISchemeHandler的設(shè)計與實現(xiàn)。

一個 scheme 的格式大概是這樣子:

schemeName://action?param1=value1¶m2=value2

例如:

qmui://home?tab=2

從技術(shù)角度來講,實現(xiàn) scheme 的跳轉(zhuǎn)并不是件很難的事情,就是下面兩個步驟:

解析 scheme 根據(jù)解析結(jié)果跳轉(zhuǎn)指定界面

但是寫代碼時如果不加以設(shè)計,就容易是堆一堆的 if else。例如:

if(action=='action1'){ doAction1(params)}else if(action=='action2'){ doAction2(params)}else { ...}

每當有新的 scheme 添加時,就去添加一個 if,直到它逐漸變成一段巨長的爛代碼,改都改不動。因而我們要勤思考、多重構(gòu),盡早通過設(shè)計出優(yōu)良的框架來解放自己的雙手。

對于 if else 這類的重構(gòu),一個基本的方式就是用查表法,將所有的條件以及其所要執(zhí)行的行為放在一個 map 里,然后使用時通過去查詢這個 map 而獲取要執(zhí)行的行為。而我們可以通過注解配合代碼生成的方式構(gòu)建這個 map,從而減少我們代碼的編寫量。除此之外,我們還需要考慮各種功能性需求:

可以設(shè)置攔截器 interceptor,例如跳某些界面,如果是非登錄的狀態(tài),可能需要跳轉(zhuǎn)到登錄界面 參數(shù)可以指定一些基礎(chǔ)類型, scheme 所攜帶的參數(shù)的值都是字符串,但我們希望它可以方便的轉(zhuǎn)換成我們需要的基礎(chǔ)類型 同一個 action 可以根據(jù)參數(shù)的不同而有不同的跳轉(zhuǎn)行為,例如都是跳轉(zhuǎn)書籍詳情,漫畫書籍和普通書籍要跳轉(zhuǎn)的界面可能不一樣 如果當前界面已經(jīng)是目標界面,可以選擇刷新當前界面或者啟動一個新界面 對于 QMUI,是同時支持 Activity 和 Fragment 的,因而 scheme 也要同時支持這兩者 可以自定義新界面的實例化方法

接口設(shè)計

任何一個庫的開發(fā),為了讓業(yè)務(wù)使用方足夠舒心,既要保證庫的功能足夠強大,也要保證使用的方便性,QMUI Scheme 對外主要是QMUISchemeHandler這個入口類, 以及ActivityScheme和FragmentScheme兩個注解。

QMUISchemeHandler

QMUISchemeHandler通過 Builder 模式實例化:

// 設(shè)置schemeNameval instance = QMUISchemeHandler.Builder('qmui://') // 防止短時間類觸發(fā)多次相同的scheme跳轉(zhuǎn) .blockSameSchemeTimeout(1000) // scheme 參數(shù) decode .addInterpolator(new QMUISchemeParamValueDecoder()) .addInterpolator(...) // 默認 fragment 實例化 factory .defaultFragmentFactory(...) // 默認 activity 實例化 factory .defaultIntentFactory(...) // 默認 scheme 匹配器 .defaultSchemeMatcher(...) .build();if(!instance.handle('qmui://xxx')){ // scheme 未被 handle,日志記錄?}

大多數(shù)場景,QMUISchemeHandler采用單例模式即可。 其可以設(shè)置多個攔截器、設(shè)置 fragment、activity 的默認實例化工廠、以及默認的匹配器。實例工廠和匹配器都是提供了默認實現(xiàn)的,大多數(shù)場景是不需要調(diào)用者關(guān)心的。而且這里都只是設(shè)置全局默認值,到了 scheme 注解那一層,還可以為每個 scheme 指定不同的值,以滿足可能的自定義需求。

ActivityScheme 與 FragmentScheme 注解

這兩個注解是非常相似的,但是因為 Fragment 有一些更多的配置項,因為獨立出來了。

@Retention(RetentionPolicy.CLASS)@Target(ElementType.TYPE)public @interface ActivityScheme { // scheme action 名 String name(); // 必須的參數(shù)列表,用于支持同一個 action 對應(yīng)多個 scheme 的場景,每一項可以是'type=4' 來指定值,或者只傳'type'來匹配任意值 String[] required() default {}; // 如果當前界面就是 scheme 跳轉(zhuǎn)的目標值,可以選擇刷新當前界面,當然當前界面必須實現(xiàn) ActivitySchemeRefreshable boolean useRefreshIfCurrentMatched() default false; // 自定義當前 scheme 的匹配實現(xiàn)方法, 傳值為 QMUISchemeMatcher 的實現(xiàn) Class<?> customMatcher() default void.class; // 自定義當前 Activity 實例工廠,傳值為 QMUISchemeIntentFactory Class<?> customFactory() default void.class; // 指定參數(shù)的類型,支持 int/bool/long/float/double 這些基礎(chǔ)類型,不指定則為 string 類型 String[] keysWithIntValue() default {}; String[] keysWithBoolValue() default {}; String[] keysWithLongValue() default {}; String[] keysWithFloatValue() default {}; String[] keysWithDoubleValue() default {};}@Retention(RetentionPolicy.CLASS)@Target(ElementType.TYPE)public @interface FragmentScheme { // 這些參數(shù)都同 ActivityScheme String name(); String[] required() default {}; Class<?> customMatcher() default void.class; String[] keysWithIntValue() default {}; String[] keysWithBoolValue() default {}; String[] keysWithLongValue() default {}; String[] keysWithFloatValue() default {}; String[] keysWithDoubleValue() default {}; //同 ActivityScheme,但當前UI必須實現(xiàn) FragmentSchemeRefreshable boolean useRefreshIfCurrentMatched() default false; // 同 ActivityScheme, 但傳值是 QMUISchemeFragmentFactory 的實現(xiàn)類 Class<?> customFactory() default void.class; // 可以承載目標 Fragment 的 activity 列表,如果當前 activity 不在列表里,則用 activities 的第一項啟動新的 activity Class<?>[] activities(); // 是否強制啟動新的 Activity boolean forceNewActivity() default false; // 可以通過 scheme 里的參數(shù)來控制是否強制啟動新的 Activity String forceNewActivityKey() default ''; }

可以看出,我們前面所羅列的各種需求,都在 SchemeHandler 以及兩個 scheme 里體現(xiàn)出來了。

使用

對于業(yè)務(wù)使用者,我們只需要在Activity或者Fragment上加上注解。QMUISchemeHandler默認會將參數(shù)解析出來并放到Activity的 intent 里或者Fragment的 arguments 里,因而我們可以在onCreate里將我們關(guān)心的值取出來:

@ActivityScheme(name='activity1')class Activity1: QMUIActivity{ override fun onCreate(...){ ... if(isStartedByScheme()){ // 通過 intent extra 獲取參數(shù)的值 val param1 = getIntent().getStringExtra(paramName) } }}@FragmentScheme(name='activity1', activities = {QDMainActivity.class})class Fragment1: QMUIFragment{ override fun onCreate(...){ ... if(isStartedByScheme()){ // 通過 arguments 獲取參數(shù)的值 val param1 = getArguments().getString(paramName) } }}

這種傳值方法很符合 Android 官方設(shè)計的做法了,這也要求Fragment遵循無參構(gòu)造器的使用方式。

對于 WebView, 我們可以通過重寫WebViewClient#shouldOverrideUrlLoading來處理 scheme 跳轉(zhuǎn):

class MyWebViewClient: WebViewClient{ override fun shouldOverrideUrlLoading(view: WebView, url: String){ if(schemeHandler.handle(url)){ return true; } return super.shouldOverrideUrlLoading(view, url); } override fun shouldOverrideUrlLoading(view: WebView, request: WebResourceRequest){ if(schemeHandler.handle(request.getUrl().toString())){ return true; } return super.shouldOverrideUrlLoading(view, request); }}

實現(xiàn)

QMUISchemeHandler采用代碼生成的方式,在編譯期生成一個SchemeMapImpl類,其實現(xiàn)了SchemeMap類

public interface SchemeMap { // 通過 action 和參數(shù)尋找 SchemeItem SchemeItem findScheme(QMUISchemeHandler handler, String schemeAction, Map<String, String> params); // 判斷 schemeAction 是否存在 boolean exists(QMUISchemeHandler handler, String schemeAction);}

而每個 scheme 的注解對應(yīng)一個SchemeItem:

ActivityScheme對應(yīng)實例化一個ActivitySchemeItem類,并加入到 map 中 FragmentScheme對應(yīng)實例化一個FragmentSchemeItem類,并加入到 map 中

在編譯期通過SchemeProcessor生成的SchemeMapImpl大概是這樣子的:

public class SchemeMapImpl implements SchemeMap { private Map<String, List<SchemeItem>> mSchemeMap; public SchemeMapImpl() { mSchemeMap = new HashMap<>(); List<SchemeItem> elements; ArrayMap<String, String> required = null; elements = new ArrayList<>(); required =null; elements.add(new FragmentSchemeItem(QDSliderFragment.class,false,new Class[]{QDMainActivity.class},null,false,'',required,null,null,null,null,null,SliderSchemeMatcher.class)); mSchemeMap.put('slider', elements); elements = new ArrayList<>(); required = new ArrayMap<>(); required.put('aa', null); required.put('bb', '3'); elements.add(new ActivitySchemeItem(ArchTestActivity.class,true,null,required,null,new String[]{'aa'},null,null,null,null)); mSchemeMap.put('arch', elements); } @Override public SchemeItem findScheme(QMUISchemeHandler arg0, String arg1, Map<String, String> arg2) { List<SchemeItem> list = mSchemeMap.get(arg1); if(list == null || list.isEmpty()) { return null; } for (int i = 0; i < list.size(); i++) { SchemeItem item = list.get(i); if(item.match(arg0, arg2)) { return item; } } return null; } @Override public boolean exists(QMUISchemeHandler arg0, String arg1) { return mSchemeMap.containsKey(arg1); }}

整體的設(shè)計以及實現(xiàn)思路就是這樣,剩下的就是各種編碼細節(jié)了。有興趣的可以通過QMUISchemeHandler#handle()進行追蹤下,或者看看SchemeProcessor是如何做代碼生成的。這個功能看上去簡單,其實也包括了 Builder 模式、責任鏈模式、工廠方法等設(shè)計模式的運用,還有 SchemeMatcher、 SchemeItem 等對面向?qū)ο蟮慕涌?、繼承、多態(tài)等的運用。讀一讀或許對你有所啟迪,或許你也能幫我發(fā)現(xiàn)某些潛在的 Bug。

總結(jié)

到此這篇關(guān)于Android scheme 跳轉(zhuǎn)的設(shè)計與實現(xiàn)的文章就介紹到這了,更多相關(guān)Android scheme 跳轉(zhuǎn)的設(shè)計與實現(xiàn)內(nèi)容請搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!

標簽: Android
相關(guān)文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
久久久久蜜桃| 国产精品99久久久久久董美香| 国产精品红桃| 麻豆视频一区| 91一区二区| 久久久久久久久99精品大| 日韩电影二区| 亚洲精品一区二区妖精| 深夜日韩欧美| 国产精品成人自拍| 国产传媒在线观看| 欧美日韩国产精品一区二区亚洲| 蜜臀av性久久久久蜜臀aⅴ流畅| 日本伊人久久| 久久99蜜桃| 日本久久成人网| 午夜久久久久| 亚洲精品无播放器在线播放| 国产精品一国产精品| 97精品国产| 久久亚洲风情| 久久99久久久精品欧美| 欧美一区久久久| 日韩一区精品视频| 国产欧美自拍一区| 久久精品国内一区二区三区水蜜桃| 国产精品试看| 国产精品一线| 欧美国产91| 欧美一区不卡| 日韩久久一区二区三区| 日本 国产 欧美色综合| 日韩**一区毛片| 日韩深夜视频| 亚洲精品第一| 久久久精品五月天| 青青草国产精品亚洲专区无| 久久中文字幕av| 国产精品久久亚洲不卡| 午夜精品网站| 国产一区二区三区四区二区| 国产偷自视频区视频一区二区| 国产精品视频一区视频二区| 美女少妇全过程你懂的久久| 欧美精品国产一区| 免费观看不卡av| 欧美国产另类| 亚洲主播在线| 日韩成人三级| 91麻豆精品| 狠狠久久婷婷| 四季av一区二区凹凸精品| 亚洲ab电影| 自由日本语亚洲人高潮| 韩国一区二区三区视频| 四虎国产精品免费久久| 激情欧美日韩一区| 精品国产乱码久久久久久1区2匹| 一区二区国产在线| 久久一区二区中文字幕| 国产69精品久久| 国产精品亚洲综合色区韩国| 欧洲毛片在线视频免费观看| 成人在线超碰| 91精品尤物| 亚洲欧美日韩国产一区| 久久久夜精品| 97精品国产| 国产精品久久久久9999高清| 亚洲日产国产精品| 国产视频久久| 亚洲a在线视频| 国产白浆在线免费观看| 国产黄色精品| 欧美一级一区| 亚洲精品系列| 丝袜诱惑制服诱惑色一区在线观看| 免费一级欧美片在线观看网站| 亚洲欧洲日韩| 在线日韩成人| 免费在线视频一区| 蘑菇福利视频一区播放| 久久久五月天| 日韩欧美中文| 亚洲人成在线网站| 国产在视频一区二区三区吞精| 国产精品超碰| 欧美极品一区二区三区| 国产精品宾馆| 欧美激情视频一区二区三区免费| 国产欧美激情| 欧美激情日韩| 国产精品草草| 久久中文字幕一区二区三区| 久久免费福利| 国产精品99久久精品| 福利片在线一区二区| 国产粉嫩在线观看| 久久婷婷丁香| 婷婷综合社区| 亚洲综合五月| 青青伊人久久| 国产精品黄色| 精品免费在线| 欧美亚洲日本精品| 在线日韩中文| 国产一区二区精品| 亚洲欧美日本日韩| 亚洲精品中文字幕99999| 国产一区 二区| 久久精品国产亚洲一区二区三区| 国产不卡人人| 蜜桃一区二区三区| 水野朝阳av一区二区三区| 中文字幕一区二区三区在线视频| 日韩高清一级| 国产精品99久久精品| 欧美日韩中文字幕一区二区三区| 一区在线视频观看| 日韩国产欧美视频| 国产精品115| 久久激情网站| 夜夜嗨网站十八久久 | 一本综合精品| 日本vs亚洲vs韩国一区三区二区| 国产日韩高清一区二区三区在线| 精品一区二区三区在线观看视频| 日本美女一区| 日韩在线观看一区二区| 青青草伊人久久| 超级白嫩亚洲国产第一| 亚洲免费婷婷| 捆绑调教美女网站视频一区| 欧美大黑bbbbbbbbb在线| 免费日本视频一区| 欧美国产专区| 成人av动漫在线观看| 91午夜精品| 久久精品91| 欧美影院视频| 色老板在线视频一区二区| 亚洲精品日韩久久| 国产在线观看www| 一区二区三区国产在线| 久久精品色播| 免费成人在线影院| 97人人精品| 亚洲欧洲美洲国产香蕉| 成人国产精品一区二区网站| 99成人在线| 久久中文字幕一区二区三区| 黄色亚洲大片免费在线观看| 卡一卡二国产精品| 日韩中文字幕区一区有砖一区| 国产+成+人+亚洲欧洲在线| 中文字幕一区二区精品区| 色在线中文字幕| 欧美一区网站| 日韩视频一区| 国产精品yjizz视频网| 日韩一区二区三区免费视频| 久久久精品久久久久久96 | 天堂中文在线播放| 日韩一区二区三区在线看| 99久久亚洲精品蜜臀| 欧美激情三区| 亚洲人成高清| 国产综合亚洲精品一区二| 欧美激情福利| 免费观看在线综合| 色老板在线视频一区二区| 国产精品极品在线观看| 亚洲影视一区| 午夜av一区| 日韩免费在线| 国产精品久久久久77777丨| 婷婷激情图片久久| 日产午夜精品一线二线三线| 欧美日本精品| 日韩精品一二三四| 欧美成人高清| 成人午夜精品| 久久精品国产久精国产| 日本v片在线高清不卡在线观看| 五月天久久777| 日韩高清成人| 91亚洲一区| 精品视频黄色| 国产精品久久久久毛片大屁完整版 | 国产美女精品| 亚洲二区精品| www在线观看黄色| 欧美精品导航| 国产精品亚洲欧美| 日本成人在线视频网站| 蜜桃传媒麻豆第一区在线观看| 五月天久久网站| 亚洲性色视频| 国产精品7m凸凹视频分类| 伊人精品一区|