Android Studio(3.x) 利用gradle生成jar,并打到system/framework生成動態庫

先簡單的介紹一下需求

做一個數據收集的模塊,分兩種方式集成:
(1)打到artifactory上
(2)生成jar,放到system/vendor下,然后打到system/framework,變成動態庫

第一種方式,之前已經做過,不在本文記錄下,有興趣的同學自行搜索,記錄下第二種方式
首先提供兩篇文章,介紹下動態庫和靜態庫以及dex的加載問題:

https://blog.csdn.net/striver_jt/article/details/89226380
http://www.lxweimin.com/p/a350876a66e1
如果你只關心打成jar包,那么不需要看上面的文件了;
如果你還想關心下打成的jar包怎么使用,那么可以看下,之所以看這兩個文章,是因為我在操作的過程中,發現生成的jar,直接導出項目可以用,但是放到源碼,整編后就不能用了,然后就一知半解的看了下。

閑話少敘,開始講解:

總共分為三步:

Step1:生成jar
Step2:轉換成dex jar,并放置到源碼中
Step3:在項目中使用

Step1:生成jar

如果你只是要將Module打成庫,給別的Android Studio項目使用,建議你打成AAR,簡潔方便,還能包含資源。
在Android Studio中生成jar的方式,基本都是copy,因為module在編譯的過程中會自己生成jar,在網上搜到的大多數的方式,其實都是在做copy的事情。
1.新建一個lib module,我的module的名字叫datacollectionlib
2.修改gradle文件,確認gradle的第一行是
apply plugin:'com.android.library'
而不是
applyplugin:'com.android.application'
前者代表的是library,后者代表的是application
3.編寫命令
先給出一個Android Studio 2.x版本上命令提供參考:
在module的gradle文件中的android 模塊里,編寫如下代碼

        task makeJar(type: Copy) {
                //刪除存在的
                delete 'build/libs/com.rambo.datacollection.jar'
                //設置拷貝的文件
                from('build/intermediates/bundles/release/')
                //打進jar包后的文件目錄
                into('build/libs/')
                //將classes.jar放入build/libs/目錄下
                //include ,exclude參數來設置過濾
                //(我們只關心classes.jar這個文件)
                include('classes.jar')
                //重命名
                rename ('classes.jar', 'com.rambo.datacollection.jar')
        }
        makeJar.dependsOn(build)

結果,是不是發現在3.0上行不通了???
其實只是路徑換了,我使用的3.6.1上的路徑如下:

Android Studio 3.6.1 classes.jar的路徑


image.png

所以我們要做的,就是把classes的路徑換下,也許在以后的Android Studio版本里,路徑還是會變化,我們要做的就是,在build目錄下找到classes.jar的路徑,填寫正確的路徑即可。
上面的操作,其實就是一個copy,rename的過程,如果你的module里還引用了第三方的jar中,這個如何打包呢,我建議采用如下方式:

task makeJar(type: Jar) {
    delete 'build/mylib.jar'//刪除舊的jar
    destinationDir = file('build/libs')//指定新jar包存放目錄
    archivesBaseName = "mylib"http://指定新jar包名字
    from file('build/classes/java/main')//你寫的代碼的來源,編譯后能找到你代碼的路徑。因人而異,有可能是 from('build/intermediates/bundles/release/classes.jar')
    //第三方庫的jar包存放位置
    from(project.zipTree("libs/fastjson-1.2.55.jar"))
    from(project.zipTree("libs/okhttp-3.10.0.jar"))
    from(project.zipTree("libs/okio-2.2.1.jar"))
    from(project.zipTree("libs/kotlin-stdlib-1.3.11.jar"))
}
makeJar.dependsOn(build)

type由‘Copy’變成了‘Jar’

至于執行task生成對應的jar的方式有兩種:
1.點擊Android Studio最右側的Gradle欄目,選擇對應的module,找到對應的task(在other目錄下),直接執行即可

image.png

2.通過Gradle命令執行
我的代碼需要通過jenkins來生成,所以需要通過命令的方式
gradlew ::datacollectionlib::makeJar
前面的兩個雙引號后面的代表的是module,后面的兩個的雙引號后面的代表的是task名稱

最后去對應的目錄下取一下文件就可以了,如果是直接用在項目里,到這一步就結束了。

Step2:轉換成dex jar,并放置到源碼中

如果你看到這一步,那么你最好看下文章一開始發的兩個鏈接的第一個鏈接。Android分靜態庫和動態庫,我的理解是,靜態庫是跟項目相關的,我一個項目需要
用A這個庫,那么項目就包含了A這個庫,如果庫發生了變化,那么對應的項目也需要重新編譯下;如果是動態庫,這個A庫一旦被加載了,
需要使用A庫的項目都可以去用,并且A庫的內容一旦變化了,相關的項目并不需要重新編譯。
我在實際的操作中,我的庫是需要做成動態庫,是不放在任何項目中的,直接放到源碼里的。一開始直接用step1生成的jar包,打到系統里了,結果發現總是報ClassNotFound的exception,
當時挺奇怪,因為在system/framework下,庫是存在的,對應的permission文件也是有的。后來同事提示說,是不是打的dex的jar,
庫并沒有加載起來。
后來經過驗證,確實是上述問題,知道問題后就很簡單了:
找到android sdk platform-tools目錄,直接一下命令:

dx --dex --output=dest.jar origin.jar

dest.jar就是需要生成的jar的文件,origin.jar就是之前生成的那個jar。兩者的區別,dest.jar放到
Android Studio里看到的是dex文件,后者能夠看到相關的代碼

接下來,我們要編寫一個permission 的xml文件,做動態庫的時候是必須要有這個文件的:
文件的命名我建議用packgae的名稱,jar包也是如此,比如說我的命名如下:
com.ruan.datacollection.jar
com.ruan.datacollection.xml

<?xml version="1.0" encoding="utf-8"?>
<permissions>
    <library name="com.ruan.datacollection"
            file="/system/framework/com.ruan.datacollection.jar"/>
</permissions>

為了驗證我們上面的操作是否有效,我們可以進行如下操作:
1.將com.ruan.datacollection.jar放置到system/framework目錄
2.將com.ruan.datacollection.xml放置到system/etc/permissions/目錄
然后重啟下設備,看看庫是否可用(第三步會說到怎么使用)

如果采用mk文件的方式,在prebuilt.mk中寫入下面兩條命令

PRODUCT_COPY_FILES += vendor/***/DataCollection/com.ruan.datacollection.jar:system/framework/com.ruan.datacollection.jar
PRODUCT_COPY_FILES += vendor/***/DataCollection/com.ruan.datacollection.xml:system/etc/permissions/com.ruan.datacollection.xml

整編下,確認這兩個文件有cp到對應的目錄就可以了

Step3:在項目中使用

因為我的項目,是需要在各個Android Studio項目中使用,所以我就介紹這種方式了,如果是需要在源碼中使用,請自行搜索下如何編寫mk

因為我們的庫是動態庫,按理說,項目是不需要放jar包的,但是如果沒有對應的文件,會導致項目編譯不過,所以還是需要有jar包的
并且這個jar包不能使用后來生成的那個dex的jar,因為dex的jar中不包含源碼,會找不到類。
如果你以前有將framework.jar導入Android Studio的經驗,這一步就很簡單了

先在app目錄下新建一個providedLibs這個目錄,將最初的jar包放置到該目錄下
在gradle文件里,編寫一下代碼

compileOnly fileTree(dir: 'providedLibs', include: ['*.jar'])

這個compileOnly表示這個庫只是參與了編譯,而不在實際的打包中,在之前的Android Studio版本中,這個詞叫provided

這樣就可以的
Tips:如果你run到一個沒有配置該動態庫的設備上,會報錯,ClassNotFound

一點點拓展

前面提到的動態庫需要采用dex的,這個其實也是Android 熱更新的原理,感興趣的可以看看ClassLoader

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

推薦閱讀更多精彩內容