支付宝客户端架构解析:iOS 容器化框架初探

本文涉及的产品
容器镜像服务 ACR,镜像仓库100个 不限时长
简介: 本文将介绍支付宝 iOS 容器化框架设计的基本思路。

1. 前言

由本章节开始,我们将从支付宝客户端的架构设计方案入手,细分拆解客户端在“容器化框架设计”、“网络优化”、“性能启动优化”、“自动化日志收集”、“RPC 组件设计”、“移动应用监控、诊断、定位”等具体实现,带领大家进一步了解支付宝在客户端架构上的迭代与优化历程。

本节将介绍支付宝 iOS 容器化框架设计的基本思路。

2. 容器化实现概览

在 mPaaS 开篇介绍中已经和大家分享过《模块化与解耦式开发在蚂蚁金服 mPaaS 中的实践》:通过容器化开发框架将业务隔离成相对独立的模块,并着力追求模块与模块之间高内聚、低耦合,因此我们实现了灵活的插件式开发,并得以将业务划分为上千个独立工程。

mPaaS iOS 框架源自于支付宝客户端,为了实现这种上千个工程之间的低耦合和相关依赖调用,mPaaS 框架直接接管了 App 的生命周期,负责整个 App 启动托管、App 生命周期管理、处理与分发 UIApplication 的代理事件。
mPaaS 框架提供了容器化环境,业务开发人员在这个容器化环境中使用 微应用服务 进行具体的业务需求开发。

mPaaS iOS 框架-w400

微应用服务 是 mPaaS 框架内定义的概念,主要是用来进行业务模块间的划分。按照是否有 UI 界面作为标准,mPaaS 框架将不同的业务模块划分为 微应用服务微应用 是 APP 运行期间带有用户界面的业务模块;服务 是 App 运行期由业务提供的轻量级抽象服务。在 mPaaS 框架中,通过 框架上下文Context 进行 微应用服务 的生命周期管理。

3. 应用生命周期管理

通过修改 main.m 函数的实现,mPaaS 框架使用自己的 ClientDelegate 类接管了 UIApplicationDelegate 中各种 App 生命周期。mPaaS 框架接入之后,ClientDelegate 完全替代了一般工程中的 AppDelegate 的角色,从而实现了整个应用的生命周期都是由框架进行管理。

int main(int argc, char * argv[]) {
    @autoreleasepool {
        // Now use mPaaS framework
        return UIApplicationMain(argc, argv, @"Application", @"ClientDelegate"); 
    }
}

为了方便用户获取 App 生命周期来开发自定义功能,mPaaS 框架提供了 DTFrameworkInterface 类里面实现了 UIApplicationDelegate 中所有代理方法的等价接入方式,只需要在 DTFrameworkInterface 的 Category 中覆盖对应的方法即可。

例如下面常见的 UIApplicationDelegate 代理方法:

@protocol UIApplicationDelegate<NSObject>

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions;

- (void)applicationDidBecomeActive:(UIApplication *)application;

- (void)applicationWillResignActive:(UIApplication *)application;

- (BOOL)application:(UIApplication *)application
            openURL:(NSURL *)url
  sourceApplication:(NSString *)sourceApplication
         annotation:(id)annotation;

@end

DTFrameworkInterface 中都提供了对应的方法:

typedef NS_ENUM(NSInteger, DTFrameworkCallbackResult)
{
    DTFrameworkCallbackResultContinue   = 0,  // 继续执行
    DTFrameworkCallbackResultReturn     = 1,  // 中断执行
    DTFrameworkCallbackResultReturnYES  = 2,  // 中断之后的逻辑,并返回 YES
    DTFrameworkCallbackResultReturnNO   = 3,  // 中断之后的逻辑,并返回 NO
};

@interface DTFrameworkInterface : NSObject

#pragma mark - 应用配置,微应用配置、服务配置、Scheme 处理器配置,可用 Category 覆盖
- (void)application:(UIApplication *)application beforeDidFinishLaunchingWithOptions:(NSDictionary *)launchOptions;
- (DTFrameworkCallbackResult)application:(UIApplication *)application handleDidFinishLaunchingWithOptions:(NSDictionary *)launchOptions;
- (void)application:(UIApplication *)application afterDidFinishLaunchingWithOptions:(NSDictionary *)launchOptions;

- (DTFrameworkCallbackResult)applicationDidBecomeActive:(UIApplication *)application;

- (DTFrameworkCallbackResult)applicationWillResignActive:(UIApplication *)application;

- (DTFrameworkCallbackResult)application:(UIApplication *)application
                                 openURL:(NSURL *)url
                                  newURL:(NSURL **)newURL
                       sourceApplication:(NSString *)sourceApplication
                              annotation:(id)annotation;
@end

由于 mPaaS 框架有一些自己的初始化逻辑需要实现,在 DTFrameworkInterface 中额外提供了 beforeDidFinishLaunchingWithOptionsafterDidFinishLaunchingWithOptions 方法,方便用户在 App 启动时确定的时间执行自己的初始化代码。

DTFrameworkInterfaceafterDidFinishLaunchingWithOptions 之前会启动 BootLoader,执行 mPaaS 框架的初始化逻辑。在嵌入式操作系统中,BootLoader 的作用是初始化硬件设备,以便为最终调用操作系统内核准备好正确的环境。类似的在 mPaaS 框架中,BootLoader用来初始化整个 mPaaS 框架环境,默认实现为依次执行下面的流程:

  • 创建window
  • 创建主NavigationController
  • 运行那些只运行一次就可以,并且需要率先启动的服务
  • 启动其它所有非lazyload的服务
  • 启动在ServicesMap的"[AUTOSTART]"数组中指定需要自动启动的服务分组
  • 显示主Window
  • 启动 Launcher 微应用,显示出首页

这样就完成了 mPaaS 框架的初始化和首页的显示。

后面将详细介绍其中关键的3个概念:微应用服务框架上下文 Context

4. 微应用

微应用就是带 UI 界面的独立业务模块,其中最特殊的一个微应用是 Launcher 微应用,Launcher 作为 App 启动之后第一个打开的微应用,一般用来创建 App 首页。在 mPaaS 框架中,各个微应用之间是高度独立、不相互依赖的。

微应用 通过 plist 文件配置来进行注册。配置微应用时需要指定 delegate 对应的类名、微应用的描述 description 以及打开微应用时使用的 name。这样 框架上下文 Context 通过微应用的 name 就可以打开指定的微应用。

微应用注册方式-w500

为了方便业务开发,每个 微应用 也存在生命周期。微应用的生命周期,是模仿 iOS App 的生命周期来做的。每个微应用需要实现自己的 DTMicroApplicationDelegate 代理,这个类似于 iOS App 中实现的 Appdelege 类。
微应用生命周期-w550

对于具体业务开发而言 微应用 的开发和一个完整的 App 一样,每个 微应用 负责控制自己应用内的页面堆栈,并根据 微应用 的生命周期执行相应的操作。在 mPaaS 框架中,所有的 微应用 都是运行在 mPaaS 框架提供的容器中,其不需要关注 App 的生命周期。对于一些特殊的业务场景,mPaaS 支持创建微应用的多个实例。

mPaaS 微应用结构-w500

5. 服务

服务微应用 不同地方在于其没有 UI 界面,是在后台执行。一旦服务启动后,其在整个客户端的生命周期中一直存在,因此服务一般用于给微应用提供通用服务,比如执行某个功能或者获取数据等。

一个常见的服务是用户登陆状态服务,每个微应用可以通过这个服务来获取到用户的登录状态和用户信息。

服务 也是通过 plist 文件配置来进行注册。服务注册时需要提供服务的唯一标识 name 和对应的实现类 class 类名。框架在创建 服务 时会利用 Objective-C 语言的运行时机制创建 服务 实现类的实例。lazyLoading 用来控制是否延迟加载该类。如果是延迟加载,在框架启动时该 服务 并不会实例化,只有在用到该 服务 时才会实例化并启动。如果是非延迟加载,则在框架启动时会启动该服务。

由于服务的特殊性,在 mPaaS 中同时提供了 ServicesMap 来批量注册类,ServicesMap 中的 [AUTOSTART] 用来说明哪些组的服务需要在 App 启动的时候最先启动。
服务注册方式-w500

这种分级启动服务的特点可以有效控制 App 的启动时间,提供很好的用户体验。

每个服务都需要实现 服务 接口:

@required

/**
 * 启动一个服务。
 * 注意:
 * 框架在完成初始化操作后,会调用该方法。
 * 如果一个服务要启动一个应用,必须在该方法被调用之后,才能启动其它的应用。
 */
- (void)start;

@optional

/**
 * 创建服务完成。
 */
- (void)didCreate;

/**
 * 服务将要销毁。
 */
- (void)willDestroy;

在增加了 服务 之后,整个 App 的结构如下图所示。后台的服务成为各个 微应用 之间沟通的桥梁。
mPaaS 容器化结构-w500

6. 框架上下文 Context

通过前面的介绍,大家已经对 微应用服务 有了深入的了解。在 mPaaS 框架中,框架上下文Context 承担了一个调度员的角色,负责各个 微应用服务 的调度、通信管理,这样就实现了每个 微应用 的打开、页面推栈以及关闭不会影响 App 其他 微应用 模块。

通过 mPaaS 框架提供的 DTContext * DTContextGet() 函数可以获取到框架上下文Context 对象。一个简化的 Context 类实现如下:

@interface Context : NSObject

@property(nonatomic, strong) UIWindow *window;
@property(nonatomic, strong) UINavigationController *navigationController;

// 根据指定的名称启动一个微应用
- (BOOL)startApplication:(NSString *)name params:(NSDictionary *)params animated:(BOOL)animated;

// 逆向遍历微应用栈,返回最新的微应用对象。
- (DTMicroApplication *)findApplicationByName:(NSString *)name;

// 根据指定的名称查到一个服务
- (id)findServiceByName:(NSString *)name;

// 注册一个服务
- (BOOL)registerService:(id)service forName:(NSString *)name;

// 反注册一个已存在的服务
- (void)unregisterServiceForName:(NSString *)name;
@end

对于业务开发人员,可以通过 框架上下文 Context 获取到主 window、启动指定的 微应用、获取一个 服务、动态注册与反注册 服务,从而实现业务之间的连接。

7. 小结

通过本节内容,我们初步了解了 mPaaS 在 iOS 端容器化框架的设计思路,通过 微应用服务 的方式完成业务模块之间的解耦和调用。框架上下文 Context 作为一个迷你的容器操作系统,为 微应用服务 的运行提供了所需的容器化环境,保证了独立的业务开发流程和流畅的用户体验。

欢迎大家体验 mPaaS 容器化开发框架,期待大家的反馈与交流。

往期阅读

《支付宝客户端架构解析:Android容器化框架初探》

关注我们微信公众号「mPaaS」,获得第一手 mPaaS 技术实践干货

目录
相关文章
|
1月前
|
弹性计算 Java Maven
从代码到容器:Cloud Native Buildpacks技术解析
Cloud Native Buildpacks(CNB)是一种标准化、云原生的容器镜像构建系统,旨在消除手动编写Dockerfile,提供可重复、安全且高效的构建流程。它通过分层策略生成符合OCI标准的镜像,实现应用与基础镜像解耦,并自动化依赖管理和更新。阿里云应用管理支持通过CNB技术一键部署应用至ECS,简化构建和运行流程。
|
5月前
|
Kubernetes Cloud Native 微服务
探索云原生技术:容器化与微服务架构的融合之旅
本文将带领读者深入了解云原生技术的核心概念,特别是容器化和微服务架构如何相辅相成,共同构建现代软件系统。我们将通过实际代码示例,探讨如何在云平台上部署和管理微服务,以及如何使用容器编排工具来自动化这一过程。文章旨在为开发者和技术决策者提供实用的指导,帮助他们在云原生时代中更好地设计、部署和维护应用。
|
6月前
|
安全 Android开发 iOS开发
深入探索Android与iOS的差异:从系统架构到用户体验
在当今的智能手机市场中,Android和iOS无疑是最受欢迎的两大操作系统。本文旨在探讨这两个平台之间的主要差异,包括它们的系统架构、开发环境、安全性、以及用户体验等方面。通过对比分析,我们可以更好地理解为何不同的用户群体可能会偏好其中一个平台,以及这些偏好背后的技术原因。
|
3月前
|
监控 Kubernetes Cloud Native
基于阿里云容器服务Kubernetes版(ACK)的微服务架构设计与实践
本文介绍了如何基于阿里云容器服务Kubernetes版(ACK)设计和实现微服务架构。首先概述了微服务架构的优势与挑战,如模块化、可扩展性及技术多样性。接着详细描述了ACK的核心功能,包括集群管理、应用管理、网络与安全、监控与日志等。在设计基于ACK的微服务架构时,需考虑服务拆分、通信、发现与负载均衡、配置管理、监控与日志以及CI/CD等方面。通过一个电商应用案例,展示了用户服务、商品服务、订单服务和支付服务的具体部署步骤。最后总结了ACK为微服务架构提供的强大支持,帮助应对各种挑战,构建高效可靠的云原生应用。
|
5月前
|
iOS开发 开发者 MacOS
深入探索iOS开发中的SwiftUI框架
【10月更文挑战第21天】 本文将带领读者深入了解Apple最新推出的SwiftUI框架,这一革命性的用户界面构建工具为iOS开发者提供了一种声明式、高效且直观的方式来创建复杂的用户界面。通过分析SwiftUI的核心概念、主要特性以及在实际项目中的应用示例,我们将展示如何利用SwiftUI简化UI代码,提高开发效率,并保持应用程序的高性能和响应性。无论你是iOS开发的新手还是有经验的开发者,本文都将为你提供宝贵的见解和实用的指导。
216 66
|
3月前
|
监控 Cloud Native Java
基于阿里云容器服务(ACK)的微服务架构设计与实践
本文介绍如何利用阿里云容器服务Kubernetes版(ACK)构建高可用、可扩展的微服务架构。通过电商平台案例,展示基于Java(Spring Boot)、Docker、Nacos等技术的开发、容器化、部署流程,涵盖服务注册、API网关、监控日志及性能优化实践,帮助企业实现云原生转型。
|
4月前
|
Kubernetes Linux 虚拟化
入门级容器技术解析:Docker和K8s的区别与关系
本文介绍了容器技术的发展历程及其重要组成部分Docker和Kubernetes。从传统物理机到虚拟机,再到容器化,每一步都旨在更高效地利用服务器资源并简化应用部署。容器技术通过隔离环境、减少依赖冲突和提高可移植性,解决了传统部署方式中的诸多问题。Docker作为容器化平台,专注于创建和管理容器;而Kubernetes则是一个强大的容器编排系统,用于自动化部署、扩展和管理容器化应用。两者相辅相成,共同推动了现代云原生应用的快速发展。
1514 11
|
5月前
|
Kubernetes 安全 数据安全/隐私保护
云卓越架构:容器安全最佳实践
本次分享由阿里云智能集团解决方案架构师张玉峰主讲,主题为“云卓越架构:容器安全最佳实践”。内容涵盖容器安全的挑战、云原生容器安全架构及典型场景。首先分析了容器安全面临的问题,如镜像漏洞和权限管理。接着介绍了容器安全架构的五个维度:身份权限管理、配置安全检查、运行时防护、镜像安全检测及发布的安全管控。最后通过具体场景展示了容器身份与权限管理、密钥管理、运行时防入侵等最佳实践,强调了安全左移的重要性,确保从开发到运行的全生命周期安全覆盖。
|
5月前
|
消息中间件 人工智能 Kubernetes
解密开源Serverless容器框架:事件驱动篇
Knative是一款基于Kubernetes的开源Serverless框架,提供了云原生、跨平台的Serverless编排标准。作为Serverless中必不可少的事件驱动能力,Knative Eventing提供了云原生的事件驱动能力。
|
6月前
|
Kubernetes Cloud Native Docker
云原生之旅:从传统架构到容器化服务的演变
随着技术的快速发展,云计算已经从简单的虚拟化服务演进到了更加灵活和高效的云原生时代。本文将带你了解云原生的概念、优势以及如何通过容器化技术实现应用的快速部署和扩展。我们将以一个简单的Python Web应用为例,展示如何利用Docker容器进行打包和部署,进而探索Kubernetes如何管理这些容器,确保服务的高可用性和弹性伸缩。
OSZAR »