Android插件化之VirtualAPK框架Jar包开发

简介: 前言:我上一篇写了VirtualAPK的一个初试,写了一个简单的demo,然后如果是开发APP的情况(不考虑大部分的坑),我觉得上一篇的那种插件化模式是可以使用的。

前言:

我上一篇写了VirtualAPK的一个初试,写了一个简单的demo,然后如果是开发APP的情况(不考虑大部分的坑),我觉得上一篇的那种插件化模式是可以使用的。那现在我这边是有一个新的需求,我不开发APP,我开发SDK,那我就要写个包含VirtualAPK框架的aar,但现在其实很多国内的人写SDK并不是用aar,而是使用jar包的方式,那我今天就做一个使用VirtualAPK框架的jar包。

一.开发流程

我就不重新写一个了,就接着上一篇的代码来讲:
https://www.jianshu.com/p/3022f4b0ae28
上一个Demo开发app模式我是用了两个module,一个宿主的module,一个插件的module。现在开发jar包,就要再加上一个壳的module来模拟调用jar包。
也就是壳的module -> shell ; 宿主的module -> home ; 插件的module -> plugin 。

二.引用jar包

1.打jar包

首先我们要在home的gradle中写打jar包的命令,这个在网上找很多。比如我的

    task buildJar(dependsOn: ['compileReleaseJavaWithJavac'], type: Jar) {
    archiveName = 'testVirtual.jar'
    def srcClassDir = [project.buildDir.absolutePath + "/intermediates/classes/release"];
    from srcClassDir
    exclude "**/R.class"
    exclude "**/R\$*.class"
    }

我这把home打出来的jar包命名为testVirtual.jar

2.引用jar包

现在我们开发shell来引用jar包,我这shell很简单,直接跳转home的类OneActivity(这个类是前一篇的Demo)。

public class MainActivity extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Intent intent = new Intent(MainActivity.this, OneActivity.class);
        startActivity(intent);
    }
}

然后把testVirtual.jar放到libs文件夹下,在shell的gradle中假如引用(这个一般默认就有)

dependencies {
    implementation fileTree(dir: 'libs', include: ['*.jar'])
}
3.补全文件

当然仅仅像上面一样还无法运行。因为我们打的是jar包而不是aar。所以我们要把home的资源补全到shell中,正巧的是我这里的资源只有一个activity_one,把它放到shell的layout文件夹中,正好demo中也只有一个activity,所以要在shell的manifest中定义。最后因为home中有引用 implementation 'com.didi.virtualapk:core:0.9.8',我们还需要把它变成jar包放到shell的libs中。
这里可以说下怎样根据implementation 找到对应的jar包:
打开c盘/用户/用户名/.gradle/caches/modules-2/files-2.1文件夹,然后根据框架的内容去找,比如我这个就找到

img_c8139311f06ef78db8bb426313703cc1.png

进去之后根据implementation 的命名找到
img_a72e4fd6e42ab25c8dbaa1f23e557875.png

3个文件夹里面我找到一个jar包不知道行不行,然后我自己是根据这个aar弄的,因为我怕框架中会有资源文件,所以是拿这个aar去解压拿jar包(事实证明没有资源文件)
img_d23c33c92280dc77d3f29c41fb4d3d1e.png

把获取到的jar包改个名字然后放进shell的libs中就能正常运行了。

注意:
有可能会报资源的id错误,说明资源可能会有冲突,解决的方法是不使用R.的方式去引用资源,可以使用context.getResources().getIdentifier(name, type, context.getPackageName());

三.调用插件

shell中没有什么要写的,因为调用插件的操作我们写在home中,只要继承home的application就行。
然后用命令行给插件打包,复习一下,用gradlew clean assemblePlugin
把插件放到特定的文件夹中(这边的操作就和上一篇一样,都是home与plugin的操作,和shell一点关系都没有)

最后运行,看过上一篇的都知道,我在上一篇中home和plugin的互动有两个地方,一是home调用plugin的一个方法打印日志然后回调改变button,二是home跳转plugin的一个activity。
如果按我的方法来做会发现第一个地方正常,home能正常调用plugin的方法,但是第二个地方报错,home无法跳转plugin的activity,报错
android.content.ActivityNotFoundException: Unable to find explicit activity class {com.example.plugin.virtualmastertest/com.didi.virtualapk.core.A$1}; have you declared this activity in your AndroidManifest.xml?

就是没在manifest中定义的意思,但是为什么我之前写的不需要在manifest中定义,而打jar包就需要,而且是com.didi.virtualapk.core.A$1。

然后我就反编译之前的home的apk,发现反编译后的manifest中多了下面这块东西


img_144b3d69e4b77eb6c15848ee60bcc7b8.png
        <activity android:exported="false" android:launchMode="standard" android:name="com.didi.virtualapk.delegate.StubActivity"/>
        <activity android:exported="false" android:launchMode="standard" android:name="com.didi.virtualapk.core.A$1"/>
        <activity android:exported="false" android:launchMode="standard" android:name="com.didi.virtualapk.core.A$2" android:theme="@android:style/Theme.Translucent"/>
        <activity android:exported="false" android:launchMode="singleTop" android:name="com.didi.virtualapk.core.B$1"/>
        <activity android:exported="false" android:launchMode="singleTop" android:name="com.didi.virtualapk.core.B$2"/>
        <activity android:exported="false" android:launchMode="singleTop" android:name="com.didi.virtualapk.core.B$3"/>
        <activity android:exported="false" android:launchMode="singleTop" android:name="com.didi.virtualapk.core.B$4"/>
        <activity android:exported="false" android:launchMode="singleTop" android:name="com.didi.virtualapk.core.B$5"/>
        <activity android:exported="false" android:launchMode="singleTop" android:name="com.didi.virtualapk.core.B$6"/>
        <activity android:exported="false" android:launchMode="singleTop" android:name="com.didi.virtualapk.core.B$7"/>
        <activity android:exported="false" android:launchMode="singleTop" android:name="com.didi.virtualapk.core.B$8"/>
        <activity android:exported="false" android:launchMode="singleTask" android:name="com.didi.virtualapk.core.C$1"/>
        <activity android:exported="false" android:launchMode="singleTask" android:name="com.didi.virtualapk.core.C$2"/>
        <activity android:exported="false" android:launchMode="singleTask" android:name="com.didi.virtualapk.core.C$3"/>
        <activity android:exported="false" android:launchMode="singleTask" android:name="com.didi.virtualapk.core.C$4"/>
        <activity android:exported="false" android:launchMode="singleTask" android:name="com.didi.virtualapk.core.C$5"/>
        <activity android:exported="false" android:launchMode="singleTask" android:name="com.didi.virtualapk.core.C$6"/>
        <activity android:exported="false" android:launchMode="singleTask" android:name="com.didi.virtualapk.core.C$7"/>
        <activity android:exported="false" android:launchMode="singleTask" android:name="com.didi.virtualapk.core.C$8"/>
        <activity android:exported="false" android:launchMode="singleInstance" android:name="com.didi.virtualapk.core.D$1"/>
        <activity android:exported="false" android:launchMode="singleInstance" android:name="com.didi.virtualapk.core.D$2"/>
        <activity android:exported="false" android:launchMode="singleInstance" android:name="com.didi.virtualapk.core.D$3"/>
        <activity android:exported="false" android:launchMode="singleInstance" android:name="com.didi.virtualapk.core.D$4"/>
        <activity android:exported="false" android:launchMode="singleInstance" android:name="com.didi.virtualapk.core.D$5"/>
        <activity android:exported="false" android:launchMode="singleInstance" android:name="com.didi.virtualapk.core.D$6"/>
        <activity android:exported="false" android:launchMode="singleInstance" android:name="com.didi.virtualapk.core.D$7"/>
        <activity android:exported="false" android:launchMode="singleInstance" android:name="com.didi.virtualapk.core.D$8"/>
        <service android:exported="false" android:name="com.didi.virtualapk.delegate.LocalService"/>
        <service android:exported="false" android:name="com.didi.virtualapk.delegate.RemoteService" android:process=":daemon">
            <intent-filter>
                <action android:name="com.example.bn.virtualapktest.intent.ACTION_DAEMON_SERVICE"/>
            </intent-filter>
        </service>
        <provider android:authorities="com.example.bn.virtualapktest.VirtualAPK.Provider" android:exported="false" android:name="com.didi.virtualapk.delegate.RemoteContentProvider" android:process=":daemon"/>

PS:我也不懂这个是什么,也许就是那个坑位的意思(之后我研究之后会再说)
最后我们把这段东西放到shell的manifest中,就能正常的运行了,就可以实现VirtualAPK打jar包后插件化,就是shell+home+plugin的模式。

这篇到这里就结束,和上篇差了挺多时间,这段时间挺忙,基本没能弄这些,最后这个框架我可能之后最后写一篇关于原理的,结合插件化和其它框架讲。

目录
相关文章
|
5天前
|
Android开发 开发者 Kotlin
探索安卓开发中的新特性
【9月更文挑战第14天】本文将引导你深入理解安卓开发领域的一些最新特性,并为你提供实用的代码示例。无论你是初学者还是经验丰富的开发者,这篇文章都会给你带来新的启示和灵感。让我们一起探索吧!
|
2天前
|
Java Linux Android开发
深入理解Android开发:从基础到高级
【9月更文挑战第17天】本文将深入探讨Android开发的各个方面,包括应用开发、操作系统等。我们将通过代码示例来展示如何创建一个简单的Android应用,并解释其背后的原理。无论你是初学者还是有经验的开发者,这篇文章都将为你提供有价值的信息和启示。
|
1天前
|
前端开发 Java 数据库
💡Android开发者必看!掌握这5大框架,轻松打造爆款应用不是梦!🏆
在Android开发领域,框架犹如指路明灯,助力开发者加速应用开发并提升品质。本文将介绍五大必备框架:Retrofit简化网络请求,Room优化数据库访问,MVVM架构提高代码可维护性,Dagger 2管理依赖注入,Jetpack Compose革新UI开发。掌握这些框架,助你在竞争激烈的市场中脱颖而出,打造爆款应用。
17 3
|
9天前
|
IDE 开发工具 Android开发
安卓与iOS开发对比:平台选择对项目成功的影响
【9月更文挑战第10天】在移动应用开发的世界中,选择正确的平台是至关重要的。本文将深入探讨安卓和iOS这两大主要移动操作系统的开发环境,通过比较它们的市场份额、开发工具、编程语言和用户群体等方面,为开发者提供一个清晰的指南。我们将分析这两个平台的优势和劣势,并讨论如何根据项目需求和目标受众来做出最佳选择。无论你是初学者还是有经验的开发者,这篇文章都将帮助你更好地理解每个平台的特性,并指导你做出明智的决策。
|
5天前
|
XML 编解码 Android开发
安卓开发中的自定义视图控件
【9月更文挑战第14天】在安卓开发中,自定义视图控件是一种高级技巧,它可以让开发者根据项目需求创建出独特的用户界面元素。本文将通过一个简单示例,引导你了解如何在安卓项目中实现自定义视图控件,包括创建自定义控件类、处理绘制逻辑以及响应用户交互。无论你是初学者还是有经验的开发者,这篇文章都会为你提供有价值的见解和技巧。
13 3
|
7天前
|
API Android开发 iOS开发
安卓与iOS开发中的线程管理对比
【9月更文挑战第12天】在移动应用的世界中,安卓和iOS平台各自拥有庞大的用户群体。开发者们在这两个平台上构建应用时,线程管理是他们必须面对的关键挑战之一。本文将深入探讨两大平台在线程管理方面的异同,通过直观的代码示例,揭示它们各自的设计理念和实现方式,帮助读者更好地理解如何在安卓与iOS开发中高效地处理多线程任务。
|
9天前
|
开发框架 Android开发 iOS开发
探索安卓与iOS开发的差异:构建未来应用的指南
在移动应用开发的广阔天地中,安卓与iOS两大平台各占半壁江山。本文将深入浅出地对比这两大操作系统的开发环境、工具和用户体验设计,揭示它们在编程语言、开发工具以及市场定位上的根本差异。我们将从开发者的视角出发,逐步剖析如何根据项目需求和目标受众选择适合的平台,同时探讨跨平台开发框架的利与弊,为那些立志于打造下一个热门应用的开发者提供一份实用的指南。
23 5
|
9天前
|
开发工具 Android开发 iOS开发
安卓与iOS开发:平台选择的艺术与科学
在移动应用开发的广阔天地中,安卓与iOS两大平台如同东西方哲学的碰撞,既有共通之处又各具特色。本文将深入探讨这两个平台的设计理念、开发工具和市场定位,旨在为开发者提供一份简明扼要的指南,帮助他们在这场技术与商业的博弈中找到自己的道路。通过比较分析,我们将揭示每个平台的优势与局限,以及它们如何影响应用的性能、用户体验和市场接受度。无论你是初涉江湖的新手,还是经验丰富的老手,这篇文章都将为你的选择提供新的视角和思考。
23 5
|
8天前
|
搜索推荐 Android开发 UED
安卓开发中的自定义视图:打造个性化用户界面
【9月更文挑战第11天】在安卓应用开发领域,自定义视图是实现独特用户体验的基石。本文将引导你通过一个简单的自定义视图示例,探索如何从零开始创建并应用自定义组件,以增强你的应用界面。我们将一起学习如何扩展View类,重写onDraw方法,处理触摸事件,并最终在我们的安卓项目中使用这个自定义视图。无论你是初学者还是有一定经验的开发者,这篇文章都将为你提供清晰的步骤和实用的技巧,帮助你提升用户界面设计的能力。
|
9天前
|
开发工具 Android开发 Swift
探索安卓与iOS开发的差异:从新手到专家的旅程
在数字时代的浪潮中,移动应用开发已成为连接世界的桥梁。本文将深入探讨安卓与iOS这两大主流平台的开发差异,带领读者从零基础出发,逐步了解各自的特点、开发环境、编程语言及市场策略。无论你是梦想成为移动应用开发者的初学者,还是希望扩展技能边界的资深开发者,这篇文章都将为你提供宝贵的见解和实用的建议。