Skip to content

20如何将JavaScript应用切换至TypeScript?

在前面几讲,我们学习了如何从零开始新建 TypeScript Node.js、Web 项目。

然而,TypeScript 作为 JavaScript 的超集,花费了漫长的时间才替代了 JavaScript,并成长为构建应用的主流技术。比如我自己所负责的项目,最初也主要是基于 ES6(JavaScript) 源码构建的,甚至极个别项目是基于无须转译的 ES5(JavaScript) 代码构建的。

从 JavaScript 进化到 TypeScript,也就意味着需要大量的迁移、重构操作。因此,接下来我们将学习将 JavaScript 技术栈项目迁移到 TypeScript 的操作步骤和实用技巧。

迁移步骤

调整项目结构

首先,我们可以参照 18 讲和 19 讲的内容调整项目结构,比如使用 src 目录组织源码,typings 目录组织类型声明定义,lib 目录作为 Node.js 模块的构建产物,build 目录作为 Web 项目的构建产物。

然后,我们需要在项目根目录下创建一个 tsconfig.json,让源码和单测共享一个配置文件。

因为如今大多数的 JavaScript 项目都是基于 ES6+ 组织源码,再转译为 JavaScript,其项目结构基本可以划分为如下所示,所以我们只需要创建一个 tsconfig.json 即可。

java
JavaScript2TypeScriptProject
├── src
│   ├── a.js
│   └── b.js
├── build 或则 lib
├── typings
├── package.json
└── tsconfig.json

接下来就是如何配置 tsconfig.json 了,下面我们一起探讨一下。

配置 tsconfig

在正式讲解之前,我们先插播一道思考题:Node.js 项目需要如何配置?

提示信息:区别仅在于 Node.js 项目需要指定 rootDir、outDir。

以配置 React Web 项目为例,为了尽可能少改动源码、让项目正常运行起来,我们不要一步到位开启严格模式,而应该尽量宽松地配置 tsconfig,如下配置所示。

java
{
  "compilerOptions": {
    "target": "es5",
    "lib": [
      "dom",
      "dom.iterable",
      "esnext"
    ],
    "allowJs": true,
    "skipLibCheck": true,
    "esModuleInterop": true,
    "allowSyntheticDefaultImports": true,
    "forceConsistentCasingInFileNames": true,
    "module": "esnext",
    "moduleResolution": "node",
    "resolveJsonModule": true,
    "isolatedModules": true,
    "noEmit": true,
    "jsx": "react",
    "typeRoots": ["node_modules/@types", "./typings"]
  },
  "include": [ "src", "typings" ]
}

其中,比较重要的配置项分为如下 5 个。

  • 第 3 行配置"target"为 "es5",用来将 TypeScript 转译为低版本、各端兼容性较好的 ES5 代码。

  • 第 9 行开启的 allowJs,它允许 JavaScript 和 TypeScript 混用,这使得我们可以分批次、逐模块地迁移代码。

  • 第 20 行我们把 typings 目录添加到类型查找路径,让 TypeScript 可以查找到自定义类型声明,比如为缺少类型声明的第三方模块补齐类型声明。

  • 第 22 行我们把 src 和 typings 目录添加到 TypeScript 需要识别的文件中(也可以按照实际需要添加其他目录或者文件,比如说独立的单测文件目录 tests)。

  • 因为是 React Web 项目,所以我们还需要在第 19 行将"jsx"配置为"react"。

注意:因为 Web 项目中不会直接使用 tsc 转译 TypeScript,所以我们无需配置 rootDir、outDir,甚至可以开启 noEmit 配置(如上边配置第 18 行所示,开启该配置 tsc 不会生成转译产物)。

接下来,我们需要结合项目所使用的构建工具集成 TypeScript 构建环境。

构建工具集成 TypeScript

下面我们以非常常见的构建工具 Webpack 集成 TypeScript 为例。

首先我们需要安装如下所示依赖,比如所有用到的第三模块类型声明(通过"npm i -D @types/模块名"进行安装)以及需要用来加载并转译 TypeScript 代码的 Webpack Loader。

java
npm install -D typescript;
npm install -D @types/react;
npm install -D @types/react-dom;
... // 其他必要依赖
npm install -D ts-loader;

然后,我们选择 ts-loader 作为 TypeScript 加载器,并在 webpack.config.js 配置文件中添加 resolve 和 module 规则,如下配置所示:

javascript
module.exports = {
  // 其他配置 ...,
  resolve: {
    extensions: [".ts", ".tsx", ".js", ".jsx", ".json"],
  },
  module: {
    rules: [
      // 其他配置 loader 规则...,
      {
        test: /\.tsx?$/,
        use: [
          {
            loader: "ts-loader",
            options: { transpileOnly: true }
          }
        ]
      }
    ],
  },
  plugins: [
    // ...其他配置
    new require('fork-ts-checker-webpack-plugin')({
      async: false,
      tsconfig: '...' // tsconfig.json 文件地址
    });
  ]
  // 其他配置...
};

首先,我们在第 4 行的 extensions 配置中添加了 .ts、.tsx 文件后缀名,是为了让 Webpack 在解析模块的时候同时识别 TypeScript 文件。

注意:因为 Webpack 是以从左到右的顺序读取 extensions 配置并查找文件,所以按照如上配置,当碰到模块同名的情况,Webpack 将优先解析到 TypeScript 模块。

然后,我们在 17~19 行的 rules 配置中添加了 ts-loader,是为了让 Webpack 使用 ts-loader 加载和转译 .ts、.tsx 文件。

一个比较好的实践是,我们可以开启 ts-loader 的 transpileOnly 配置,让 ts-loader 在处理 TypeScript 文件时,只转译而不进行静态类型检测,这样就可以提升构建速度了。

不过,这并不意味着构建时静态检测不重要,相反这是保证类型安全的最后一道防线。此时,我们可以通过其他性能更优的插件做静态类型检测。

最后,我们在第 22 行引入了 fork-ts-checker-webpack-plugin 专门对 TypeScript 文件进行构建时静态类型检测(可以通过如下命令,安装该插件)。这样,只要出现任何 TypeScript 类型错误,构建就会失败并提示错误信息。

我们可以通过如下命令安装 fork-ts-checker-webpack-plugin 插件。

shell
npm install -D fork-ts-checker-webpack-plugin;

实际上,静态类型检测确实会耗费性能和时间,尤其是项目特别庞大的时候,这个损耗会极大地降低开发体验。此时,我们可以根据实际情况优化 Webpack 配置,比如仅在生产构建时开启静态类型检测、开发构建时关闭静态类型检测,这样既可以保证开发体验,也能保证生产构建的安全性。

除了使用 ts-loader 之外,现在我们也可以使用版本号大于 7 的 babel-loader 作为 TypeScript 的加载器。

具体操作:首先,我们可以通过如下命令安装处理 TypeScript 的 babel preset。

java
// npm i -D babel-loader; // 确保安装版本 > 7
npm i -D @babel/preset-typescript;

注意:因为 React Web 项目必然已经安装了 babel-loader(必须依赖),所以我们不用重新安装 babel-loader,只需确保 babel-loader 的版本号大于 7 即可。

然后,我们在 webpack.config.js 中添加支持 TypeScript 的配置,如下代码所示:

javascript
module.exports = {
// 其他配置 ...
  ​resolve: 
    ​extensions: [".ts", ".tsx", ".js", ".jsx", ".json"]
  ​},
module: {
rules: [
      {
        test: /\.(js|jsx|ts|tsx)$/,
        use: ['babel-loader']
      },
      // ...其他配置
    ​]
  ​},
// ...其他配置
};

在以上配置中的第 4 行、第 9 行,我们配置并使用了 babel-loader 来转换 .ts、.tsx 文件。

最后,我们在 babel 配置文件中添加了如下所示的 typescript presets(参见第 5 行)。

java
{
   "presets": [
     "@babel/preset-env",
     "@babel/preset-react",
     ['@babel/preset-typescript', { allowNamespaces: true }]
   ],
   // ...其他配置
}

注意:因为每个项目中使用的模板不同,所以 babel 配置项可能在 .babelrc、babel.config.js 单独的配置文件中或者内置在 package.json 中。

这样,babel-loader 就可以加载并转换 TypeScript 代码了。

需要注意:因为 babel-loader 也是只对 TypeScript 代码做转换,而不进行静态类型检测,所以我们同样需要引入 fork-ts-checker-webpack-plugin 插件做静态类型检测。

配置好构建工具后,接下来需要迁移 JavaScript 代码,我把这个过程形容为"愚公移山"。

愚公移山

为什么形容为"愚公移山"?因为将 JavaScript 迁移到 TypeScript 是一项个没有太大技术含量的体力活,同时也是一项长久、渐进的过程

迁移 JavaScript 代码的具体操作:首先,我们需要逐个将 .js 文件重名为 .ts、.jsx 文件重名为 .tsx,比如将项目的主入口文件 index.js 改成 index.ts(相应的 webpack.config.js 也需要更改)。然后,我们启动本地服务(npm start)。

不出意外的话,IDE(比如我们推荐的 VS Code)和 fork-ts-checker-webpack-plugin 都会提示 index.ts 有 N 个各式各样的类型错误。

如果我们希望前期始于一个比较高且好的起点,比如在 tsconfig.json 中配置 noImplicitAny 为 true(禁用隐式 any),这样就会提示更多的类型错误。

注意:作为过来人,建议你在 tsconfig.json 的配置上一步到位开启 strict 严格模式。一方面因为我们的课程是基于严格模式编写的,学以致用,另一方面是为了后续无需重复迁移过程,一步到位。当然,你也可以根据项目的实际诉求,选择开启严格模式一步到位或宽松配置 tsconfig。

解决错误

接下来我们要做的事情就是综合利用前面课程的知识(例如 17 讲中介绍的较为常见的错误和分析),逐个解决迁移后的 TypeScript 文件中的各种类型错误。

缺少类型注解

我们看到的第一个错误大概率是缺少某个模块的类型声明文件 ts(7016),比如说缺少路由组件 react-router-dom 的类型声明。

此时,我们可以先通过以下命令尝试安装 DefinitelyTyped 上可能存在的类型声明依赖。

java
npm i -D @types/react-router-dom;

如果命令执行成功,则说明类型声明存在,并且安装成功,这也意味着我们快速且低成本地解决了一个错误。如果 DefinitelyTyped 上恰好没有定义好的依赖类型声明,那么我们就需要自己解决这个问题了。

回想一下 18 讲中是如何解决依赖的 ecstatic 模块缺少类型声明问题的,首先我们需要频繁使用 declare module 补齐类型声明。然后,我们将各种补齐类型声明的文件统一放在 typings 目录中,比如示例 1 中自定义的 jQuery.d.ts(注意:DefinitelyTyped 有 jQuery 类型定义),示例 2 中声明的静态资源 svg、png、jpg、gif 文件模块的 images.d.ts。

示例 1

java
// jQuery.d.ts
declare module 'jQuery';

示例 2

typescript
// images.d.ts
declare module '*.svg';
declare module '*.png';
declare module '*.jpg';
declare module '*.gif';

关于全局变量、属性缺少类型定义的错误,我们也可以使用 declare 或者扩充相应的接口类型进行解决。

首先我们可以创建一个 global.d.ts 补齐缺少的类型声明,如下代码所示:

typescript
declare var $: any;
interface Window {
  __REDUX_DEVTOOLS_EXTENSION__: any;
}
interface NodeModule {
  hot?: {
    accept: (id: string, callback: (...args: any) => void) => void;
  };
}

在示例中的第 1 行,我们声明了全局变量 $,从而解决了找不到 $ 的 ts(2581) 错误。在第 2~4 行,我们扩充了 Window 接口,从而解决了访问 window.REDUX_DEVTOOLS_EXTENSION 时提示属性不存在的 ts(2339) 错误。然后在第 5 到第 9 行,我们扩充了 NodeModule 接口,从而解决了调用 module.hot.accept 方法时提示的 ts(2339) 错误。

注意:不要在 global.d.ts 内添加顶层的 import 或者 export 语句。

插播一道思考题:回忆一下 TypeScript 中 script 和 module 的区别。

隐式 any

接下来就是大量函数参数具有隐式 any 类型的 ts(7006) 错误,此时我们需要给所有函数添加类型注解。在解决这些错误时,如果我们结合 05 讲的知识(比如可选参数、剩余参数函数和函数重载)将会得心应手。

一个好的实践建议:如果我们确实需要暂时使用万金油类型 any 来绕过静态类型检测,则可以声明一个具有特殊含义的全局类型 AnyToFix 来代替 any。比如我们可以在 global.d.ts 内添加如下所示的 AnyToFix 类型别名定义。

typescript
/** 需要替换成更明确的类型 */
type AnyToFix = any;

这样,我们就可以在任何地方使用 AnyToFix 替代 any ,比如下图中的 func 函数参数 arg 的类型就是 AnyToFix。并且在条件成熟时,我们可以很方便地筛选出需要类型重构的 func 函数,然后将其参数类型修改为更明确的类型。

动态类型

另一类极有可能出现的错误是 JavaScript 动态类型特性造成的。

如下示例第 1~3 行所示,我们习惯先定义一个空对象,再动态添加属性,迁移到 TypeScript 后就会提示一个对象上属性不存在的 ts(2339) 错误 。

javascript
const obj = {};
obj.id = 1; // ts(2339)
obj.name = '乾元亨利贞'; // ts(2339)

此时,我们需要通过重构代码解决这个问题,具体操作是预先定义完整的对象结构或类型断言。

代码重构后的示例如下:

javascript
interface IUserInfo {
  id: number;
  name: string;
}
const obj = {} as IUserInfo;
obj.id = 1; // ok
obj.name = '乾元亨利贞'; // ok

在第 5 行中,我们使用了类型断言解决了 ts(2339) 错误。

有用的坏习惯

必要时,我们可以使用 // @ts-ignore 注释强制关闭下一行代码静态类型检测,但这绝对是一个坏习惯,示例如下:

Tips:我们需要铭记所有绕过静态类型检测的方法都是魔鬼,尽量避免使用。

typescript
const objString = {
  toString: () => '乾元亨利贞'
}
function getString(str: string) {
  console.log(`${str}`);
}
// @ts-ignore
getString(objString); // ts(2345)

在示例中的第 7 行,因为我们使用了 // @ts-ignore 注释强行关闭第 8 行的静态类型检测,所以第 8 行并不会提示 ts(2345) 错误。

另外,我们还可以使用 // @ts-nocheck 注释强制关闭整个文件静态类型检测。不过,我建议任何时候都不要使用这个注释。

另外一个有用的坏习惯是双重类型断言,即先把源类型值断言为 unknown,再把 unknown 断言为目标类型。比如上边使用 // @ts-ignore 注释的示例,我们也可以将它改造为双重类型断言,如下代码所示:

typescript
getString(objString as unknown as string);

这样也不会提示 ts(2345) 错误了。

自动迁移工具

如上边所提到,迁移过程是一项没有技术含量的体力活,因为其中存在很多重复、简单、有规律的操作,比如说将 JavaScript 文件修改为 TypeScript 文件、将模块引入方式从 ES5 require 改为 ES 6 import、将参数隐式 any 类型改为显式 any,这就意味着我们可以借助程序自动完成部分重复的迁移操作。

比如我们可以使用 Airebnb 开源迁移工具ts-migrate,快速地将 JavaScript 项目转换为基本可运行的 TypeScript 项目。因为该工具通过语法分析,可以快速推断出逻辑比较简单的函数/对象/类的类型。如下示例 1 中,JavaScript 函数 mult 经 ts-migrate 自动转换为 TypeScript 后,如下示例 2 中所示的参数 first、second 以及函数返回值类型都被明确为 number。

示例 1

javascript
function mult(first, second) {
    return first * second;
}

示例 2

typescript
function mult(first: number, second: number): number {
    return first * second;
}

小结和预告

以上就是JavaScript 应用迁移到 TypeScript 的全部内容,在实际迁移过程中,你需要综合利用前面所有课程的知识才能得心应手。课后建议你挑选一个中小型 JavaScript 项目尝试迁移。

插播一道思考题:将 JavaScript 应用迁移到 TypeScript 的过程中,最常见的错误有哪些?都是如何解决的?

下一讲是结束语,我们将一起回顾过往课程中涉及的一些重难知识点、实用技巧,以及概览 TypeScript 新版本中新增的若干重要特性,敬请期待。

如果你觉得本专栏有价值,欢迎分享给更多好友~