首页 > 其他分享 >你曾遇到的某大厂奇葩问题:Android组件化开发,组件间的Activity页面跳转

你曾遇到的某大厂奇葩问题:Android组件化开发,组件间的Activity页面跳转

时间:2023-06-19 15:08:05浏览次数:59  
标签:app 跳转 Application common Activity 组件 public


组件化开发有什么好处?

1、当项目越来越大时,app的业务越来越复杂,会出现业务功能复杂混乱,各功能块、页面相互依赖,相互调用太多导致耦合度高,而采用组件化开发,我们就可以将功能模块合理的划分,降低功能耦合度。
2、不采用组件化开发时,编译速度缓慢,修改一个页面布局编译一下还得等几分钟。使用组件开发后,每次修改只需要编译对应的模块即可。
3、有利于团队协作开发,开发人员之间职责明确,每一个开发人员只需要关注和负责自己的功能点,互不干扰,提高效率。
4、在多渠道或者合作方需求不一致时可以快速拼接功能模块打包相应的App。

组件化开发项目结构

组件化就是要将项目的各个功能拆成多个模块,可分为app主模块,登录注册模块,个人中心模块,功能模块等。

组件化架构设计:

你曾遇到的某大厂奇葩问题:Android组件化开发,组件间的Activity页面跳转_移动开发

 

1、主工程模块,主要是APP壳,里面不涉及任何逻辑代码,只有权限等配置写在app模块下的AndroidManifest.xml中。

2、常规业务模块,该层的组件就是我们真正的业务组件了。我们通常按照功能模块来划分业务组件,例如注册登录、用户个人中心、APP的首页模块等。这里的每个业务组件都是一个小的APP,只需要修改一下对应的module的build.gradle,就可以单独编译,单独打包成APK在手机上运行。

3、功能组件,一个公共模块,所有的常规业务模块都依赖他。字符串、颜色、尺寸资源等写在该模块下,该组件是一些通用的工具类。

组件之间必须遵循以下规则:
1、只有上层的组件才能依赖下层组件,不能反向依赖,否则可能会出现循环依赖的情况;
2、同一层之间的组件不能相互依赖,这也是为了组件之间的彻底解耦;

常规业务模块间的Activity跳转

由于常规业务模块之间是不能相互依赖的,所以不能直接使用startActivity进行跳转。在这里介绍两种方法:

方法1:
用功能组件(common组件)来统一管理业务组件间的跳转。

业务组件间不能相互依赖,而A组件需要跳转到B组件,那么我们可以让B组件来提供具体的跳转方法,让common组件来承担这个跳转方法的调用,然后A组件依赖common组件即可,是一个迂回的策略。

其他的组件要跳转到另外的组件,也是同样的处理。随着业务增加可能会有X组件,Y组件等等,那么X组件负责提供跳转到X组件的方法,Y组件负责提供跳转到Y组件的方法,并且这些方法,都要经由common组件来管理。

具体代码实现:

1.各个组件应该提供跳转到自己的方法,供别的组件调用
具体如何实现呢?对于A组件的跳转来说,首先,common组件要给别的组件提供一个接口,用于跳转到A组件。
这个接口写在common组件里:

/**
 * 安装用户组件对外暴露的接口
 */

public interface IUserInstallService {
    // 跳转到安装用户页面,其中extra是要传递的数据
    void launch(Context context, String extra);
}

然后在A组件里,实现此接口(当然A组件的gradle文件得添加对common组件的依赖):

public class UserInstallService implements IUserInstallService {
    @Override
    public void launch(Context context, String extra) {
        Intent intent = new Intent(context, UserInstallActivity.class);
        intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
        intent.putExtra("extra_data", extra);    // 传递数据
        context.startActivity(intent);
    }
}

这样,跳转方法就已经暴露出来可以供其他组件调用了,那在B组件里,我们怎么调用呢?得想办法拿到UserInstallService 的实例,然后就能调用它的launch方法来跳转了。

因为common组件还要管理其它组件的跳转,所以这些跳转得统一管理起来:

2.对组件间的跳转进行统一管理
我们在common组件里写一个工厂类,用于分配这些跳转:

public class ServiceFactory {

    private static ServiceFactory instance;//单例模式

    private ServiceFactory() {
    }

    public static ServiceFactory getInstance() {
        if (instance == null) {
            synchronized (ServiceFactory.class) {
                if (instance == null)
                    instance = new ServiceFactory();
            }
        }
        return instance;
    }

    // 安装用户组件的跳转服务的注册和获取
    private IUserInstallService mIUserInstallService;

    public IUserInstallService getIUserInstallService() {
        return mIUserInstallService;
    }

    public void setIUserInstallService(IUserInstallService mIUserInstallService) {
        this.mIUserInstallService = mIUserInstallService;
    }
// 其他组件的跳转服务的注册和获取,与A组件的一样
}

那么很显然,在B组件里,我们就要想办法通过common组件的ServiceFactory 的getLoginService()方法来获取A组件的UserInstallService 的实例。要get,就首先要set,否则拿到的是一个null对象。那么这个set应该放在哪里实现呢?

3.利用Java反射将跳转服务进行实例化
set UserInstallService 对象的操作肯定得放在跳转之前,即get之前。最好在B组件初始化的时候就完成set,而且这个set 操作得放在A组件里,要不然又产生依赖了。

有了这个思路,就可以实现如下:
在common组件里,再增加一个接口:

public interface IAppComponent {
    public void initialize(Application app);
}

这个接口用来表示各个组件的初始化。各个组件都要重写自己的Application,实现这个接口。比如A组件重写的Application类如下:

public class OrgApp extends Application implements IAppComponent{

    @Override
    public void onCreate() {
        super.onCreate();
        initialize(this);
    }

    @Override
    public void initialize(Application app) {
        ServiceFactory.getInstance().setIUserInstallService(new UserInstallService());
    }
}

如果想让A组件的用于组件单独运行时,需要在A组件的AndroidManifest.xml里,指定这个类为组件的application:(注:作为module运行时记得删除android:name=".OrgApp",不然程序会报错。)

<application
        android:name=".OrgApp"
        android:allowBackup="true"
        android:label="@string/app_name"
        android:theme="@style/AppTheme">

但是别忘了,当组件单独运行时,Application类的onCreate()可以走到;而A组件作为module集成之后,Application类是不会被加载的。

怎么办呢?因为app壳组件的Application是肯定会被加载的,所以可以在这里,用反射来加载其他组件的Application类。这也是为什么要在common组件里新建IAppComponent 接口类的原因。app壳组件的Application类也要重写并在manifest里指定。在app壳组件的Application初始化时,可以对其他的组件进行挨个加载,这样,上面我们想要的set 操作就可以在这里完成了。

为了管理要加载的组件,我们在common组件里新建一个AppConfig类,如下;

public class AppConfig {
    public static String[] COMPONENTS = {
            "mod.activity.com.orginfo.application.OrgApp",//这个是A组件的Application类
            // 还有其他的组件的Application类的全名,也都放这里
            "mod.activity.com.login.application.LoginApp"
    };
}

上面这个AppConfig类用来记录所有的组件的Application类的全名。
下面是app壳组件的新Application类:

public class AppApplication extends Application implements IAppComponent {

    @Override
    public void onCreate() {
        super.onCreate();
        initialize(this);
    }

    @Override
    public void initialize(Application app) {
        // 遍历所有的组件的Application类,依次用反射的方式实例化
        for (String component : AppConfig.COMPONENTS) {
            try {
                Class<?> clazz = Class.forName(component);
                Object object = clazz.newInstance();
                // 实例化后,调用各个组件的 set 方法
                if (object instanceof IAppComponent) {
                    ((IAppComponent) object).initialize(app);
                }
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }
}

经过这样的处理,上面第二步最后“该在哪里set 跳转服务”的问题就解决了。总结本篇实现的组件间跳转原理如下:

在应用启动的时候,它的APP壳组件的Application类会进行初始化,在这里我们通过反射的方式初始化了其他各组件的Application类,而各组件的Application类在初始化时,又会通过set操作把自己的跳转服务注册到common组件。这样的话,通过common组件get对应的服务,即可实现跳转。

我们来试一下,在B组件里,跳转到A组件,就可以这么写了;

// 安装用户
ServiceFactory.getInstance().getIUserInstallService().launch(mContext, "");

这样B组件就不依赖A组件也可以进行跳转,实现了我们的期望。

还有一点,要是A组件没有被集成到app里,那么ServiceFactory.getInstance().getIUserInstallService()就是null,会报空指针异常。我们需要把getIUserInstallService()补充下:

public IUserInstallService getIUserInstallService() {
        if (mIUserInstallService == null)
            mIUserInstallService = new EmptyUserInstallService();
        return mIUserInstallService;
    }

这里EmptyUserInstallService是对IUserInstallService 接口的一个空实现,目的只是为了避免这个空指针异常,就不贴代码了。

方法2:
采用路由来实现页面跳转,比较成熟的有:

美团的WMRouter:

阿里的ARouter:

常规业务模块间的数据通讯

使用EventBus实现module间的通讯

组件化项目的混淆方案

组件化项目的Java代码混淆方案采用在集成模式下集中在app壳工程中混淆,各个业务组件不配置混淆文件。集成开发模式下在app壳工程中build.gradle文件的release构建类型中开启混淆属性,其他buildTypes配置方案跟普通项目保持一致,Java混淆配置文件也放置在app壳工程中,各个业务组件的混淆配置规则都应该在app壳工程中的混淆配置文件中添加和修改。

之所以不采用在每个业务组件中开启混淆的方案,是因为 组件在集成模式下都被 Gradle 构建成了 release 类型的arr包,一旦业务组件的代码被混淆,而这时候代码中又出现了bug,将很难根据日志找出导致bug的原因;另外每个业务组件中都保留一份混淆配置文件非常不便于修改和管理,这也是不推荐在业务组件的 build.gradle 文件中配置 buildTypes (构建类型)的原因。

注意事项:
1、需要跨module使用的依赖需要用api,而不是implementation,否则common模块引入库其他模块使用不了。
2、其他module想作为APP单独运行时,需要修改module所在的build.gradle:

application属性,可以独立运行的Android程序,也就是我们的APP;

apply plugin: ‘com.android.application’

library属性,不可以独立运行,一般是Android程序依赖的库文件

apply plugin: ‘com.android.library’

更多后续欢迎关注,有任何建议可以私信沟通,期待与你的交流。

标签:app,跳转,Application,common,Activity,组件,public
From: https://blog.51cto.com/u_16163453/6513725

相关文章

  • Jetpack组件库(含Jetpack Compose)从入门到精通全家桶【附Demo】
    前言开发应用程序就像搭积木。我们对产品业务及功能模块的划分和封装,就像在搭建积木一样。积木不能太大,这不利于修改和拆解;积木也不能太小,否则管理起来可能会很混乱。只有基于稳健、合理的架构,项目才能轻松应对需求的变化,才有可能健康成长。没有良好架构的应用程序,就像没有搭好底......
  • Android-Kotlin-Activity直接的跳转
    1.选中应用包名packageName,右键:2.选中Kotlin:3.创建Kotlin的Activity完成:第一个Activity,MainActivitypackagecn.kotlinimportandroid.content.Intentimportandroid.support.v7.app.AppCompatActivityimportandroid.os.Bundleimportandroid.util.Logimportandroid.view.......
  • Android应用开发者,你们真的了解Activity的生命周期吗?
    一开始,我觉得Activity的生命周期虽然过于复杂,但它不应该是一个难题。我的意思是:对于Android开发新手来说,如何正确地处理Activity生命周期可能有点困难,但是我无法想象对于那些富有经验的android开发者来说,这依然是一个棘手的问题。我还是想的太简单了。一会儿我会告诉你整个故事,但是......
  • 监听Activity生命周期方式及案例讲解
    本篇文章主要讲解如何快速实现Activity生命周期监听,以及其在官方lifecycle、第三方库Glide、PermissionX中的应用1.Activity生命周期监听Fragment实现Activity生命周期监听众所周知,Fragment中生命周期分发主要是依赖Activity,所以为了监听Activity的生命周期我们直接添加一个空的Fr......
  • Android进阶宝典 -- JetPack Navigation的高级用法(解决路由跳转新建Fragment页面问题)
    相信有相当一部分的伙伴,在项目开发中依然使用Activity作为页面承载体,有10个页面就会有10个Activity,这种方式当然没问题,但是如果涉及到页面间数据共享,那么使用多Activity就不是很方便了,需要Activity传递各种数据,涉及到数据的序列化与反序列化;因此产生了单Activity和多Fragment架构,所......
  • Tab切换以及倒计时组件封装
    1、Tab组件功能支持默认选中tab子元素可以是文本或者图片自定义tab的数量,并自适应展示实现方式用ul>li标签遍历传入的tabs数组参数渲染判断是否传入背景,未传则显示文字绑定点击事件特点简单易用可适配性2、倒计时组件功能常用于榜单或者活动结束倒计时......
  • 得到、微信、美团、爱奇艺APP组件化架构实践
    一、背景随着项目逐渐扩展,业务功能越来越多,代码量越来越多,开发人员数量也越来越多。此过程中,你是否有过以下烦恼?项目模块多且复杂,编译一次要5分钟甚至10分钟?太慢不能忍?改了一行代码或只调了一点UI,就要run整个项目,再忍受一次10分钟?合代码经常发生冲突?很烦?被人偷偷改了自己模块的代......
  • Android组件开发简介
    一、背景一个app随着业务增加,代码放在同一个模块中会越来越臃肿,同时也导致多人开发的一个难度。组件化可以把业务单独分出来,形成一个单独模块,可单独运行、测试等,相互之间不会影响。另外一个优势,如果一个公司有多个app,总会出现一些相同业务,如登录/注册。我们可以单独把公共业务封装......
  • 【Android】一文读懂 Activity 的生命周期
    作为Android开发人,如果说连Activity生命周期都没搞懂,会走非常多的弯路,所以这篇文章我就对Activity生命周期的生命周期进行一个简单的总结。单Activity生命周期的整体流程首先,我们创建一个My_A_Activity,并且打印它的各个生命周期方法。classMy_A_Activity:AppCompatActivity(){......
  • 前端Vue图片上传组件支持单个文件多个文件上传 自定义上传数量 预览删除图片 图片压缩
    前端Vue图片上传组件支持单个文件多个文件上传自定义上传数量预览删除图片图片压缩,下载完整代码请访问uni-app插件市场址:https://ext.dcloud.net.cn/plugin?id=13099效果图如下:1.0.0(2023-06-18)组件初始化使用方法<!--count:最大上传数量 imageList:图片上传选......