Skip to content

从 JavaScript 迁移到 TypeScript

TypeScript 不是凭空存在的。 它从 JavaScript 生态系统和大量现存的 JavaScript 而来。 将 JavaScript 代码转换成 TypeScript 虽乏味却不是难事。 接下来这篇教程将教你怎么做。 在开始转换 TypeScript 之前,我们假设你已经理解了足够多本手册里的内容。

如果你打算要转换一个 React 工程,推荐你先阅读React 转换指南

设置目录

如果你在写纯 JavaScript,你大概是想直接运行这些 JavaScript 文件, 这些文件存在于srclibdist目录里,它们可以按照预想运行。

若如此,那么你写的纯 JavaScript 文件将做为 TypeScript 的输入,你将要运行的是 TypeScript 的输出。 在从 JS 到 TS 的转换过程中,我们会分离输入文件以防 TypeScript 覆盖它们。 你也可以指定输出目录。

你可能还需要对 JavaScript 做一些中间处理,比如合并或经过 Babel 再次编译。 在这种情况下,你应该已经有了如下的目录结构。

那么现在,我们假设你已经设置了这样的目录结构:

text
projectRoot
├── src
│   ├── file1.js
│   └── file2.js
├── built
└── tsconfig.json

如果你在src目录外还有tests文件夹,那么在src里可以有一个tsconfig.json文件,在tests里还可以有一个。

书写配置文件

TypeScript 使用tsconfig.json文件管理工程配置,例如你想包含哪些文件和进行哪些检查。 让我们先创建一个简单的工程配置文件:

javascript
{
    "compilerOptions": {
        "outDir": "./built",
        "allowJs": true,
        "target": "es5"
    },
    "include": [
        "./src/**/*"
    ]
}

这里我们为 TypeScript 设置了一些东西:

  1. 读取所有可识别的src目录下的文件(通过include)。
  2. 接受 JavaScript 做为输入(通过allowJs)。
  3. 生成的所有文件放在built目录下(通过outDir)。
  4. 将 JavaScript 代码降级到低版本比如 ECMAScript 5(通过target)。

现在,如果你在工程根目录下运行tsc,就可以在built目录下看到生成的文件。 built下的文件应该与src下的文件相同。 现在你的工程里的 TypeScript 已经可以工作了。

早期收益

现在你已经可以看到 TypeScript 带来的好处,它能帮助我们理解当前工程。 如果你打开像VS CodeVisual Studio这样的编译器,你就能使用像自动补全这样的工具。 你还可以配置如下的选项来帮助查找 BUG:

  • noImplicitReturns 会防止你忘记在函数末尾返回值。
  • noFallthroughCasesInSwitch 会防止在switch代码块里的两个case之间忘记添加break语句。

TypeScript 还能发现那些执行不到的代码和标签,你可以通过设置allowUnreachableCodeallowUnusedLabels选项来禁用。

与构建工具进行集成

在你的构建管道中可能包含多个步骤。 比如为每个文件添加一些内容。 每种工具的使用方法都是不同的,我们会尽可能的包涵主流的工具。

Gulp

如果你在使用时髦的 Gulp,我们已经有一篇关于使用 Gulp结合 TypeScript 并与常见构建工具 Browserify,Babelify 和 Uglify 进行集成的教程。 请阅读这篇教程。

Webpack

Webpack 集成非常简单。 你可以使用awesome-typescript-loader,它是一个 TypeScript 的加载器,结合source-map-loader方便调试。 运行:

text
npm install awesome-typescript-loader source-map-loader

并将下面的选项合并到你的webpack.config.js文件里:

javascript
module.exports = {
  entry: './src/index.ts',
  output: {
    filename: './dist/bundle.js',
  },

  // Enable sourcemaps for debugging webpack's output.
  devtool: 'source-map',

  resolve: {
    // Add '.ts' and '.tsx' as resolvable extensions.
    extensions: ['', '.webpack.js', '.web.js', '.ts', '.tsx', '.js'],
  },

  module: {
    loaders: [
      // All files with a '.ts' or '.tsx' extension will be handled by 'awesome-typescript-loader'.
      { test: /\.tsx?$/, loader: 'awesome-typescript-loader' },
    ],

    preLoaders: [
      // All output '.js' files will have any sourcemaps re-processed by 'source-map-loader'.
      { test: /\.js$/, loader: 'source-map-loader' },
    ],
  },

  // Other options...
};

要注意的是,awesome-typescript-loader必须在其它处理.js文件的加载器之前运行。

这与另一个 TypeScript 的 Webpack 加载器ts-loader是一样的。 你可以到这里了解两者之间的差别。

你可以在React 和 Webpack 教程里找到使用 Webpack 的例子。

转换到 TypeScript 文件

到目前为止,你已经做好了使用 TypeScript 文件的准备。 第一步,将.js文件重命名为.ts文件。 如果你使用了 JSX,则重命名为.tsx文件。

第一步达成? 太棒了! 你已经成功地将一个文件从 JavaScript 转换成了 TypeScript!

当然了,你可能感觉哪里不对劲儿。 如果你在支持 TypeScript 的编辑器(或运行tsc --pretty)里打开了那个文件,你可能会看到有些行上有红色的波浪线。 你可以把它们当做在 Microsoft Word 里看到的红色波浪线一样。 但是 TypeScript 仍然会编译你的代码,就好比 Word 还是允许你打印你的文档一样。

如果对你来说这种行为太随便了,你可以让它变得严格些。 如果,你不想在发生错误的时候,TypeScript 还会被编译成 JavaScript,你可以使用noEmitOnError选项。 从某种意义上来讲,TypeScript 具有一个调整它的严格性的刻度盘,你可以将指针拔动到你想要的位置。

如果你计划使用可用的高度严格的设置,最好现在就启用它们(查看启用严格检查)。 比如,如果你不想让 TypeScript 将没有明确指定的类型默默地推断为any类型,可以在修改文件之前启用noImplicitAny。 你可能会觉得这有些过度严格,但是长期收益很快就能显现出来。

去除错误

我们提到过,若不出所料,在转换后将会看到错误信息。 重要的是我们要逐一的查看它们并决定如何处理。 通常这些都是真正的 BUG,但有时必须要告诉 TypeScript 你要做的是什么。

由模块导入

首先你可能会看到一些类似Cannot find name 'require'.Cannot find name 'define'.的错误。 遇到这种情况说明你在使用模块。 你仅需要告诉 TypeScript 它们是存在的:

typescript
// For Node/CommonJS
declare function require(path: string): any;

typescript
// For RequireJS/AMD
declare function define(...args: any[]): any;

最好是避免使用这些调用而改用 TypeScript 的导入语法。

首先,你要使用 TypeScript 的module标记来启用一些模块系统。 可用的选项有commonjsamdsystem,and umd

如果代码里存在下面的 Node/CommonJS 代码:

javascript
var foo = require('foo');

foo.doStuff();

或者下面的 RequireJS/AMD 代码:

javascript
define(['foo'], function (foo) {
  foo.doStuff();
});

那么可以写做下面的 TypeScript 代码:

typescript
import foo = require('foo');

foo.doStuff();

获取声明文件

如果你开始做转换到 TypeScript 导入,你可能会遇到Cannot find module 'foo'.这样的错误。 问题出在没有声明文件来描述你的代码库。 幸运的是这非常简单。 如果 TypeScript 报怨像是没有lodash包,那你只需这样做

text
npm install -S @types/lodash

如果你没有使用commonjs模块模块选项,那么就需要将moduleResolution选项设置为node

之后,你应该就可以导入lodash了,并且会获得精确的自动补全功能。

由模块导出

通常来讲,由模块导出涉及添加属性到exportsmodule.exports。 TypeScript 允许你使用顶级的导出语句。 比如,你要导出下面的函数:

javascript
module.exports.feedPets = function (pets) {
  // ...
};

那么你可以这样写:

typescript
export function feedPets(pets) {
  // ...
}

有时你会完全重写导出对象。 这是一个常见模式,这会将模块变为可立即调用的模块:

javascript
var express = require('express');
var app = express();

之前你可以是这样写的:

javascript
function foo() {
  // ...
}
module.exports = foo;

在 TypeScript 里,你可以使用export =来代替。

typescript
function foo() {
  // ...
}
export = foo;

过多或过少的参数

有时你会发现你在调用一个具有过多或过少参数的函数。 通常,这是一个 BUG,但在某些情况下,你可以声明一个使用arguments对象的函数而不需要写出所有参数:

javascript
function myCoolFunction() {
  if (arguments.length == 2 && !Array.isArray(arguments[1])) {
    var f = arguments[0];
    var arr = arguments[1];
    // ...
  }
  // ...
}

myCoolFunction(
  function (x) {
    console.log(x);
  },
  [1, 2, 3, 4]
);
myCoolFunction(
  function (x) {
    console.log(x);
  },
  1,
  2,
  3,
  4
);

这种情况下,我们需要利用 TypeScript 的函数重载来告诉调用者myCoolFunction函数的调用方式。

typescript
function myCoolFunction(f: (x: number) => void, nums: number[]): void;
function myCoolFunction(f: (x: number) => void, ...nums: number[]): void;
function myCoolFunction() {
  if (arguments.length == 2 && !Array.isArray(arguments[1])) {
    var f = arguments[0];
    var arr = arguments[1];
    // ...
  }
  // ...
}

我们为myCoolFunction函数添加了两个重载签名。 第一个检查myCoolFunction函数是否接收一个函数(它又接收一个number参数)和一个number数组。 第二个同样是接收了一个函数,并且使用剩余参数(...nums)来表示之后的其它所有参数必须是number类型。

连续添加属性

有些人可能会因为代码美观性而喜欢先创建一个对象然后立即添加属性:

javascript
var options = {};
options.color = 'red';
options.volume = 11;

TypeScript 会提示你不能给colorvolumn赋值,因为先前指定options的类型为{}并不带有任何属性。 如果你将声明变成对象字面量的形式将不会产生错误:

typescript
let options = {
  color: 'red',
  volume: 11,
};

你还可以定义options的类型并且添加类型断言到对象字面量上。

typescript
interface Options {
  color: string;
  volume: number;
}

let options = {} as Options;
options.color = 'red';
options.volume = 11;

或者,你可以将options指定成any类型,这是最简单的,但也是获益最少的。

anyObject,和{}

你可能会试图使用Object{}来表示一个值可以具有任意属性,因为Object是最通用的类型。 然而在这种情况下**any是真正想要使用的类型**,因为它是最灵活的类型。

比如,有一个Object类型的东西,你将不能够在其上调用toLowerCase()

越普通意味着更少的利用类型,但是any比较特殊,它是最普通的类型但是允许你在上面做任何事情。 也就是说你可以在上面调用,构造它,访问它的属性等等。 记住,当你使用any时,你会失去大多数 TypeScript 提供的错误检查和编译器支持。

如果你还是决定使用Object{},你应该选择{}。 虽说它们基本一样,但是从技术角度上来讲{}在一些深奥的情况里比Object更普通。

启用严格检查

TypeScript 提供了一些检查来保证安全以及帮助分析你的程序。 当你将代码转换为了 TypeScript 后,你可以启用这些检查来帮助你获得高度安全性。

没有隐式的any

在某些情况下 TypeScript 没法确定某些值的类型。 那么 TypeScript 会使用any类型代替。 这对代码转换来讲是不错,但是使用any意味着失去了类型安全保障,并且你得不到工具的支持。 你可以使用noImplicitAny选项,让 TypeScript 标记出发生这种情况的地方,并给出一个错误。

严格的nullundefined检查

默认地,TypeScript 把nullundefined当做属于任何类型。 这就是说,声明为number类型的值可以为nullundefined。 因为在 JavaScript 和 TypeScript 里,nullundefined经常会导致 BUG 的产生,所以 TypeScript 包含了strictNullChecks选项来帮助我们减少对这种情况的担忧。

当启用了strictNullChecksnullundefined获得了它们自己各自的类型nullundefined。 当任何值可能null,你可以使用联合类型。 比如,某值可能为numbernull,你可以声明它的类型为number | null

假设有一个值 TypeScript 认为可以为nullundefined,但是你更清楚它的类型,你可以使用!后缀。

typescript
declare var foo: string[] | null;

foo.length; // error - 'foo' is possibly 'null'

foo!.length; // okay - 'foo!' just has type 'string[]'

要当心,当你使用strictNullChecks,你的依赖也需要相应地启用strictNullChecks

this没有隐式的any

当你在类的外部使用this关键字时,它会默认获得any类型。 比如,假设有一个Point类,并且我们要添加一个函数做为它的方法:

typescript
class Point {
  constructor(public x, public y) {}
  getDistance(p: Point) {
    let dx = p.x - this.x;
    let dy = p.y - this.y;
    return Math.sqrt(dx ** 2 + dy ** 2);
  }
}
// ...

// Reopen the interface.
interface Point {
  distanceFromOrigin(point: Point): number;
}
Point.prototype.distanceFromOrigin = function (point: Point) {
  return this.getDistance({ x: 0, y: 0 });
};

这就产生了我们上面提到的错误 - 如果我们错误地拼写了getDistance并不会得到一个错误。 正因此,TypeScript 有noImplicitThis选项。 当设置了它,TypeScript 会产生一个错误当没有明确指定类型(或通过类型推断)的this被使用时。 解决的方法是在接口或函数上使用指定了类型的this参数:

typescript
Point.prototype.distanceFromOrigin = function (this: Point, point: Point) {
  return this.getDistance({ x: 0, y: 0 });
};
从 JavaScript 迁移到 TypeScript has loaded