从有前端到现在,JavaScript 语言一直都是实现前端逻辑的首选。但是,由于 JavaScript 是一个弱类型语言,很难进行相关的类型检测。因此在构建大型应用时,使用 JavaScript 难免会遇到一些隐式类型转换等相关的问题,从而导致程序的 bug。
在当前的选择中,有两个流派,都能够解决 JavaScript 弱类型语言带来的弊病,给前端带来强类型语言的支持。
作为最近被大家关注的越来越多的强类型语言,ReasonML 的发展也是需要我们持续关注的。
说了这么多背景,我们来正式介绍下 ReasonML 这门语言。首先,让我们来看下[官网][1]对于 ReasonML 的介绍。
Reason lets you write simple, fast and quality type safe code while leveraging both the JavaScript & OCaml ecosystems. Reason利用 JavaScript 和 OCaml 语言的生态,让你编写简单、快速和高质量类型安全的代码。
从这个介绍中我们可以知道, ReasonML 是从 OCaml 语言衍生出来的,可以支持 JavaScript 的新的强类型语言。首页介绍中,还提到了这个语言的三个特点:
听了这么多关于 ReasonML 的介绍,我们来简单的看下相关的语法。通过相关的语法和示例,我们能够帮助我们更好的理解这门语言。
我们就使用官方的一些简单的示例来快速入门这个语言。
因为目前浏览器无法直接识别强类型语言,因此我们需要通过编译器,将强类型语言编译成 JavaScript 以后才能够在前端浏览器或者 Node.js 中运行。
首先,我们来看下如何进行安装:
npm install -g bs-platform
首先,我们通过 NPM 来对编译平台 bs-platform
进行全局安装,安装完成后,我们就可以使用这个 cli 自带的命令了。
安装完成后,我们需要初始化一个项目,因此我们需要执行以下命令:
bsb -init my-new-project -theme basic-reason
通过这个命令,我们就创建了一个名字为 my-new-project
的项目文件了。
这个时候,我们进入这个项目文件夹中,看看这里面到底初始化了哪些东西。首先我们来看下 package.json
文件。
{
"name": "my-new-project",
"version": "0.1.0",
"scripts": {
"build": "bsb -make-world",
"start": "bsb -make-world -w",
"clean": "bsb -clean-world"
},
"keywords": [
"BuckleScript"
],
"author": "",
"license": "MIT",
"devDependencies": {
"bs-platform": "^4.0.18"
}
}
接下来,我们先来看下 src/Demo.re
文件的内容。
Js.log("Hello, BuckleScript and Reason!");
我们需要重点关注的就是,我们可以通过 npm run build
命令来编译整个项目,它会将 src/Demo.re
编译成 src/Demo.re.js
文件。让我们来看下编译出来的内容是什么样子的。
// Generated by BUCKLESCRIPT VERSION 4.0.18, PLEASE EDIT WITH CARE
'use strict';
console.log("Hello, BuckleScript and Reason!");
/* Not a pure module */
大家可以看到,我们通过 ReasonML 的编译器,将 ReasonML 的代码编译成了 JavaScript。
说完了构建编译相关的流程,我们来正式看下 ReasonML 这门语言的语法。
ReasonML 的类型系统可以自动进行类型推断,在本文介绍中我会尽可能详细的进行介绍,但是如果没有声明具体类型,大家可以自主进行推断。
我们可以通过下面这个表格来快速看下当前的数据结构:
数据类型 | 示例 |
---|---|
字符串 | "Hello" |
字符 | 'x' |
整型数字 | 23 , -23 |
浮点型数字 | 23.0 , -23.0 |
整型数字加法 | 23 + 1 |
浮点型数字加法 | 23.0 +. 1.0 |
整型数字除法/乘法 | 2 / 23 * 1 |
浮点型数字除法/乘法 | 2.0 /. 23.0 *. 1.0 |
浮点型数字求幂 | 2.0 ** 2.0 |
字符串组合 | "Hello " ++ "World" |
比较运算符 | > , < , >= , =< |
布尔运算符 | ! , && , || |
引用(浅)比较,结构(深)比较 | === , == |
不可变列表 | [1, 2, 3] |
不可变前置声明(Immutable Prepend) | [item1, item2, ...theRest] |
元组(Tuple) | [1, "string"] |
数组 | [|1, 2, 3|] |
记录(Records) | type player = {score: int}; {score: 100} |
对象 | type tesla = {var red = "red"; pub color = red;}; tesla#color |
注释 | /* Comment here */ |
这里面有一些内容需要详细介绍下差别。
字符与字符串。在 ReasonML 中,字符与字符串分别是用单引号和双引号来进行表示,而不是统一认为是字符串,单双引号通用。
浅比较和深比较。在 JavaScript 中,==
和 ===
对于对象和数组之类的变量来说,都是进行地址的比较。而在 ReasonML 中,我们可以在运算符中实现深比较。
不可变列表与数组。在 JavaScript 中,数组可以存储任意类型的内容。而在 ReasonML 中,出现了一个不可变列表,只能存储同一种数据类型(比如全部都是整型数字),并且是不可变数据类型。ReasonML 的数组是一个可变数据类型,但是仍然只能存储同一种数据类型。如果需要实现存储不同的数据类型,则需要使用元组(Tuple)——一个不可变的有序类型,具体代码如下:
let ageAndName = (24, "Lil' Reason");
对象与记录。在 ReasonML 中,出现了对象和记录两种相似的数据类型,我们来看下两者的区别。记录是一个需要提前声明的默认不可变的数据结构,在 ReasonML 中推荐使用。而在 ReasonML 的对象,则是一个不需要提前声明的数据结构。不过在 ReasonML 中,推荐优先使用记录。
关于语法相关的内容,我只是简单介绍了一下核心的数据结构,有很多内容没有介绍到,如果大家想要系统的学习 ReasonML 的话,可以看一下官方文档。
如果我们需要在 ReasonML 中使用 JavaScript 代码,我们可以按照如下的方法:
[%bs.raw {| console.log('here is some javascript for you') |}];
上面的代码经过编译后,可以得到如下的 JavaScript 代码。
'use strict';
console.log('here is some javascript for you');
这个方法与全局注入变量的方式类似,会直接将上述代码替换成编译后的 JavaScript 代码。因此我们可以这么用:
let x = [%bs.raw {| 'here is a string from javascript' |}];
得到的代码为:
var x = ( 'here is a string from javascript' );
许多的语法差异我们在上述语法介绍中都已经介绍过了,如果需要详细的比对,可以看官方文档中的语法比较。
ReasonML 是一门比 TypeScript 约束严格的多的强类型语言(TypeScript 编译报错可以选择忽略掉,不影响使用)。强类型语言对于大型的项目开发来说,确实可以带来明显的优势。但是,我们能不能够大规模使用 ReasonML 呢?
先说下个人的基本判断:持续关注,不建议在大型应用场景中使用。
从 ReasonML 目前的情况来看,它与 TypeScript 非常相似。
TypeScript 由于对 JavaScript 的生态完全兼容,所以即使我们需要进行部分代码的重写,我们仍然可以快速的复用 JavaScript 的强大生态。
而由于 ReasonML 来说,这个方面就会明显相差不少。与此同时,ReasonML 的相关语法与 JavaScript 相差较大,因此对于前端工程师的学习成本来说,也有一定的提提升。
综上所述,如果大家需要在前端使用强类型语言来构建大型项目,建议选择 TypeScript 语言。
如果在迁移 TypeScript 中有什么问题,可以看下我之前写的一篇文章——旧项目 TypeScript 改造问题与解决方案记。
黄珏,2015年毕业于华中科技大学,目前任职于美团基础研发平台大象业务部,独立负责大象 Web SDK 的开发与维护。
本文未经作者允许,禁止转载。
Copyright© 2013-2020
All Rights Reserved 京ICP备2023019179号-8