Litho是Facebook推出的一套高效構建Android UI的聲明式框架,主要目的是提高RecyclerView複雜列表的滑動性能和下降內存佔用。下面是Litho官網的介紹:java
Litho is a declarative framework for building efficient user interfaces (UI) on Android. It allows you to write highly-optimized Android views through a simple functional API based on Java annotations. It was primarily built to implement complex scrollable UIs based on RecyclerView.
With Litho, you build your UI in terms of components instead of interacting directly with traditional Android views. A component is essentially a function that takes immutable inputs, called props, and returns a component hierarchy describing your user interface.Litho是高效構建Android UI的聲明式框架,經過註解API建立高優的Android視圖,很是適用於基於Recyclerview的複雜滾動列表。Litho使用一系列組件構建視圖,代替了Android傳統視圖交互方式。組件本質上是一個函數,它接受名爲Props的不可變輸入,並返回描述用戶界面的組件層次結構。react
Litho是一套徹底不一樣於傳統Android的UI框架,它繼承了Facebook一貫大膽創新的風格,突破性地在Android上實現了React風格的UI框架。架構圖以下:android
應用層:上層Android應用接入層。git
規範層(API):容許用戶使用聲明式的API(註解)來構建符合Flexbox規範的佈局。github
佈局層:Litho使用可掛載組件、佈局組件和Flexbox組件來構建佈局,其中可掛載組件和佈局組件容許用戶使用規範來定義,各個組件的具體用法下面的組件規範中會詳細介紹。在Litho中每個組件都是一個獨立的功能模塊。Litho的組件和React的組件相相似,也具備屬性和狀態的概念,經過狀態的變動來控制組件的展現樣式。緩存
佈局測量:Litho使用Yoga來完成組件佈局的異步或同步(可根據場景定製)測量和計算,實現了佈局的扁平化。性能優化
佈局渲染:Litho不只支持使用View來渲染視圖,還可使用更輕量的Drawable來渲染視圖。Litho實現了大量使用Drawable來渲染的基礎組件,能夠進一步拍平佈局。架構
除了上面提到的扁平化佈局,Litho還實現了佈局的細粒度複用和異步計算佈局的能力,對於這些功能的實如今Litho的特性及原理剖析中詳細介紹。下面先介紹一下你們比較關心的Litho使用方法。框架
Litho的使用方式相比於傳統的Android來講有些另類,它拋棄了經過XML定義佈局的方式,採用聲明式的組件在Java中構建佈局。異步
Android傳統佈局:首先在資源文件res/layout目錄下定義佈局文件xx.xml,而後在Activity或Fragment中引用佈局文件生成視圖,示例以下:
<?xml version="1.0" encoding="utf-8"?> <TextView xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="Hello World" android:textAlignment="center" android:textColor="#666666" android:textSize="40dp" />
public class MainActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.helloworld); } }
Litho佈局:Litho拋棄了Android原生的佈局方式,經過組件方式構建佈局生成視圖,示例以下:
public class MainActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); ComponentContext context = new ComponentContext(this); final Text.Builder builder = Text.create(context); final Component = builder.text("Hello World") .textSizeDip(40) .textColor(Color.parseColor("#666666")) .textAlignment(Layout.Alignment.ALIGN_CENTER) .build(); LithoView view = LithoView.create(context, component); setContentView(view); } }
Litho中的視圖單元叫作Component,能夠直觀的翻譯爲「組件」,它的設計理念來自於React組件化的思想。每一個組件持有描述一個視圖單元所必須的屬性和狀態,用於視圖佈局的計算工做。視圖最終的繪製工做是由組件指定的繪製單元(View或者Drawable)來完成的。
Litho組件的建立方式也和原生View的建立方式有着很大的區別。Litho使用註解定義了一系列的規範,咱們須要使用Litho的註解來定義本身的組件生成規則,最終由Litho在編譯期自動編譯生成真正的組件。
2.2.1 組件規範
Litho提供了兩種類型的組件規範,分別是Layout Spec規範和Mount Spec規範。下面分別介紹兩種規範的使用方式:
Layout Spec規範:用於生成佈局類型組件的規範,佈局組件在邏輯上等同於Android中的ViewGroup,用於組織其餘組件構成一個佈局。它要求咱們必須使用@LayoutSpec註解來註明,並實現一個標註了@OnCreateLayout註解的方法。示例以下:
@LayoutSpec class HelloComponentSpec { @OnCreateLayout static Component onCreateLayout(ComponentContext c, @Prop String name) { return Column.create(c) .child(Text.create(c) .text("Hello, " + name) .textSizeRes(R.dimen.my_text_size) .textColor(Color.BLACK) .paddingDip(ALL, 10) .build()) .child(Image.create(c) .drawableRes(R.drawable.welcome) .scaleType(ImageView.ScaleType.CENTER_CROP) .build()) .build(); } }
最終Litho會在編譯時生成一個名爲HelloComponent的組件。
public final class HelloComponent extends Component { @Prop(resType = ResType.NONE,optional = false) String name; private HelloComponent() { super(); } @Override protected Component onCreateLayout(ComponentContext c) { return (Component) HelloComponentSpec.onCreateLayout((ComponentContext) c, (String) name); } ... public static Builder create(ComponentContext context, int defStyleAttr, int defStyleRes) { Builder builder = sBuilderPool.acquire(); if (builder == null) { builder = new Builder(); } HelloComponent instance = new HelloComponent(); builder.init(context, defStyleAttr, defStyleRes, instance); return builder; } public static class Builder extends Component.Builder<Builder> { private static final String[] REQUIRED_PROPS_NAMES = new String[] {"name"}; private static final int REQUIRED_PROPS_COUNT = 1; HelloComponent mHelloComponent; ... public Builder name(String name) { this.mHelloComponent.name = name; mRequired.set(0); return this; } @Override public HelloComponent build() { checkArgs(REQUIRED_PROPS_COUNT, mRequired, REQUIRED_PROPS_NAMES); HelloComponent helloComponentRef = mHelloComponent; release(); return helloComponentRef; } } }
Mount Spec規範:用來生成可掛載類型組件的規範,用來生成渲染具體View或者Drawable的組件。一樣,它必須使用@MountSpec註解來標註,並至少實現一個標註了@onCreateMountContent的方法。Mount Spec相比於Layout Spec更復雜一些,它擁有本身的生命週期:
除了上述兩種組件類型,Litho中還有一種特殊的組件——Layout,它不能使用規範來生成。Layout是Litho中的容器組件,相似於Android中的ViewGroup,可是隻能使用Flexbox的規範。它能夠包含子組件節點,是Litho各組件鏈接的紐帶。Layout組件只是Yoga在Litho中的代理,組件的全部佈局相關的屬性都會直接設置給Yoga,並由Yoga完成佈局的計算。Litho實現了兩個Layout組件Row和Column,分別對應Flexbox中的行和列。
2.2.2 Litho的屬性
在Litho中屬性分爲兩種,不可變屬性稱爲Props,可變屬性稱爲State,下面分別介紹一下兩種屬性:
Props屬性:組件中使用@Prop註解標註的參數集合,具備單向性和不可變性。下面經過一個簡單的例子瞭解一下如何在組件中定義和使用Props屬性:
@MountSpec class MyComponentSpec { @OnPrepare static void onPrepare( ComponentContext c, @Prop(optional = true) String prop1) { ... } @OnMount static void onMount( ComponentContext c, SomeDrawable convertDrawable, @Prop(optional = true) String prop1, @Prop int prop2) { if (prop1 != null) { ... } } }
在上面的代碼中,共使用了三次Prop註解,分別標註prop1和prop2兩個變量,即定義了prop1和prop2兩個屬性。Litho會在自動編譯生成的MyComponent類的Builder類中生成這兩個屬性的同名方法。按照以下代碼,即可以去使用上面定義的屬性:
MyComponent.create(c) .prop1("My prop 1") .prop2(256) .build();
State屬性:意爲「狀態」屬性,State屬性雖然可變,可是其變化由組件內部控制,例如:輸入框、Checkbox等都是由組件內部去感知用戶行爲,並更新組件的State屬性。因此一個組件一旦建立,咱們便沒法經過任何外部設置去更改它的屬性。組件的State屬性雖然不容許像Props屬性那樣去顯式設置,可是咱們能夠定義一個單獨的Props屬性來當作某個State屬性的初始值。
Litho官網首頁經過4個段落重點介紹了Litho的4個特性。
Litho採用聲明式的API來定義UI組件,組件經過一組不可變的屬性來描述UI。這種組件化的思想靈感來源於React,關於聲明式組件的用法上面已經詳細介紹過了。
傳統Android佈局由於UI與邏輯分離,因此開發工具都有強大的預覽功能,方便開發者調整佈局。而Litho採用React組件化的思想,經過組件鏈接了邏輯與佈局UI,雖然Litho也提供了對Stetho的支持,藉助於Chrome開發者工具對界面進行調試,不過使用起來並無那麼方便。
Android系統在繪製時爲了防止頁面錯亂,頁面全部View的測量(Measure)、佈局(Layout)以及繪製(Draw)都是在UI線程中完成的。當頁面UI很是複雜、視圖層級較深時,不免Measure和Layout的時間會過長,從而致使頁面渲染時候丟幀出現卡頓狀況。Litho爲解決該問題,提出了異步佈局的思想,利用CPU的閒置時間提早在異步線程中完成Measure和Layout的過程,僅在UI線程中完成繪製工做。固然,Litho只是提供了異步佈局的能力,它主要使用在RecyclerView等能夠提早知道下一個視圖長什麼樣子的場景。
3.2.1 異步佈局原理剖析
針對RecyclerView等滑動列表,因爲能夠提早知道接下來要展現的一個甚至多個條目的視圖樣式,因此只要提早建立好下一個或多個條目的視圖,就能夠提早完成視圖的佈局工做。
那麼Android原生爲何不支持異步佈局呢?主要有如下兩個緣由:
使用Litho佈局,咱們能夠獲得一個極致扁平的視圖效果。它能夠減小渲染時的遞歸調用,加快渲染速度。
下面是同一個視圖在Android和Litho實現下的視圖層級效果對比。能夠看到,一樣的樣式,使用Litho實現的佈局要比使用Android原生實現的佈局更加扁平。
3.3.1 扁平化視圖原理剖析
Litho使用Flexbox來建立佈局,最終生成帶有層級結構的組件樹。而後Litho對佈局層級進行了兩次優化。
原理以下圖所示,Litho會先把組件樹拍平成沒有層級的列表,而後使用Drawable來繪製對應的視圖單元。
Litho使用Drawable代替View能帶來多少好處呢?Drawable和View的區別在於前者不能和用戶交互,只能展現,所以Drawable不會像View那樣持有不少變量和引用,因此Drawable比View從內存上看要輕量不少。舉個例子:50個一樣展現「Hello world」的TextView和TextDrawable在內存佔比上,前者幾乎是後者的8倍。對比圖以下,Shallow Size表示對象自身佔用的內存大小。
3.3.2 繪製單元的降級策略
因爲Drawable不具備交互能力,因此對於使用Drawable沒法實現的交互場景,Litho會自動降級成View。主要有如下幾種場景:
對於以上場景的使用請仔細考慮,過多的使用會致使Litho的層級優化效果變差。
3.3.3 對比Android的約束佈局
爲了解決佈局嵌套問題,Android推出了約束佈局(ConstraintLayout),使用約束佈局也能夠達到扁平化視圖的目的,那麼使用Litho的好處是什麼呢?
Litho能夠更好地實現複雜佈局。約束佈局雖然能夠實現扁平效果,可是它使用了大量的約束來固定視圖的位置。隨着佈局複雜程度的增長,約束條件變得愈來愈多,可讀性也變得愈來愈差。而Litho則是對Flexbox佈局進行的扁平化處理,因此實際使用的仍是Flexbox佈局,對於複雜的佈局Flexbox佈局可讀性更高。
Litho中的全部組件均可以被回收,並在任何位置進行復用。這種細粒度的複用方式能夠極大地提升內存使用率,尤爲適用於複雜滑動列表,內存優化很是明顯。
3.4.1 原生RecyclerView複用原理剖析
原生的RecyclerView視圖按模板類型進行存儲並複用,也就是說模板類型越多,所需存儲的模板種類也就越多,致使內存佔用愈來愈大。原理以下圖。滑出屏幕的itemType1和itemType2都會在Recycler緩存池保存,等待後面滑進屏幕的條目的複用。
3.4.2 細粒度複用優化內存原理剖析
在Litho中,item在回收前,會把LithoView中掛載的各個繪製單元拆分出來(解綁),由Litho本身的緩存池去分類回收,在展現前由LithoView按照組件樹的樣式組裝(掛載)各個繪製單元,這樣就達到了細粒度複用的目的。原理以下圖。滑出屏幕的itemType1會被拆分紅一個個的視圖單元。LithoView容器由Recycler緩存池回收,其餘視圖單元由Litho的緩存池分類回收。
使用細粒度複用的RecyclerView的緩存池再也不須要區分模板類型來緩存大量的視圖模板,只須要緩存LithoView容器。細粒度回收的視圖單元數量要遠遠小於原來緩存在各個視圖模板中的視圖單元數量。
美團對Litho進行了二次開發,在美團的MTFlexbox動態化實現方案(簡稱動態佈局)中把Litho做爲底層UI渲染引擎來使用。經過動態佈局的預覽工具,爲Litho提供實時預覽能力,同時能夠有效發揮Litho的性能優化效果。
目前Litho+動態佈局的實現方案已經應用在了美團App中,給美團App帶來了不錯的性能提高。後續博主會詳細介紹Litho+動態佈局在美團性能優化的實踐方案。
因爲Litho中使用了大量Drawable替換View,而且實現了視圖單元的細粒度複用,所以複雜列表滑動時內存優化比較明顯。美團首頁內存佔用隨滑動頁數變化走勢圖以下。隨着一頁一頁地滑動,內存優化了30M以上。(數據採集自Vivo x20手機內存佔用狀況)
FPS的提高主要得益於Litho的異步佈局能力,提早計算佈局能夠減小滑動時的幀率波動,因此滑動過程較平穩,不會有高低起伏的卡頓感。(數據採集自魅藍2手機一段時間內連續fps的波動狀況)
Litho相對於傳統Android是顛覆式的,它採用了React的思路,使用聲明式的API來編寫UI。相比於傳統Android,確實在性能優化上有很大的進步,可是若是徹底使用Litho開發一款應用,須要本身實現不少組件,而Litho的組件須要在編譯時生成,實時預覽方面也有所欠缺。相對於直接使用Litho的高成本,把Litho封裝成Flexbox佈局的底層渲染引擎是個不錯的選擇。