此文是我的出版书籍《React Native 精解与实战》连载分享,此书由机械工业出版社出版,书中详解了 React Native 框架底层原理、React Native 组件布局、组件与 API 的介绍与代码实战,以及 React Native 与 iOS、Android 平台的混合开发底层原理讲解与代码实战演示,精选了大量实例代码,方便读者快速学习。

书籍还配套了视频教程「80 节实战课精通 React Native 开发」,此视频课程建议配合书籍学习,书籍中原理性的东西讲解的比较清晰,而视频教程对于组件、API 等部分的代码实战开发讲解比较直观。

书籍相关所有资料请访问:http://rn.parryqiu.com


本章将深入讲解 React Native 的底层原理,万丈高楼平地起,非常深入地理解 React Native 底层的实现,在你开发或遇到难题调试时非常有帮助。

此部分包含 React Native 的框架构成、工作原理、UI 层的渲染与重绘以及组件间通信、React Native 与各个平台的通信实现以及 React Native 中的生命周期。

如果需要直接开始 React Native 的开发与实战,请直接跳至第四章开始学习。

3.1 React Native 框架构成

React Native 框架内部已提供了很多的内置组件,如图 3-1 所示。如 View、Text 等基本组件,用于一些功能布局的 Button、Picker 等,用于列表展示的各种 List 组件和对应 iOS 平台与 Android 平台的特定组件、API 等。同时也提供了供编写与原生平台交互的接口,在后续的章节我们会进行与原生平台的混合实战开发实战。



图 3-1 React Native 框架构成

3.2 React Native 工作原理

在 React 框架介绍的章节,我们理解了如何将代码渲染至虚拟 DOM 并更新到真实 DOM 的过程。在 React Native 框架中,渲染到 iOS 平台与 Android 平台的过程如图 3-2 所示。



图 3-2 React Native 渲染

在 React 框架中,JSX 源码通过 React 框架最终渲染到了浏览器的真实 DOM 中,而在 React Native 框架中,JSX 源码通过 React Native 框架编译后,通过对应平台的 Bridge 实现了与原生框架的通信。如果我们在程序中调用了 React Native 提供的 API,那么 React Native 框架就通过 Bridge 调用原生框架中的方法。

因为 React Native 的底层为 React 框架,所以如果是 UI 层的变更,那么就映射为虚拟 DOM 后进行 diff 算法,diff 算法计算出变动后的 JSON 映射文件,最终由 Native 层将此 JSON 文件映射渲染到原生 App 的页面元素上,最终实现了在项目中只需要控制 state 以及 props 的变更来引起 iOS 与 Android 平台的 UI 变更。

编写的 React Native代码最终会打包生成一个 main.bundle.js 文件供 App 加载,此文件可以在 App 设备本地,也可以存放于服务器上供 App 下载更新,后续章节讲解的热更新就会涉及到 main.bundle.js 位置的设置问题。

3.3 React Native 与原生平台通信

在与原生框架通信中,如图 3-3 所示,React Native 采用了 JavaScriptCore 作为 JS VM,中间通过 JSON 文件与 Bridge 进行通信。而如果在使用 Chrome 浏览器进行调试时,那么所有的 JavaScript 代码都将运行在 Chrome 的 V8 引擎中,与原生代码通过 WebSocket 进行通信。



图 3-3 React Native 与原生平台的通信

关于 React Native 框架与原生平台的通信原理的详细介绍,后续的混合开发章节将会有详细的讲解与实战开发。

3.4 组件间通信

React Native 开发最基本的元素就是组件,React Native 与 React 一样,也会涉及到组件之间的通信,用于数据在组件之间的传递,下面列出了几种常用的组件间通信的方式。

父子组件的通信

如同之前的章节介绍 React 组件间传递参数一样,在 React Native 中,父组件向子组件传递值,可以通过 props 的形式。

在下例中,父组件通过调用子组件并赋值子组件的 name 为 React,子组件通过 this.props.name 获取父组件传递过来的 name 的字符串值 React。

完整代码在本书配套源码的 03-04 文件夹。

/**
* 章节: 03-04
* 父子组件通信,在父组件中调用子组件
* FilePath: /03-04/parent-2-child.js
* @Parry
*/ <ChildComponent name='React'/> /**
* 章节: 03-04
* 子组件实现,通过 props 获取父页面传递的值
* FilePath: /03-04/parent-2-child.js
* @Parry
*/ class ChildComponent extends Component {
render() {
return (
<Text>Hello {this.props.name}!</Text>
);
}
}

子父组件的通信

在开发过程中,不仅有父子之间的通信,有时还会有子组件向父组件通信传递值的需求,比如当子组件的某个值变更后,需要通知到父组件做相应的变更与响应,那么就会需要子父组件之间的通信。

示例代码如下,在父组件的定义中,在调用子组件时,同样向子组件传递了一个参数,不过这个参数是一个函数,此函数用于接收后续子组件向父组件传递过来的数据,与之前父组件向子组件传递数据不太一样。

完整代码在本书配套源码的 03-04 文件夹。

/**
* 章节: 03-04
* 子父组件通信,父组件的实现
* FilePath: /03-04/child-2-parent.js
* @Parry
*/
import React, {Component} from 'react';
import ChildComponent from './ChildComponent' class App extends Component {
constructor(props) {
super(props)
this.state = {
name: 'React'
}
} //传递到子组件的参数,不过参数是一个函数。
handleChangeName(nickName) {
this.setState({name: nickName})
} render() {
return (
<div>
<p>父组件的 name:{this.state.name}</p>
<ChildComponent
onChange={(val) => {
this.handleChangeName(val)
}}/>
</div>
);
}
} export default App;

下面为子组件的定义,子组件在页面中定义了一个按钮,点击此按钮后,调用自身的一个函数 handleChange,修改了自身 state 中的值 name 为 nickName 定义的值 Parry,那么此子组件的页面上的字符串将由之前的 Hello React! 变为 Hello Parry!,同时使用了 this.props.changeName,也就是父组件调用时传递过来的函数,向父组件传递了 nickName 的值 Parry。

父组件在接收到子组件的调用后,调用了父组件自身的函数 handleChangeName 修改了自身的 state 中的 name 的值为 Parry,也就是子组件传递过来的 Parry,所以同时,父组件的页面上的值也同时由之前的 React 变更成了 Parry。

/**
* 章节: 03-04
* 子父组件通信,子组件的实现
* FilePath: /03-04/child-2-parent.js
* @Parry
*/ import React, {Component} from 'react' export default class ChildComponent extends Component {
constructor(props) {
super(props) this.state = {
name: 'React'
}
} handleChange() {
const nickName = 'Parry';
this.setState({name: nickName})
//调用父组件传递过来的函数参数,传递值到父组件去。
this
.props
.changeName(nickName)
} render() {
const {name} = this.state;
return (
<div>
<p>Hello {name}!</p>
<Button
onPress={this
.handleChange
.bind(this)}
title="修改一下 name 为 Parry"/>
</div>
)
}
}

多级组件之间的通信

如果组件之间的父子层级非常多,需要进行组件之间的传递,这时候当然可以通过上面介绍的方法进行一级一级的传递,但是当这种组件间层级很深的时候,这样的传递方法不是一个太好的方法。

解决的方法是首先要在设计 App 时,需要注意不能让组件之间的层级关系太深,一是为了避免组件之间通信的冗长,还有一个原因是太深的嵌套逻辑,用户体验上也不会很好,可以想象一下用户从最底层一层层操作返回到最顶层时的体验。

第二就是可以使用如 context 对象或 global 等方式进行多级组件间的通信,但是这种方式不推荐。

无直接关系组件间通信

前面提到的都是有层级关系的组件间的通信方式,而如果组件间没有层级的关系的话,可以通过如 AsyncStorage 或 JSON 文件等方式进行无直接关系组件间的通信。

当然,还可以使用 EventEmitter / EventTarget / EventDispatcher继承或实现接口的方式、Signals 模式或 Publish / Subscribe 的广播形式,都可以达到无直接关系组件间的通信。

这些组件间的通信方式使得组件之间的数据可以传递起来,后续的实战章节会有详细的代码实现,这里主要进行了理论部分的介绍。掌握这部分知识后才可以将 App 开发中的基本单位,也就是组件串联起来。


《React Native 精解与实战》书籍连载「React Native 底层原理」的更多相关文章

  1. 《React Native 精解与实战》书籍连载「Android 平台与 React Native 混合开发」

    此文是我的出版书籍<React Native 精解与实战>连载分享,此书由机械工业出版社出版,书中详解了 React Native 框架底层原理.React Native 组件布局.组件与 ...

  2. 《React Native 精解与实战》书籍连载「iOS 平台与 React Native 混合开发」

    此文是我的出版书籍<React Native 精解与实战>连载分享,此书由机械工业出版社出版,书中详解了 React Native 框架底层原理.React Native 组件布局.组件与 ...

  3. 《React Native 精解与实战》书籍连载「配置 iOS 与 Android 开发环境」

    此文是我的出版书籍<React Native 精解与实战>连载分享,此书由机械工业出版社出版,书中详解了 React Native 框架底层原理.React Native 组件布局.组件与 ...

  4. 《React Native 精解与实战》书籍连载「React Native 网络请求与列表绑定」

    此文是我的出版书籍<React Native 精解与实战>连载分享,此书由机械工业出版社出版,书中详解了 React Native 框架底层原理.React Native 组件布局.组件与 ...

  5. 《React Native 精解与实战》书籍连载「React Native 中的生命周期」

    此文是我的出版书籍<React Native 精解与实战>连载分享,此书由机械工业出版社出版,书中详解了 React Native 框架底层原理.React Native 组件布局.组件与 ...

  6. 《React Native 精解与实战》书籍连载「Node.js 简介与 React Native 开发环境配置」

    此文是我的出版书籍<React Native 精解与实战>连载分享,此书由机械工业出版社出版,书中详解了 React Native 框架底层原理.React Native 组件布局.组件与 ...

  7. 《React Native 精解与实战》书籍连载「React 与 React Native 简介」

    此文是我的出版书籍<React Native 精解与实战>连载分享,此书由机械工业出版社出版,书中详解了 React Native 框架底层原理.React Native 组件布局.组件与 ...

  8. 《React Native 精解与实战》书籍连载「React Native 源码学习方法及其他资源」

    此系列文章将整合我的 React 视频教程与 React Native 书籍中的精华部分,给大家介绍 React Native 源码学习方法及其他资源. 最后的章节给大家介绍 React Native ...

  9. [书籍精读]《React Native精解与实战》精读笔记分享

    写在前面 书籍介绍:本书由架构师撰写,包含ReactNative框架底层原理,以及与iOS.Android混合开发案例,精选了大量实例代码,方便读者快速学习.主要内容分为两大部分,第1部分" ...

随机推荐

  1. Centos7开启ssh免密码登录

    1.输入命令:cd .ssh进入rsa公钥私钥目录(清空旧秘钥) 2.在当前目录下执行ssh-keygen -t rsa,三次回车后生成新的公钥(id_rsa.pub)私钥(id_rsa)文件(每个节 ...

  2. JAVA多线程的问题以及处理(一)【转】

    多线程编程为程序开发带来了很多的方便,但是也带来了一些问题,这些问题是在程序开发过程中必须进行处理的问题. 这些问题的核心是,如果多个线程同时访问一个资源,例如变量.文件等,时如何保证访问安全的问题. ...

  3. python连接sqlserver数据库

    1.准备工作 python3.6连接sqlserver数据库需要引入pymssql模块 pymssql官方:https://pypi.org/project/pymssql/ 没有安装的话需要: pi ...

  4. 使用Navicat for Mysql连接mysql服务器

    使用Navicat for Mysql连接mysql服务器 在windows上用Navicat for Mysql 连接装在虚拟机Ubuntu上的mysql服务器时出现的问题的解决方案. Navica ...

  5. Vue 学习笔记之快速入门篇

    Vue (读音 /vjuː/,类似于 view) 是一套用于构建用户界面的渐进式框架.与其它大型框架不同的是,Vue 被设计为可以自底向上逐层应用.Vue 的核心库只关注视图层,不仅易于上手,还便于与 ...

  6. 基于centOS7:新手篇→tomcat的部署方式

    一.自动部署 将项目直接拷贝到webapps目录下,通过项目名直接访问 二.在server.xml中指定项目 打开Tomcat/conf/server.xml文件,在host标签中加入以下参数并重启T ...

  7. Java多线程 Socket使用

    点我跳过黑哥的卑鄙广告行为,进入正文. Java多线程系列更新中~ 正式篇: Java多线程(一) 什么是线程 Java多线程(二)关于多线程的CPU密集型和IO密集型这件事 Java多线程(三)如何 ...

  8. MySQL初识

    1.MySQL版本 社区版:免费的,功能够用. 商业版:更能更加强大,更加稳定,但是收费的. 2.每个版本都分四个版本发布 Alpha版本:一般只在开发公司内部使用,不对外公开,测试.自我检查版本: ...

  9. (转)Spring Boot 2 (十):Spring Boot 中的响应式编程和 WebFlux 入门

    http://www.ityouknow.com/springboot/2019/02/12/spring-boot-webflux.html Spring 5.0 中发布了重量级组件 Webflux ...

  10. [NOI2017]蔬菜

    [NOI2017]蔬菜 题目描述 大意就是有\(n\)种物品,第\(i\)个物品有\(c_i\)个,单价是\(a_i\).然后每天你可以卖出最多\(m\)个物品.每天结束后第\(i\)种物品会减少\( ...