日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問(wèn) 生活随笔!

生活随笔

當(dāng)前位置: 首頁(yè) > 编程资源 > 编程问答 >内容正文

编程问答

Weex-初次见到你

發(fā)布時(shí)間:2025/3/21 编程问答 35 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Weex-初次见到你 小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

Weex 剛剛開(kāi)源,非常幸運(yùn)能在 Weex 團(tuán)隊(duì)實(shí)習(xí),也見(jiàn)證了 Weex 正式開(kāi)源的夜晚,作為一名 Androider,當(dāng)然是萬(wàn)分激動(dòng)。近三個(gè)月的實(shí)習(xí)過(guò)程,Weex Android 的源碼也搗騰了不少,寫篇文章紀(jì)念一下~ 寫的不對(duì)的地方,還望各大神指點(diǎn)迷津!

PS:這里只對(duì) Weex Android 源碼進(jìn)行分析。


一、Weex第一眼

  • A framework for building Mobile cross-platform UI (一款輕量級(jí)的移動(dòng)端跨平臺(tái)動(dòng)態(tài)性技術(shù)解決方案)
  • 通過(guò) Html 搭建組件結(jié)構(gòu),flexbox 負(fù)責(zé)界面布局, Js 控制數(shù)據(jù)和邏輯
  • 相比 RN,Weex 真正做到了 write once run anywhere
  • RN 可以認(rèn)為是一個(gè)全新的跨平臺(tái)移動(dòng)開(kāi)發(fā)框架,比較重,適合一個(gè)完整應(yīng)用的開(kāi)發(fā);而 Weex 是為了增強(qiáng)移動(dòng)端動(dòng)態(tài)性而生的輕量級(jí)框架,具有極強(qiáng)的可擴(kuò)展性,能夠比較容易的融入成熟的Native項(xiàng)目中。
  • 跟 QZone 的前輩聊到過(guò) RN,QZone 接入 RN 后,總體性能還不如原來(lái)騰訊的 Webview 方案,而 Weex 在加載性能上是要占優(yōu)勢(shì)的。

二、Weex的重要器官

1、WXDomObject

  • DomObject 包括了 <template> 在 Dom 樹(shù)中的所有信息,如 style、attr、event、ref(結(jié)點(diǎn)的唯一標(biāo)識(shí)符)、parent、children

2、WXComponent

  • Component 負(fù)責(zé)承載 Native View,可以通過(guò)泛型指定承載 View的類型:

  • abstract class WXComponent{};
  • class WXListComponent extends WXVContainer{};

    • Component 會(huì)保留 DomObject 的強(qiáng)引用,兩者實(shí)例是一一對(duì)應(yīng)的。
    • 通過(guò)調(diào)用 initComponentHostView 創(chuàng)建 Component 需要承載的 View,所有 的Component 必須重寫 initComponentHostView 方法,返回需要承載的 View 的最外層容器。如下:

  • class WXBaseRefresh extends WXVContainer {

    @Override
    protected WXFrameLayout initComponentHostView(Context context) {
    return new WXFrameLayout(context);
    }
    }

  • 3、WXModule:

    • 通過(guò) Module 可以將 Native Api 暴露給Js。

    4、WXSDKInstance

    • Weex 的渲染單位。
    • 明確兩個(gè)容器的概念:

      • Instance RootView: Weex 最外層容器, Native 接入方可以設(shè)置 Instance RootView 大小, 最終通過(guò) onViewCreated 返回給用戶的View也就是 Instance RootView。
      • JS Root: JS 可以描述的最外層容器, 為 RootView 的唯一子節(jié)點(diǎn), 受 JS 樣式的控制。
    • Instance 寬高設(shè)置遵循以下幾個(gè)原則

      • Instance RootView 的寬高優(yōu)先遵循 instance 設(shè)置的寬高,如果開(kāi)發(fā)者沒(méi)有設(shè)置,則與 JS Root 節(jié)點(diǎn)的寬高保持一致。
      • JS Root 節(jié)點(diǎn)的寬高優(yōu)先遵循 CSS 樣式設(shè)置的寬高,如果沒(méi)有設(shè)置,則使用 instance 上設(shè)置的寬高,如果 instance 也沒(méi)有設(shè)置,則使用 layout 出來(lái)的寬高
      • 特殊情況,當(dāng) scroller 和 list 作為 JS Root 時(shí),如果不設(shè)置高度, 會(huì)給 scroller 和 list 設(shè)置 flex:1
      • 綜上所述,Instance RootView 和 JS Root 的寬高可以不一致,應(yīng)該根據(jù)需求正確的設(shè)置 Instance 的寬高,也可以在運(yùn)行時(shí)動(dòng)態(tài)的改變 Instance 的寬高。

    三、Weex工作原理

    1、渲染原理

    • .we文件由 <template> 、 <style> 、 <script> 三部分組成;首先,transformer 會(huì)將 .we 文件轉(zhuǎn)換成 Js Bundle,JSFramework 根據(jù) Js Bundle 生成 Virtual Dom,根據(jù)Virtual Dom 控制 Native 的視圖層;首次渲染時(shí),會(huì)將所有結(jié)點(diǎn)都交給 Native Render 渲染,在 UI 更新時(shí),計(jì)算出最小 dif,讓 Native 僅渲染發(fā)生改變的結(jié)點(diǎn)。

    2、派發(fā)渲染指令的樞紐:WXDomModule

    • 上面提到,JSFramework 根據(jù) Virtual Dom 計(jì)算出來(lái)的 dif,將渲染指令(Json)通過(guò) Js Engine 發(fā)送給 Native Render 進(jìn)行渲染。而 WXDomModule 會(huì)接收到所有渲染指令,然后將指令post 給 DomHandler,最后由 DomHandler 來(lái)派發(fā)渲染任務(wù)。
    • DomStatement 在 Dom 線程中創(chuàng)建 DomObject 和 Component,RenderStatement 負(fù)責(zé)在 UI 線程中渲染 View;每個(gè) WXSDKInstance 會(huì)持有一個(gè) DomStatement 和 RenderStatement 實(shí)例。
    • RenderStatement 會(huì)從 DomStatementclone 一份 DomObject,是為了避免兩個(gè)線程同時(shí)操作 Dom 造成的同步問(wèn)題。
    • 主要有如下指令:
    • createBody:DomStatement 首先在 Dom 線程中創(chuàng)建 JS Root 對(duì)應(yīng)的 Component,然后會(huì)將 JS Root 添加到 WXSDKInstance 作為 GodCom 的子節(jié)點(diǎn),從而生成 Component 樹(shù)的最頂端。生成 Component 樹(shù)后,將 createBody 任務(wù) post 到 UI 線程,由 RenderStatement 創(chuàng)建 WXSDKInstance 的 Rootview,并通過(guò) onViewCreated 回調(diào)給 WXSDKInstance 的上下文。
    • addElement:首先,DomStatement 在 Dom 線程中創(chuàng)建 DomObject 和對(duì)應(yīng)的 Component 實(shí)例,加入 Dom 樹(shù)和 Component 樹(shù);然后將 addElement 任務(wù) post 到 UI 線程,RenderStatement 會(huì)觸發(fā) Component 完成以下任務(wù): createView(初始化 Component 承載的 View)、applyLayoutAndEvent(觸發(fā) setLayout 和 setPadding、綁定 Event)、bindData(給 View 設(shè)置 style、attr)、addChild(將 View 加入 View 樹(shù))
    • removeElement:是 addElement 的逆向操作,將 View、Component、DomObject 分別從各自的樹(shù)中刪除,并銷毀數(shù)據(jù)回收資源。
    • moveElement:將 View、Component、DomObject 在樹(shù)中移動(dòng)位置,move 操作最終被拆分成一次 remove 操作和一次 add 操作。
    • addEvent:綁定事件。
    • removeEvent:撤銷事件綁定。
    • updateAttrs:當(dāng)結(jié)點(diǎn) attr 被改變時(shí),會(huì)觸發(fā) updateAttrs,最終會(huì)觸發(fā) WXComponent 中的 updateProperties 刷新 UI。
    • updateStyle:與 updateAttrs 類似。
    • createFinish:JsFramework 將所有渲染指令都發(fā)出后,會(huì)觸發(fā) createFinish,最后會(huì)觸發(fā) onRenderSuccess 回調(diào)。
    • updateFinish:JsFramework 將所有 update 指令發(fā)出后,會(huì)觸發(fā) updateFinish,最后會(huì)觸發(fā) onUpdateFinish 回調(diào)。

    3、Js與Native的通信方式

    (1)Js調(diào)用Native

    • Js 調(diào)用 Native 必須以 Module 的方式實(shí)現(xiàn),@WXModuleAnno 注解會(huì)將 Module 方法暴露給Js。
    • Js 調(diào)用 Module 方法:

      this.$call('modal', 'toast', {'message': 'naviBar.rightItem.click','duration': duration});
    • modal 是 Module 名字,toast 是 Module 的方法名。
    • Js 調(diào)用 Native 方法,均通過(guò)如下方式完成:

      WXModuleManager.callModuleMethod(instanceId, (String) task.get(WXDomModule.MODULE),(String) task.get(WXDomModule.METHOD), (JSONArray) task.get(WXDomModule.ARGS));
    static boolean callModuleMethod(String instanceId, String moduleStr, String methodStr, JSONArray args) {//通過(guò) ModuleFactory 拿到 module 的實(shí)例ModuleFactory factory = sModuleFactoryMap.get(moduleStr);final WXModule wxModule = findModule(instanceId, moduleStr, factory);wxModule.mWXSDKInstance = WXSDKManager.getInstance().getSDKInstance(instanceId);/**** 通過(guò)反射拿到方法和參數(shù),構(gòu)造 Invoker 對(duì)象*/Map<String, Invoker> methodsMap = factory.getMethodMap();final Invoker invoker = methodsMap.get(methodStr);invoker.invoke(wxModule, params);}

    (2)Native 調(diào)用 Js 之 fireEvent

    • fireEvent 一般在 Component 中使用,多用于事件監(jiān)聽(tīng)

      WXSDKManager.getInstance().fireEvent(mInstanceId, getRef(), WXEventType.ONCLICK);
    • 第一個(gè)參數(shù)表示其所在 WXSDKInstance 的 id,第二個(gè)參數(shù)是 Component 的 ref(唯一標(biāo)識(shí)),第三個(gè)參數(shù)是事件名稱。
    <text onclick="onclick">weex</text> <script>module.exports = {methods: {onclick: function(param) {param.x;param.y;},}}</script>

    (3)Native調(diào)用Js之JSCallback

    • JSCallback 一般在 Module 中使用,可以參考 WXStreamModule 的實(shí)現(xiàn)

      @WXModuleAnno public void fetch(String optionsStr, final JSCallback callback) {Options.Builder builder = new Options.Builder().setMethod("GET").setUrl(url);extractHeaders(headers, builder);final Options options = builder.createOptions();sendRequest(options, new ResponseCallback() {@Overridepublic void onResponse(WXResponse response, Map<String, String> headers) {if (callback != null) {Map<String, Object> resp = new HashMap<>();resp.put(STATUS, response.statusCode);resp.put("ok", (code >= 200 && code <= 299));......resp.put(STATUS_TEXT, Status.getStatusText(response.statusCode));resp.put("headers", headers);callback.invoke(resp);}}}); }
    • 在 callModuleMethod 中已經(jīng)將 JSCallback 與 instanceIs、callbackId 封裝成 SimpleJSCallback,只需調(diào)用 invoke,傳入?yún)?shù)即可。
    • 那么 JS 這邊又該如何接收回調(diào)呢?

      stream.fetch({ method: 'GET', url: GET_URL, }, function(ret) { if(!ret.ok){me.getResult = "request failed"; }else{console.log('get:'+ret);me.getResult = ret.data; } });

    四、存在的問(wèn)題

    • List 是業(yè)務(wù)中較為常用的容器,而 Weex Android 的 List 是通過(guò) 原生的 RcyclerView 實(shí)現(xiàn)的,由于 RecyclerView 的復(fù)用機(jī)制,給 List 組件帶來(lái)了不少坑,這里主要說(shuō)下我在開(kāi)發(fā)中碰到的幾個(gè)例子。

    1、RecyclerView原理

    • 首先簡(jiǎn)單介紹一下 RecyclerView 的復(fù)用原理~
    • Recycler 是 RecyclerView 中管理組件復(fù)用的核心內(nèi)部類,而 Recycler 中有幾個(gè)重要的成員變量:

      • mCachedViews:剛剛從屏幕中滑出的 ViewHolder,且 bind 的數(shù)據(jù)未修改,會(huì)緩存在 mCachedViews 中 , mCachedViews 中的 ViewHolder 隨時(shí)可以重新顯示在屏幕上而不需要重新 bindData;針對(duì)每一種 ViewType 的 ViewHolder,緩存的數(shù)量默認(rèn)為2
      • mAttachedScrap、mChangedScrap:mCachedViews 中被修改過(guò)的臟塊,會(huì)轉(zhuǎn)移到 scrap 中(Attached 和 Changed 的區(qū)別僅僅在于對(duì) itemAnimation 的支持,這里不展開(kāi)說(shuō)了,統(tǒng)稱為 scrap);scrap 中的 ViewHolder,如要重新顯示在屏幕上,需要重新 bindData。
      • RecycledViewPool:RecycledViewPool 是可以支持多個(gè) RecyclerView 共享的 ViewHolder 緩存池,當(dāng) mCachedViews 或 scrap 滿了之后,會(huì)將末尾的元素移動(dòng)到 RecycledViewPool 中,這里可以理解為分級(jí)緩存。RecyclerViewPool 里有兩個(gè)成員變量,SparseArray> mScrap 和 SparseIntArray mMaxScrap,mScrap 根據(jù) ViewType 對(duì) ViewHolder 進(jìn)行二級(jí)索引存儲(chǔ);mMaxScrap 表示每一類 ViewType 的允許緩存 ViewHolder 的最大數(shù)量;但是 RecycledViewPool 并沒(méi)有對(duì)不同 ViewType 的數(shù)量進(jìn)行限制,所以這里會(huì)涉及到一些坑,下面展開(kāi)說(shuō)明。

    2、Cell復(fù)用存在的問(wèn)題

    • 在 Android 中,RecyclerView 提供了復(fù)用機(jī)制來(lái)減少內(nèi)存開(kāi)銷、提升滑動(dòng)效率,Weex 中 List 也暴露出相應(yīng)的 API 支持 Cell 復(fù)用:設(shè)置相同 scopeValue 的 Cell 支持 ViewHolder 復(fù)用。但是,List 在對(duì) scopeValue 的支持上,還存在一些問(wèn)題:

    (1)Cell 使用 if 控制子元素發(fā)生 crash

    • Cell 復(fù)用的前提條件是 view 層級(jí)和布局完全一致,如果使用 if 控制 Cell 子元素的可見(jiàn)性,可能導(dǎo)致復(fù)用時(shí)舊 Cell 和新 Cell 結(jié)構(gòu)不一致,在重新 bindData 時(shí)產(chǎn)生 crash。

    (2)Cell 復(fù)用后產(chǎn)生文字截?cái)?/strong>

    • 為了提升滑動(dòng)效率,Cell 被復(fù)用時(shí)不會(huì)觸發(fā) setLayout,如果在 Cell 子元素中含有不定寬度的 text 組件,復(fù)用后不會(huì)重新計(jì)算 text 寬度,所以導(dǎo)致文字截?cái)唷?/li>
    • 建議: 目前 List 對(duì) scopeValue 的支持還不夠完善,使用時(shí)要多加注意,前端應(yīng)該遵循“Cell 可復(fù)用的前提是 view 層級(jí)和布局完全一致”的規(guī)則,對(duì)于內(nèi)部結(jié)構(gòu)不確定、樣式可能發(fā)生變化的 Cell 不要進(jìn)行復(fù)用;如果使用得當(dāng),scopeValue 還是很強(qiáng)大的~

    3、Cell 內(nèi)存泄露

    • 由于 Weex 重寫了 Recycler.ViewHolder,使得 ViewHolder 持有 Cell 的強(qiáng)引用,由于 RecyclerView 會(huì)將 ViewHolder 緩存在 RecycledViewPool 中,導(dǎo)致 Cell 被 remove 后,無(wú)法被 JVM 回收,導(dǎo)致 Cell 樹(shù)內(nèi)存泄漏;這個(gè)問(wèn)題在 0.7.0 中已經(jīng)修復(fù),ViewHolder 持有 Cell 的軟引用,List 持有 Cell 的強(qiáng)引用,可以保證 Cell 在正確的時(shí)機(jī)被回收。

    4、無(wú)用的 ViewHolder 緩存

    • 如果不指定 Cell 的 scopeValue,會(huì)使得每一個(gè) Cell 都有不同的 ViewType,之前提到 RecycledViewPool 不限制不同類型的 ViewHolder,所以這里會(huì)導(dǎo)致 ViewHolder 不限數(shù)量的緩存堆積在 RecycledViewPool 中,而且根本不會(huì)參與復(fù)用,所以理解為“無(wú)用的緩存”。在 v0.7.0 中解決了這個(gè)問(wèn)題,限制了不同 ViewType 在 RecycledViewPool 中緩存的數(shù)量;由于 Cell 和 Cell 中承載的 View 都會(huì)保存在內(nèi)存中,可以省去 Component 創(chuàng)建,所以 ViewHolder 的創(chuàng)建耗時(shí)不多。
    • v0.6.1:Cell 被 RecyclerView 緩存池引用
    • v0.6.1:內(nèi)存情況
    • v0.7.0:內(nèi)存情況(可以明顯看到內(nèi)存被釋放的過(guò)程)

    5、Weex 中 ViewHolder 創(chuàng)建過(guò)程存在的問(wèn)題

    • 當(dāng) RecyclerView 在緩存中找不到合適的 ViewHolder 復(fù)用時(shí),會(huì)調(diào)用 onCreateViewHolder 創(chuàng)建新的 ViewHolder;Weex 繼承 RecyclerView.ViewHolder 實(shí)現(xiàn)了自己的ListBaseViewHolder,將 Cell 作為 ViewHolder 的成員變量,所以在 onCreateViewHolder 創(chuàng)建 ViewHolder 的時(shí)候,需要找到一個(gè)“ViewType一致,且沒(méi)有被 ViewHolder 綁定過(guò)”的 Cell 與其關(guān)聯(lián)。
    • 在 v0.6.1 中,查找方式是遍歷所有的 Cell,直到找到合適的那個(gè),這樣存在一個(gè)問(wèn)題,當(dāng) List 元素特別多的時(shí)候,查找過(guò)程的性能消耗會(huì)有一些劣勢(shì):
    for (int i = 0; i < childCount(); i++) {WXComponent component = getChild(i);if (component == null || component.isUsing() || getItemViewType(i) != viewType)continue;....... }
    • 在 v0.7.0 中,做了相應(yīng)優(yōu)化,與 RecyclerView 的緩存池使用同一種查找方式,也就是根據(jù) ViewType 對(duì) Cell 做分類的二級(jí)索引,這樣就避免了多余的循環(huán)。
    ArrayList<WXComponent> mTypes = mViewTypes.get(viewType); ....... for (int i = 0; i < mTypes.size(); i++) {WXComponent component = mTypes.get(i);if (component == null || component.isUsing()) {continue;}....... }private int generateViewType(WXComponent component) {long id;try {id = Integer.parseInt(component.getDomObject().ref);String type = component.getDomObject().attr.getScope();if (!TextUtils.isEmpty(type)) {if (mRefToViewType == null) {mRefToViewType = new ArrayMap<>();}if (!mRefToViewType.containsKey(type)) {mRefToViewType.put(type, id);}id = mRefToViewType.get(type);}} catch (RuntimeException e) {id = RecyclerView.NO_ID;}return (int) id; }

    五、快速接入Weex

    • sdk的接入就不贅述了,主要說(shuō)下Native這邊需要的代碼配置。

    1、首先創(chuàng)建 WXSDKInstance 實(shí)例

    mInstance = new WXSDKInstance(this);mInstance.setImgLoaderAdapter(new ImageAdapter(this));mInstance.registerRenderListener(this);

    2、實(shí)現(xiàn)渲染的監(jiān)聽(tīng)接口

    public class WXBaseActivity implements IWXRenderListener {}mInstance.registerRenderListener(this);

    3、指定要渲染的Page

    mInstance.render( TAG, // path表示需要渲染的js文件的路徑 WXFileUtils.loadFileContent(path, WXPageActivity.this), null, null, ScreenUtil.getDisplayWidth(WXPageActivity.this), ScreenUtil.getDisplayHeight(WXPageActivity.this), // 傳入WXRenderStrategy.APPEND_ASYNC表示異步渲染 WXRenderStrategy.APPEND_ASYNC);
    • 這里需要說(shuō)明一下,第5、6個(gè)參數(shù)是用來(lái)指定 Instance 的寬高

    4、實(shí)現(xiàn)回調(diào)

    @Override public void onViewCreated(WXSDKInstance instance, View view) {// 這里返回的view就是weex將我們指定path的js文件渲染出來(lái)的view }

    總結(jié)

    以上是生活随笔為你收集整理的Weex-初次见到你的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。

    如果覺(jué)得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。