开通VIP,畅享免费电子书等14项超值服
首页
好书
留言交流
下载APP
联系客服
2021.11.12
另外electron-react还可作为使用Electron+React+Mobx+Webpack技术栈的脚手架工程。
桌面应用程序,又称为GUI程序(GraphicalUserInterface),但是和GUI程序也有一些区别。桌面应用程序将GUI程序从GUI具体为“桌面”,使冷冰冰的像块木头一样的电脑概念更具有人性化,更生动和富有活力。
我们电脑上使用的各种客户端程序都属于桌面应用程序,近年来WEB和移动端的兴起让桌面程序渐渐暗淡,但是在某些日常功能或者行业应用中桌面应用程序仍然是必不可少的。
传统的桌面应用开发方式,一般是下面两种:
直接将语言编译成可执行文件,直接调用系统API,完成UI绘制等。这类开发技术,有着较高的运行效率,但一般来说,开发速度较慢,技术要求较高,例如:
一开始就有本地开发和UI开发。一次编译后,得到中间文件,通过平台或虚机完成二次加载编译或解释运行。运行效率低于原生编译,但平台优化后,其效率也是比较可观的。就开发速度方面,比原生编译技术要快一些。例如:
不过,上面两种对前端开发人员太不友好了,基本是前端人员不会设计的领域,但是在这个【大前端】的时代,前端开发者正在想方设法涉足各个领域,使用WEB技术开发客户端的方式横空出世。
使用WEB技术进行开发,利用浏览器引擎完成UI渲染,利用Node.js实现服务器端JS编程并可以调用系统API,可以把它想像成一个套了一个客户端外壳的WEB应用。
在界面上,WEB的强大生态为UI带来了无限可能,并且开发、维护成本相对较低,有WEB开发经验的前端开发者很容易上手进行开发。
本文就来着重介绍使用WEB技术开发客户端程序的技术之一【electron】
Electron是由Github开发,用HTML,CSS和JavaScript来构建跨平台桌面应用程序的一个开源库。Electron通过将Chromium和Node.js合并到同一个运行时环境中,并将其打包为Mac,Windows和Linux系统下的应用来实现这一目的。
当然,我们也要认清它的缺点:性能比原生桌面应用要低,最终打包后的安装包和其他文件都比较大。
兼容性
虽然你还在用WEB技术进行开发,但是你不用再考虑兼容性问题了,你只需要关心你当前使用Electron的版本对应Chrome的版本,一般情况下它已经足够新来让你使用最新的API和语法了,你还可以手动升级Chrome版本。同样的,你也不用考虑不同浏览器带了的样式和代码兼容问题。
Node环境
这可能是很多前端开发者曾经梦想过的功能,在WEB界面中使用Node.js提供的强大API,这意味着你在WEB页面直接可以操作文件,调用系统API,甚至操作数据库。当然,除了完整的NodeAPI,你还可以使用额外的几十万个npm模块。
跨域
你可以直接使用Node提供的request模块进行网络请求,这意味着你无需再被跨域所困扰。
强大的扩展性
借助node-ffi,为应用程序提供强大的扩展性(后面的章节会详细介绍)。
现在市面上已经有非常多的应用在使用electron进行开发了,包括我们熟悉的VSCode客户端、GitHub客户端、Atom客户端等等。印象很深的,去年迅雷在发布迅雷X10.1时的文案:
从迅雷X10.1版本开始,我们采用Electron软件框架完全重写了迅雷主界面。使用新框架的迅雷X可以完美支持2K、4K等高清显示屏,界面中的文字渲染也更加清晰锐利。从技术层面来说,新框架的界面绘制、事件处理等方面比老框架更加灵活高效,因此界面的流畅度也显著优于老框架的迅雷。至于具体提升有多大?您一试便知。
你可以打开VSCode,点击【帮助】【切换开发人员工具】来VSCode客户端的界面。
Electron结合了Chromium、Node.js和用于调用操作系统本地功能的API。
Chromium是Google为发展Chrome浏览器而启动的开源项目,Chromium相当于Chrome的工程版或称实验版,新功能会率先在Chromium上实现,待验证后才会应用在Chrome上,故Chrome的功能会相对落后但较稳定。
Chromium为Electron提供强大的UI能力,可以在不考虑兼容性的情况下开发界面。
Node.js是一个让JavaScript运行在服务端的开发平台,Node使用事件驱动,非阻塞I/O模型而得以轻量和高效。
单单靠Chromium是不能具备直接操作原生GUI能力的,Electron内集成了Nodejs,这让其在开发界面的同时也有了操作系统底层API的能力,Nodejs中常用的Path、fs、Crypto等模块在Electron可以直接使用。
为了提供原生系统的GUI支持,Electron内置了原生应用程序接口,对调用一些系统功能,如调用系统通知、打开系统文件夹提供支持。
在开发模式上,Electron在调用系统API和绘制界面上是分离开发的,下面我们来看看Electron关于进程如何划分。
Electron区分了两种进程:主进程和渲染进程,两者各自负责自己的职能。
Electron运行package.json的main脚本的进程被称为主进程。一个Electron应用总是有且只有一个主进程。
职责:
可调用的API:
由于Electron使用了Chromium来展示web页面,所以Chromium的多进程架构也被使用到。每个Electron中的web页面运行在它自己的渲染进程中。
主进程使用BrowserWindow实例创建页面。每个BrowserWindow实例都在自己的渲染进程里运行页面。当一个BrowserWindow实例被销毁后,相应的渲染进程也会被终止。
你可以把渲染进程想像成一个浏览器窗口,它能存在多个并且相互独立,不过和浏览器不同的是,它能调用NodeAPI。
在上面的章节我们提到,渲染进和主进程分别可调用的ElectronAPI。所有Electron的API都被指派给一种进程类型。许多API只能被用于主进程中,有些API又只能被用于渲染进程,又有一些主进程和渲染进程中都可以使用。
你可以通过如下方式获取ElectronAPI
在后面的章节我们会选择其中常用的模块进行详细介绍。
你可以同时在Electron的主进程和渲染进程使用Node.jsAPI,)所有在Node.js可以使用的API,在Electron中同样可以使用。
import{shell}from"electron";importosfrom"os";document.getElementById("btn").addEventListener("click",()=>{shell.showItemInFolder(os.homedir());});有一个非常重要的提示:原生Node.js模块(即指,需要编译源码过后才能被使用的模块)需要在编译后才能和Electron一起使用。
主进程和渲染进程虽然拥有不同的职责,然是他们也需要相互协作,互相通讯。
ipcRenderer是一个EventEmitter的实例。你可以使用它提供的一些方法从渲染进程发送同步或异步的消息到主进程。也可以接收主进程回复的消息。
在渲染进程引入ipcRenderer:
import{ipcRenderer}from"electron";异步发送:
通过channel发送同步消息到主进程,可以携带任意参数。
在内部,参数会被序列化为JSON,因此参数对象上的函数和原型链不会被发送。
ipcRenderer.send("sync-render","我是来自渲染进程的异步消息");同步发送:
constmsg=ipcRenderer.sendSync("async-render","我是来自渲染进程的同步消息");注意:发送同步消息将会阻塞整个渲染进程,直到收到主进程的响应。
主进程监听消息:
ipcMain模块是EventEmitter类的一个实例。当在主进程中使用时,它处理从渲染器进程(网页)发送出来的异步和同步信息。从渲染器进程发送的消息将被发送到该模块。
ipcMain.on:监听channel,当接收到新的消息时listener会以listener(event,args...)的形式被调用。
在主进程中可以通过BrowserWindow的webContents向渲染进程发送消息,所以,在发送消息前你必须先找到对应渲染进程的BrowserWindow对象。:
ipcMain.on("sync-render",(event,data)=>{console.log(data);event.sender.send("main-msg","主进程收到了渲染进程的【异步】消息!");});渲染进程监听:
ipcRenderer.on:监听channel,当新消息到达,将通过listener(event,args...)调用listener。
ipcRenderer.on("main-msg",(event,msg)=>{console.log(msg);});4.6通信原理ipcMain和ipcRenderer都是EventEmitter类的一个实例。EventEmitter类是NodeJS事件的基础,它由NodeJS中的events模块导出。
EventEmitter的核心就是事件触发与事件监听器功能的封装。它实现了事件模型需要的接口,包括addListener,removeListener,emit及其它工具方法.同原生JavaScript事件类似,采用了发布/订阅(观察者)的方式,使用内部_events列表来记录注册的事件处理器。
remote模块为渲染进程(web页面)和主进程通信(IPC)提供了一种简单方法。使用remote模块,你可以调用main进程对象的方法,而不必显式发送进程间消息,类似于Java的RMI。
import{remote}from"electron";remote.dialog.showErrorBox("主进程才有的dialog模块","我是使用remote调用的");但实际上,我们在调用远程对象的方法、函数或者通过远程构造函数创建一个新的对象,实际上都是在发送一个同步的进程间消息。
在上面通过remote模块调用dialog的例子里。我们在渲染进程中创建的dialog对象其实并不在我们的渲染进程中,它只是让主进程创建了一个dialog对象,并返回了这个相对应的远程对象给了渲染进程。
Electron并没有提供渲染进程之间相互通信的方式,我们可以在主进程中建立一个消息中转站。
渲染进程之间通信首先发送消息到主进程,主进程的中转站接受到消息后根据条件进行分发。
在两个渲染进程间共享数据最简单的方法是使用浏览器中已经实现的HTML5API。其中比较好的方案是用StorageAPI,localStorage,sessionStorage或者IndexedDB。
就像在浏览器中使用一样,这种存储相当于在应用程序中永久存储了一部分数据。有时你并不需要这样的存储,只需要在当前应用程序的生命周期内进行一些数据的共享。这时你可以用Electron内的IPC机制实现。
将数据存在主进程的某个全局变量中,然后在多个渲染进程中使用remote模块来访问它。
在主进程中初始化全局变量:
import{ipcRenderer,remote}from"electron";const{getGlobal}=remote;constmainId=getGlobal("mainId");constdirname=getGlobal("__dirname");constdeviecMac=getGlobal("device").mac;在渲染进程中改变:
getGlobal("myField").name="code秘密花园";多个渲染进程共享同一个主进程的全局变量,这样即可达到渲染进程数据共享和传递的效果。