Android-组件化开发+详解+源码+实战

Android-组件化开发+详解+源码+实战,第1张

概述经过一段时间的学习,我对组件开发也有了一套自己的见解,所以以下内容仅仅是我的见解,如有不足希望评论提出。组件化主要从三个方面进行拆分1.GradleGradle内主要设置开关在Module(app)中,开关的目的是判断Module是否以library的方式在程序中运行在Module(module)中,来判

经过一段时间的学习,我对组件化开发也有了一套自己的见解,所以以下内容仅仅是我的见解,如有不足希望评论提出。

组件化主要从三个方面进行拆分

1 . Gradle

Gradle 内主要设置开关
在Module(app)中,开关的目的是判断Module是否以library的方式在程序中运行
在Module(module)中,来判断Module是library还是Application,从这两种方式中为Module设置不同的清单文件(两种方式的清单文件不同)

2 . 代码

代码中组件化又笼统的分为三部分

2.1 第一部分为library部分,组件化的App部分与Module不能直接交流,这时就需要一个library作为中间商,library可以被App和Module共同引入,所以通过library作为中间商最合适不过。 也可以将library理解为 MVP 模式中的 P层。

2.2 第二部分则为Module部分,该部分根据开关,既可以作为library依赖于App部分运行,也可以作为一个单独的App运行(换句话说:当Module为library时,必须依赖于App部分才能运行,此时可以与App进行数据交互等 *** 作。当Module作为Application时,Module就相当于一个独立的个体,与App部分不能进行数据交互,但是可以与其他依赖进行交互)

2.3 第三部分为App部分,也是主程序部分,该部分位于程序的主界面,Module部分则作为主界面的组成成分。

3 . 清单文件

此部分主要分为两个模块

3 .1 当该模块作为Application时,为该模块设置一个自定义的Application,主要用于初始化 *** 作

3 .2 为Module作为library和Application分别设置不同的清单文件

接下来进行实战演练

需求:从App界面跳转到 Module界面

我们按照上面步骤进行编写
首先,不建议只去看此代码,实战才是学习编程的硬道理,自己敲一遍,两遍,三遍,这些代码就是你自己的了,我们和大牛的差距不是智力方面,是经验方面,碰到的问题多了,经手的代码多了,添一些自己的理解,就成自己的了。
最后会免费送上源码供第一次接触的小伙伴学习。

首先新建一个 名为ZJH2的Project


file-New-NewModule-Phone&tablet。然后新建一个Module 名为module,
这里需要注意Module的Activity名字尽量不要和App中的一样,避免错乱,我的名字为ModuleActivity


file-New-NewModule-AndrID library。然后新建一个library 名为mylibrary.


再次讲述一下需求,从app的界面,跳转到module的界面。。。。

接下来正式进入正题

Gradle部分
1 . gradle.propertIEs部分添加

isModuleApplication = true

作为开关,当为true时,可以单独运行,但是不能与app进行数据交互


2 . build.gradle(module) 内
分别在以下三个部分进行修改,如图,对开关进行判断,然后进行不同的调用,其中清单文件部分,会在后面讲到。

f (isModuleApplication.toBoolean()){    apply plugin: 'com.androID.application'}else{    apply plugin: 'com.androID.library'}
if (isModuleApplication.toBoolean()){    applicationID "com.example.module"}
sourceSets{    main{        // 在独立运行或者作为libarary调试时,使用不同的AndroIDManifest.xml文件        if (isModuleApplication.toBoolean()){            manifest.srcfile 'src/main/manifest/AndroIDManifest.xml'        }else {            manifest.srcfile 'src/main/AndroIDManifest.xml'        }    }}
implementation project(':mylibrary')

3 . build.gradle(app) 内
这里面就是先导入 中间商 ,在判断Module是否可以作为library引入。

implementation project(':mylibrary')if (!isModuleApplication.toBoolean()){    implementation project(':module')}

2 . 代码部分
1 . library部分
此处要是一句一句介绍实在是太麻烦了,我把讲解直接写在代码头部注释了,library部分需要建立这些类和接口,第一次感觉可能会有点多,不过多敲几遍之后就感觉不到了(主要是麻木了)


由上而下贴图咯
AppConfig

/** * 这个类用来统计每个Module的 Application 的, * 用于当Module 作为 Application 时使用,别看现在这个Demo的 Module 少,组件化的目的就是用于大量功能代码的,所以一个个的添加多累累,通过这种集中营的方式老方便了 */public class AppConfig {    public static final String[] Components = {          "com.example.module.ModuleApplication"    };}

libraryApplication

/** * 此界面用于初始化 *** 作,因为跳转 *** 作需要用到 Set/Get  *** 作,所以Set *** 作要在整个程序开始的时候进行 *** 作 * 所以众所周知,整个程序开始的时候是App部分运行的时候,所以这个接口就用在了 App部分的 Application * 也可以理解为将 初始化 放在整个程序运行的第一个界面,就是初始化越早越好咯 * * */public interface libraryApplication {    voID initialize(androID.app.Application application);}

ModuleClass

import androID.content.Context;/** * 这个类用来具体实现接口内的方法的,当调用时用来返回接口和方法滴 * 暂时先这么理解吧,俺也在分享中慢慢学习 * */public class ModuleClass implements ModuleService{    @OverrIDe    public voID myIntent(Context txt) {    }}

ModuleService

import androID.content.Context;/** * 此接口用来写要实现什么方法滴,需要什么方法就写在这里面 * 这次咱就用到一个跳转方法 * 该接口需要让对应的类去实现 * */public interface ModuleService {    voID myIntent(Context txt);//我就是那个跳转方法}

ServiceFactory

/** * 这个是中间商的关键部门,用来与 App 和 Module 进行交互的主类 * 你看这个中间商 里面的 类和接口 这么多,不找个 老大管着 不得乱套啊 * 所以其模块需要用到其他模块的时候就从这个类里面找 * * */public class ServiceFactory {    private ModuleService moduleService;    public static final ServiceFactory service = new ServiceFactory();    public ServiceFactory(){}    public static ServiceFactory getInstance(){        return service;    }    public ModuleService getMainService(){        if (moduleService == null) {            moduleService = new ModuleClass();        }        return moduleService;    }    public voID setMainService(ModuleService moduleService){        this.moduleService = moduleService;    }}

2 . Module部分

就这些小代码就够了

上图:
ModuleActivity

import androIDx.appcompat.app.AppCompatActivity;import androID.os.Bundle;//Module Activity没啥好说滴public class ModuleActivity extends AppCompatActivity {    @OverrIDe    protected voID onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentVIEw(R.layout.activity_module);    }}

ModuleApplication

import androID.app.Application;import com.example.mylibrary.libraryApplication;import com.example.mylibrary.ServiceFactory;//当Module作为Application时进行初始化用滴public class ModuleApplication extends Application implements libraryApplication {    @OverrIDe    public voID onCreate() {        super.onCreate();    }    @OverrIDe    public voID initialize(Application application) {        ServiceFactory.getInstance().setMainService(new ModuleService());    }}

ModuleService

@H_419_246@

import androID.content.Context;import androID.content.Intent;/** * 因为别的模块使用跳转方法 跳不过来,所以把跳转方法 写到自己的模块里面,供其他模块调用这样太方便了吧,又获取一个新的思路其实同一个模块的也可以使用此方式,把需要的方法写在自己的类里面,供外部的类去调用,专业术语好像叫 减少代码的冗余这是什么道理 - 和自己有关的事情尽量不要放到别人那里,心里不踏实 */public class ModuleService implements com.example.mylibrary.ModuleService {    @OverrIDe    public voID myIntent(Context txt) {        txt.startActivity(new Intent(txt,ModuleActivity.class));    }}

3 . App部分

代码更少咯,这部分主要就是初始化 *** 作,真正项目中也是,App部分的代码很少,基本有个 启动页就不错了,其他都在相关组件里面实现


上图:
MainActivity

import androIDx.appcompat.app.AppCompatActivity;import androID.os.Bundle;import androID.vIEw.VIEw;import androID.Widget.TextVIEw;import com.example.mylibrary.ServiceFactory;/** * 此界面就是在布局文件中添加一个 跳转事件 * */public class MainActivity extends AppCompatActivity {    @OverrIDe    protected voID onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentVIEw(R.layout.activity_main);        TextVIEw tv_jump = findVIEwByID(R.ID.tv_jump);        tv_jump.setonClickListener(new VIEw.OnClickListener() {            @OverrIDe            public voID onClick(VIEw v) {                ServiceFactory.getInstance().getMainService().myIntent(MainActivity.this);            }        });    }}

MainApplication

import androID.app.Application;import androID.util.Log;import com.example.mylibrary.AppConfig;import com.example.mylibrary.libraryApplication;import com.example.mylibrary.ServiceFactory;import static androIDx.constraintlayout.motion.utils.Oscillator.TAG;/** * 这里就是进行初始化 *** 作的部分,将这个类放到 清单文件中就OK了,后面会讲到 * */public class MainApplication extends Application implements libraryApplication {    public static Application application;    public static Application getApplication(){        return application;    }    @OverrIDe    public voID onCreate() {        super.onCreate();        initialize(this);    }    @OverrIDe    public voID initialize(Application application) {        // 遍历所有的组件的Application类,依次用反射的方式实例化        for (String component : AppConfig.Components) {            try {                Class<?> clazz = Class.forname(component);                Object object = clazz.newInstance();                // 实例化后,调用各个组件的 set 方法                if (object instanceof libraryApplication) {                    ((libraryApplication) object).initialize(application );                }            } catch (Exception e) {                e.printstacktrace();                Log.e(TAG, "TAG"+e.getMessage());            }        }    }}

4 . 清单文件部分

1.App的清单文件

3 . Module的清单文件


首先将AndroID Studio 设置为 Project 模式
在 module - mian 下面建立 manifest 文件夹 ,名字与Gradle部分当时设置的那个文件夹名字要对应起来,


将原来的 清单文件复制到 manifest里面去
manifest 里面的清单文件

<?xml version="1.0" enCoding="utf-8"?><manifest xmlns:androID="http://schemas.androID.com/apk/res/androID"    package="com.example.module">    <application        androID:name=".ModuleApplication"        androID:allowBackup="true"        androID:icon="@mipmap/ic_launcher"        androID:label="@string/app_name"        androID:roundIcon="@mipmap/ic_launcher_round"        androID:supportsRtl="true"        androID:theme="@style/theme.ZJH2">        <activity androID:name=".ModuleActivity">            <intent-filter>                <action androID:name="androID.intent.action.MAIN" />                <category androID:name="androID.intent.category.LAUNCHER" />            </intent-filter>        </activity>    </application></manifest>

外面的清单文件

<?xml version="1.0" enCoding="utf-8"?><manifest xmlns:androID="http://schemas.androID.com/apk/res/androID"    package="com.example.module">    <application>        <activity androID:name=".ModuleActivity"/>    </application></manifest>

然后运行一下小程序,就能正常点击跳转了,这是源生的 组件跳转方法,其实现在用阿里的 ARouter 只需两句话就能跳转了

源码下载链接 组件化源码
推荐链接 阿里ARouter开源组件化框架项目实践 ARouter

总结

以上是内存溢出为你收集整理的Android-组件化开发+详解+源码+实战全部内容,希望文章能够帮你解决Android-组件化开发+详解+源码+实战所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

欢迎分享,转载请注明来源:内存溢出

原文地址: https://outofmemory.cn/web/1211065.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-06-04
下一篇 2022-06-04

发表评论

登录后才能评论

评论列表(0条)

保存