热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

第一章常见跨平台解决方案及Flutter架构

近些年来,不断的有前端跨平台方案涌现出来,比如基于WebView的Cordova,还有渲染成原生控件的Reactive、Weex等,那么,这些跨平台方案有什么通用的实现思路呢,而Flutter的

近些年来,不断的有前端跨平台方案涌现出来,比如基于WebView的Cordova,还有渲染成原生控件的Reactive、Weex等,那么,这些跨平台方案有什么通用的实现思路呢,而Flutter的设计思路与他们的区别又是什么呢。带着这些疑问,本章将会介绍什么是跨平台,常见的跨平台方案有哪些,以及Flutter的实现方案。

1.1 跨平台解决方案

1.1.1 什么是跨平台

我们知道,CPU有不同的架构和指令集,上层也有不同的操作系统,一个系统的可执行文件在另一个系统上就是不可执行的,比如Windows的exe文件在Mac上就不能直接执行。这是因为不同平台提供的API不同,因此需要不同平台单独维护代码。这样就带来了几个问题:

1.开发成本大,各平台都要独立开发一套代码

2.测试成本大,各平台都需要单独测试

3.维护成本大,各平台都要独立迭代维护

因此,大家都在试图找到一种方案,能够实现一份代码运行在任意平台,这样就可以大大提升开发、测试效率,降低维护成本,这种方案就是跨平台。

1.1.2 各领域跨平台方案

1 浏览器

浏览器是一种历史悠久的跨平台方案,通过在不同平台安装对应平台的浏览器,来屏蔽底层的差异,浏览器提供统一的DOM API,这样上层的网页开发就可以统一按照W3C标准,实现一套代码运行在不同平台的浏览器中,由浏览器引擎完成页面渲染。

image.png

图1.1 浏览器方案

2 Docker

Docker是一个开源的应用容器引擎,Docker可以让开发者将他们的开发环境、代码、配置文件等一并打包到一个轻量级、可移植的容器中,然后发布和应用到任意平台。Docker是通过通过在操作系统之上加一个虚拟层,在这层划分一到多个容器,容器里再去跑系统、App,然后动态的将硬件资源分配给容器,这样可以实现硬件和软件的分离。

image.png

图1.2 Docker

3 Java

Java实现跨平台是因为Java虚拟机JVM,我们将不同平台的虚拟机安装在不同的操作系统中,开发者只需要将Java源码编译成JVM能执行的字节码就可以了,这样就实现了一套代码运行在不同的操作系统中。

image.png

1.1.3 前端主流跨平台方案

1 Hybird App

Hybird App的主要原理就是通过H5技术来实现一个网页,再通过原生的网页加载控件WebView来加载。WebView是一个基于webkit的引擎,可以解析DOM元素,展示HTML页面的控件,他的原理和浏览器展示页面的原理是相同的,实质上就是一个浏览器的内核。对于访问系统文件、蓝牙等,都需要原生开发,然后暴露给WebView供Java调用,调用的方式称为WebView Java Bridge,简称:JsBridge。

image.png

基于WebView的框架有点很明显,他们几乎可以完全继承现代Web开发的所有成果,包括丰富的控件库、满足各种需求的页面框架、完全的动态化、自动化测试工具等等,而且Web开发人员,不需要太多的学习成本就可以开发 一个App。但是WebView框架有一个致命的缺点,那就是WebView的渲染效率和Javascript执行性能太差。

2 ReactNative

随着移动设备的普及和使用范围越来越广,App的性能已经越来越重要了,为了解决WebView性能差的问题,以ReactNative为代表的一类框架出现了,它将最终渲染工作还给了系统,虽然同样使用HTML+CSS+JS的UI构建逻辑,但是最终会生成对应的原生控件,以充分利用原生控件较高的绘制效率。

image.png

ReactNative中所有的标签都不是真实控件,JS代码中的DOM布局会被解析成原生平台的控件,如标签对应ViewGroup/UIView,对应ScrollView/UIScrollView,对应ImageView/UIImageView,最终堆叠出一系列原生控件进行渲染。这种策略同时也将框架本身绑在了系统的控件上,抛开框架本身需要处理大量平台相关的逻辑,随着系统版本和API的变化,开发者也要不断的处理不同平台的差异。这种设计思想在后来阿里开源的Weex框架中也有所体现,但是Weex使用了V8引擎和Vue的设计理念。

3 Unity、Qt for mobile

Unity就是大家熟知的3D游戏引擎,Unity和Qt都采用了自绘UI+原生的跨平台技术,这种技术的思路是通过在不同平台实现一个统一的渲染引擎接口来绘制UI,而不依赖系统的原生控件,这样就做到了不同平台UI的一致性,也不会依赖原生控件,受原生系统的限制。而且,因为自绘引擎是直接调用系统API来绘制UI,所以性能非常好,和原生控件接近。 Flutter就属于这一类跨平台技术,即从头到尾写一套跨平台的UI框架,包括UI控件、渲染逻辑甚至开发语言。渲染引擎依靠跨平台的Skia图形库来实现,可以最大程度上保证不同平台、不同设备的体验一致性,逻辑处理使用支持AOT的Dart语言,执行效率比Javascript高的多。

1.1.4 跨平台通用原理

通过上述跨平台方案的了解,我们不难发现,其实跨平台的解决方案都很类似,都是提供一个容器,容器屏蔽底层差异,并向上层提供统一的API。跨平台分为渲染跨平台和逻辑跨平台,Weex、ReactNative的渲染是通过Virtual Dom,生成对应的原生控件,再由Android、iOS各自渲染。逻辑部分则使用了JS引擎,执行代码。

不论具体实现怎样,思路都大同小异。跨平台需要实现一个渲染引擎、一个VM虚拟机,再基于这套架构实现各种组件和API。Flutter就是使用了Skia渲染引擎和Dart VM,实现了跨平台。

image.png

图1.7 跨平台方案

1.2Flutter简介

1.2.1 什么是Flutter

Flutter是Google在2015年开源的一套全新的跨平台UI框架,作为新一代的跨平台方案,Flutter能帮助开发者通过一套代码库高效构建多平台的精美应用,支持移动、Web、桌面和嵌入式平台,而且Flutter是未来新操作系统Fuchsia的默认开发软件。它的优势主要包括:

  • 跨平台开发。真正做了一套代码可以同时在Android和iOS等平台上运行,避免了过高的维护成本,提高了开发效率。

  • 性能优越。通过“自绘UI+原生系统”实现高帧率的流畅UI,性能媲美原生。渲染引擎依靠跨平台的Skia图形库来实现,直接调用系统的图形绘制相关接口,可以在最大程度上保证不同平台、不同设备的体验一致性。逻辑处理使用了支持AOT的Dart语言,同时执行效率也比Javascript高得多。

  • 热重载快速开发。Flutter选用了Google于2011年推出的Dart作为其开发语言。Dart既支持是AOT(Ahead Of Time)编译,也支持是JIT(Just In Time)编译。AOT指运行前编译,JIT是即时编译,边运行边编译。两种编译方式的主要区别在于是否在运行时进行编译。AOT保证了运行期间页面加载的高性能,JIT保证了Flutter在开发阶段可以达到亚秒级热重载,而且不会丢失状态,从而大大提升了开发效率。

1.2.2 Flutter的优缺点

现在,我们来对前端跨平台方案做一个对比,大家可以结合自身的业务情况选择合适的跨平台方案: 跨平台方案:

方案原生Web容器React NativeFlutter
静态性能适中
滚动性能适中
区块级支持一般一般
适用场景对性能和稳定性要求极高,希望提供在各平台上交互和体验极致的核心功能和页面对性能要求不高,重展示弱交互,更关注迭代和交付效率的页面希望能平衡性能、展示交互能力和迭代交付效率,对包大小敏感的页面对性能、稳定性和多端体验一致性要求高、对动态化及包大小无特殊要求,更关注提升原生迭代效率的中等核心页面
跨平台
开发效率
体验一致性一般
页面级支持很高
动态化支持
布局能力一般
生态
维护成本适中
包大小
App Store政策风险
独立App开发支持一般
1.3Flutter架构设计

接下来,我们来了解一下Flutter的架构设计,Flutter架构是非常清晰的,总共分为三层:

  • 框架层(Framework):Flutter框架层是纯Dart语言实现的一个响应式框架,开发者平常需要通过该层和Flutter系统进行交互。

  • 引擎层(Engine):引擎层提供了一系列Flutter核心API的底层实现,例如图形(通过Skia),文字布局,文件等,是连接框架和系统(Andoird/iOS)的桥梁。引擎层绝大部分是用C++实现的,其为Flutter系统的核心。

  • 嵌入层(Embedder):嵌入层为Flutter提供了一个入口,通过该入口访问底层系统提供的服务,例如输入法、绘制Surface等。嵌入层基本是由平台对应的语言实现的,例如在Android上是由Java和C++实现,在iOS上是由Objective-C实现。

image.png

1.3.1 Framework框架

Flutter框架最顶层是我们经常用到的Framework框架层,整个框架层都是由Dart语言实现,该层提供一套基础库,用于处理动画、绘图和手势等,并且封装了一套UI组件库Widgets。在基础组件库上,Flutter还提供了Material和Cupertino两种视觉风格的组件库。通常情况下,你会发现你仅仅使用这两层中的Widgets就够用了,并且目前你所能看到或者使用的,也基本都来自这两层,比如页面脚手架组件Scaffold和FloatingActionButton来自Meterial层,Column和GestureDetector来自Widgets层。

Widgets层给我们提供了可以直接使用的UI组件,主要分为三类:

  • 布局相关(layout):比如Column和Row,这两个Widget可以帮我们实现水平或者垂直排列组件。

  • 绘制相关(Parning):比如Text和Image,这类的组件可以帮我们展示或者渲染一些内容到屏幕上。

  • 手势检测相关:比如GestureDetector组件,这个组件能够检测到屏幕点击或者滑动事件。

在Widgets层下面是Rendering层,Rendering层简化了布局和绘制过程。在运行时Rendering层会构建一个Widget树,当页面发生变化时,会根据一定的算法计算出有变化的部分,然后更新Widget树。

dart:ui是Framework框架的最底层,它负责处理与Flutter Engine的通信。Flutter框架基于这一层来构建应用程序,比如输入驱动、绘制文字、布局和渲染系统等。所以我们可以仅仅使用dart:ui库中的类(例如Canvas、Paint和TextField)来构建一个Flutter App。但是这种方法需要精确的计算出你布局中所有元素的坐标,需要绘制到屏幕,同时对外界的输入时间能够捕获并作出响应。

1.3.2 Engine引擎层

Flutter Engine层是纯C++实现的,其中包括了Skia引擎、Dart运行时、文字排版引擎等。

  • 渲染引擎Skia:Skia(全称Skia Graphics Library(SGL))是一个由C++编写的二维开源图形库,Skia就是Flutter向GPU提供数据的途径。谷歌Chrome浏览器、Chrome OS、安卓、火狐浏览器、火狐操作系统以及其它许多产品都使用它作为图形引擎。底层渲染能力统一了,上层开发接口和功能体验也就随即统一了,因此,Skia保证了同一套代码调用在Android和iOS平台上的渲染效果是完全一致的。

  • Dart:主要包括:Dart Runtime,Garbage Collection(GC),如果是Debug模式的话,还包括JIT(Just In Time)支持。Release和Profile模式下,是AOT(Ahead Of Time)编译成了原生的arm代码,并不存在JIT部分。

  • Text:即文本渲染,其渲染层次如下:衍生自Minikin的libtxt库(用于字体选择,分隔行);HartBuzz用于字形选择和成型;Skia作为渲染/GPU后端,在Android和Fuchsia上使用FreeType渲染,在iOS上使用CoreGraphics来渲染字体。

1.3.3 Embedding嵌入层

Embedding嵌入层是用于呈现所有Flutter内容的原生系统应用,充当着宿主操作系统和Flutter之间粘合剂的角色。当启动一个Flutter应用时,嵌入层会提供一个入口,初始化Flutter引擎,获取UI和栅格化线程,创建Flutter可以写入的纹理。嵌入层同时负责管理应用的生命周期、输入操作、窗口的大小变化、线程管理和平台消息的传递。Flutter拥有Android、iOS、Windows、macOS和Linux的平台嵌入层,每一个平台都有各自的一套API。

  • 在iOS和macOS上,Flutter分别通过UIViewController和NSViewController载入嵌入层。嵌入层会创建一个FlutterEngine,作为Dart VM和Flutter运行时的宿主,还有一个FlutterViewController,关联对应的FlutterEngine,传递UIKit或者Cocoa的输入事件到 Flutter,并将FlutterEngine渲染的帧内容通过Metal或OpenGL进行展示。

  • 在Android上,Flutter默认通过一个Activity加载嵌入层,此时Flutter视图是一个FlutterView,以视图模式或纹理模式呈现Flutter的内容。

  • 在Windows上,Flutter的宿主是一个传统的Win32应用,内容通过ANGLE进行渲染。


版权声明:本文为weixin_55362248原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明。
原文链接:https://blog.csdn.net/weixin_55362248/article/details/122544436
推荐阅读
  • 本文详细解析了使用C++实现的键盘输入记录程序的源代码,该程序在Windows应用程序开发中具有很高的实用价值。键盘记录功能不仅在远程控制软件中广泛应用,还为开发者提供了强大的调试和监控工具。通过具体实例,本文深入探讨了C++键盘记录程序的设计与实现,适合需要相关技术的开发者参考。 ... [详细]
  • Java并发机制详解及其在数据安全性保障中的应用方案 ... [详细]
  • Web开发框架概览:Java与JavaScript技术及框架综述
    Web开发涉及服务器端和客户端的协同工作。在服务器端,Java是一种优秀的编程语言,适用于构建各种功能模块,如通过Servlet实现特定服务。客户端则主要依赖HTML进行内容展示,同时借助JavaScript增强交互性和动态效果。此外,现代Web开发还广泛使用各种框架和库,如Spring Boot、React和Vue.js,以提高开发效率和应用性能。 ... [详细]
  • 如何撰写适应变化的高效代码:策略与实践
    编写高质量且适应变化的代码是每位程序员的追求。优质代码的关键在于其可维护性和可扩展性。本文将从面向对象编程的角度出发,探讨实现这一目标的具体策略与实践方法,帮助开发者提升代码效率和灵活性。 ... [详细]
  • 如何撰写初级和高级前端开发者的专业简历
    如何撰写初级和高级前端开发者的专业简历 ... [详细]
  • 初探性能优化:入门指南与实践技巧
    在编程领域,常有“尚未精通编码便急于优化”的声音。为了从性能优化的角度提升代码质量,本文将带领读者初步探索性能优化的基本概念与实践技巧。即使程序看似运行良好,数据处理效率仍有待提高,通过系统学习性能优化,能够帮助开发者编写更加高效、稳定的代码。文章不仅介绍了性能优化的基础知识,还提供了实用的调优方法和工具,帮助读者在实际项目中应用这些技术。 ... [详细]
  • iOS 设备唯一标识获取的高效解决方案与实践
    在iOS 7中,苹果公司再次禁止了对MAC地址的访问,使得开发者无法直接获取设备的物理地址。为了在开发过程中实现设备的唯一标识,苹果推荐使用Keychain服务来存储和管理唯一的标识符。此外,还可以结合其他技术手段,如UUID和广告标识符(IDFA),以确保设备的唯一性和安全性。这些方法不仅能够满足应用的需求,还能保护用户的隐私。 ... [详细]
  • JavaScript XML操作实用工具类:XmlUtilsJS技巧与应用 ... [详细]
  • 七款高效编辑器与笔记工具推荐:KindEditor自动换行功能解析
    本文推荐了七款高效的编辑器与笔记工具,并详细解析了KindEditor的自动换行功能。其中,轻笔记QingBiJi是一款完全免费的记事本软件,用户可以通过其简洁的界面和强大的功能轻松记录和管理日常事务。此外,该软件还支持多平台同步,确保用户在不同设备间无缝切换。 ... [详细]
  • 在 Android 开发中,`android:exported` 属性用于控制组件(如 Activity、Service、BroadcastReceiver 和 ContentProvider)是否可以被其他应用组件访问或与其交互。若将此属性设为 `true`,则允许外部应用调用或与之交互;反之,若设为 `false`,则仅限于同一应用内的组件进行访问。这一属性对于确保应用的安全性和隐私保护至关重要。 ... [详细]
  • 在Conda环境中高效配置并安装PyTorch和TensorFlow GPU版的方法如下:首先,创建一个新的Conda环境以避免与基础环境发生冲突,例如使用 `conda create -n pytorch_gpu python=3.7` 命令。接着,激活该环境,确保所有依赖项都正确安装。此外,建议在安装过程中指定CUDA版本,以确保与GPU兼容性。通过这些步骤,可以确保PyTorch和TensorFlow GPU版的顺利安装和运行。 ... [详细]
  • 帝国CMS中的信息归档功能详解及其重要性
    本文详细解析了帝国CMS中的信息归档功能,并探讨了其在内容管理中的重要性。通过归档功能,用户可以有效地管理和组织大量内容,提高网站的运行效率和用户体验。此外,文章还介绍了如何利用该功能进行数据备份和恢复,确保网站数据的安全性和完整性。 ... [详细]
  • 在Cisco IOS XR系统中,存在提供服务的服务器和使用这些服务的客户端。本文深入探讨了进程与线程状态转换机制,分析了其在系统性能优化中的关键作用,并提出了改进措施,以提高系统的响应速度和资源利用率。通过详细研究状态转换的各个环节,本文为开发人员和系统管理员提供了实用的指导,旨在提升整体系统效率和稳定性。 ... [详细]
  • 深入解析Linux内核中的进程上下文切换机制
    在现代操作系统中,进程作为核心概念之一,负责管理和分配系统资源,如CPU和内存。深入了解Linux内核中的进程上下文切换机制,需要首先明确进程与程序的区别。进程是一个动态的执行流,而程序则是静态的数据和指令集合。进程上下文切换涉及保存当前进程的状态信息,并加载下一个进程的状态,以实现多任务处理。这一过程不仅影响系统的性能,还关系到资源的有效利用。通过分析Linux内核中的具体实现,可以更好地理解其背后的原理和技术细节。 ... [详细]
  • 成都服务器租赁适用于哪些网站业务部署——Vecloud专业解析
    成都,作为四川省的省会,不仅是西南地区唯一的副省级城市,也是国家重要的高新技术产业基地和商贸物流中心。Vecloud专业解析指出,成都服务器租赁服务特别适合各类网站业务的部署,尤其是需要高效、稳定和安全的在线应用。无论是电子商务平台、内容管理系统还是大数据分析,成都的服务器租赁都能提供强大的支持,满足不同企业的需求。 ... [详细]
author-avatar
张诣轩压_143
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有