插件化知識梳理(2) - Small 框架之如何引入公共庫插件


相關(guān)閱讀

插件化知識梳理(1) - Small 框架之如何引入應(yīng)用插件
插件化知識梳理(2) - Small 框架之如何引入公共庫插件
插件化知識梳理(3) - Small 框架之宿主分身
插件化知識梳理(4) - Small 框架之如何實現(xiàn)插件更新
插件化知識梳理(5) - Small 框架之如何不將插件打包到宿主中
插件化知識梳理(6) - Small 源碼分析之 Hook 原理
插件化知識梳理(7) - 類的動態(tài)加載入門
插件化知識梳理(8) - 類的動態(tài)加載源碼分析
插件化知識梳理(9) - 資源的動態(tài)加載示例及源碼分析
插件化知識梳理(10) - Service 插件化實現(xiàn)及原理


一、前言

插件化知識梳理(1) - Small 框架之如何引入應(yīng)用插件 中,我們簡要地介紹了如何使用Small框架來通過插件實現(xiàn)一個Activity的跳轉(zhuǎn),也就是app.main,這里的app.main我們稱為應(yīng)用插件,除了應(yīng)用插件外,還有一種稱為公共庫插件

對于這兩種插件的職責(zé),Small官方建議的基本原則為

  • 公共庫插件
  • 把各個 第三方庫拆出來做成lib插件模塊,包括統(tǒng)計、地圖、網(wǎng)絡(luò)、圖片等庫。
  • 把老項目積累的業(yè)務(wù)公共代碼utils分離出來封裝成一個lib.utils插件
  • 把基礎(chǔ)的樣式、主題分離出來封裝成一個lib.style插件
  • 應(yīng)用插件
  • 把業(yè)務(wù)模塊拆成app模塊,他們可以依賴lib模塊,顯示調(diào)用lib中的各個API
  • 相對獨立的業(yè)務(wù)模塊先拆,先放一個插件里
  • 如果都不好拆,先把全部業(yè)務(wù)做成一個app.main主插件

總結(jié)下來就是:

  • 應(yīng)用插件模塊相互之間不直接引用,但是可以引用公共庫插件模塊
  • 公共庫插件模塊不應(yīng)當(dāng)相互引用,也不應(yīng)當(dāng)引用應(yīng)用插件模塊
  • 一個公共庫插件模塊可以被多個應(yīng)用插件模塊引用

在項目當(dāng)中,許多個模塊有可能用到一些共有的東西,例如圖片加載、網(wǎng)絡(luò)請求、公共控件等,那么我們就有必要將這些東西封裝成共同庫讓各個模塊去調(diào)用,一般來說,可以分為以下兩種:

  • 工具類,例如網(wǎng)絡(luò)請求、圖片加載、持久性存儲等。
  • 資源類,例如公共控件、圖片資源、主題風(fēng)格。

對于上面這兩類,我們一般稱為公共庫插件,它的命名方式為lib.xxx。公共庫插件模塊分為兩個階段:

  • 在開發(fā)階段,我們可以通過compile project(':lib_module_name')讓應(yīng)用插件模塊來引用它,以調(diào)用它模塊中所定義的方法或使用主題樣式。
  • 在編譯階段,共同庫插件會被打包成為一個可獨立更新的插件。

下面,我們就分這兩個方面,這介紹一下使用建立公共庫插件。

二、公共庫

2.1 新建 lib.utils 模塊作為工具類共同庫

(a) 新建插件庫 Android Library


(b) 注意包名的定義,要分為兩個部分

新建完畢之后,我們的項目結(jié)構(gòu)變?yōu)橄旅孢@樣:

(c) 編寫工具類代碼

lib.utils模塊中,引入第三方庫Glide,用于圖片的加載:

dependencies {
    //引入第三方庫Glide。
    compile 'com.github.bumptech.glide:glide:3.7.0'
}

新建一個簡單的圖片加載類ImageLoader

public class ImageLoader {

    public static void loadImage(Context context, String imgUrl, ImageView img) {
        Glide.with(context).load(imgUrl).into(img);
    }
}

2.2 新建 lib.style 作為資源類共同庫

(a) 新建插件庫 Android Library


(b) 在 res 目錄下新建 styles.xml 文件

styles.xml文件中,我們定義一些公共的樣式:

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <style name="StyleTextViewTitle">
        <item name="android:textSize">45px</item>
        <item name="android:textColor">#333333</item>
        <item name="android:padding">12dp</item>
    </style>

    <style name="StyleTextViewSubTitle">
        <item name="android:textSize">27px</item>
        <item name="android:textColor">#888888</item>
    </style>

</resources>

2.3 修改插件模塊 app.main

** (a) 引入 lib.utils 和 lib.style **

dependencies {
    //...
    compile project(':lib.utils')
    compile project(':lib.style')
}

(b) 在布局文件中,使用 lib.style 的公共樣式

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:orientation="vertical"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="@android:color/white"
    tools:context="com.demo.small.app.main.PlugActivity">
    <ImageView
        android:id="@+id/iv_header"
        android:layout_gravity="center_horizontal"
        android:layout_width="200dp"
        android:layout_height="200dp"/>
    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Small Android"
        android:layout_gravity="center_horizontal"
        style="@style/StyleTextViewTitle"/>
    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Small 插件化方案適用于將一個APK拆分為多個公共庫插件、業(yè)務(wù)模塊插件的場景。"
        style="@style/StyleTextViewSubTitle"/>
</LinearLayout>

** (c) 在代碼中,使用 lib.utils 定義的接口**

public class PlugActivity extends AppCompatActivity {
    
    private static final String IMG_URL = "http://i6.hexun.com/2017-06-02/189461191.jpg";

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_plug);
        ImageView imageView = (ImageView) findViewById(R.id.iv_header);
        //調(diào)用 lib.utils 中定義的接口。
        ImageLoader.loadImage(this, IMG_URL, imageView);
    }
}

2.4 修改宿主模塊的 bundle.json 文件:

{
  "version": "1.0.0",
  "bundles": [
    {
      "uri": "lib.utils",
      "pkg": "com.demo.small.lib.utils"
    },
    {
      "uri": "lib.style",
      "pkg": "com.demo.small.lib.style"
    },
    {
      "uri": "main",
      "pkg": "com.demo.small.app.main"
    }
  ]
}

2.6 重新編譯

2.7 最終效果


更多文章,歡迎訪問我的 Android 知識梳理系列:

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

推薦閱讀更多精彩內(nèi)容