TypeORM - 最好的Node.js ORM框架
发布于 8 年前 作者 huanz 27223 次浏览 最后一次编辑是 7 年前 来自 分享

目前正在翻译中文文档,期待大家的一起参与:

typeorm中文文档

中文文档地址:http://typeorm.cn/

TypeORM

TypeORM是一个ORM框架,它可以运行在NodeJS、浏览器、Cordova、PhoneGap和Ionic平台上,可以与TypeScript和JavaScript (ES5, ES6, ES7)一起使用。 它的目标是始终支持最新的JavaScript特性并提供额外的特性以帮助你开发任何使用数据库的应用程序 —— 不管是只有几张表的小型应用还是拥有多数据库的大型企业应用。

不同于现有的所有其他JavaScript ORM框架,TypeORM支持Active Record和Data Mapper模式,这意味着你用最有效的方法编写高质量的、松耦合的、可扩展的、可维护的应用程序。

TypeORM受到了参考了很多其他优秀ORM的实现, 比如 Hibernate, DoctrineEntity Framework.

TypeORM 的一些特性:

  • 支持Active Record和Data Mapper(你可以自由选择)
  • 实体和列
  • 数据库特性列类型
  • 实体管理
  • 存储库和自定义存储库
  • 清洁对象关系模型
  • 关联(关系)
  • 贪婪和延迟关系
  • 单向的,双向的和自引用的关系
  • 支持多重继承模式
  • 级联
  • 索引
  • 事务
  • 迁移和自动迁移
  • 连接池
  • 复制
  • 使用多个数据库连接
  • 使用多个数据库类型
  • 跨数据库和跨模式查询
  • 优雅的语法,灵活而强大的QueryBuilder
  • 左联接和内联接
  • 准确的分页连接查询
  • 查询缓存
  • 原始结果流
  • 日志
  • 监听者和订阅者(钩子)
  • 支持闭包表模式
  • 在模型或者分离的配置文件中声明模式
  • json / xml / yml / env格式的连接配置
  • 支持 MySQL / MariaDB / Postgres / SQLite / Microsoft SQL Server / Oracle / WebSQL / sql.js
  • 支持 MongoDB NoSQL 数据库
  • 在NodeJS / 浏览器 / Ionic / Cordova / Electron平台上工作
  • 支持 TypeScript 和 JavaScript
  • 产生出高性能、灵活、清洁和可维护的代码
  • 遵循所有可能的最佳实践
  • 命令行工具

还有更多...

使用TypeORM你的模型是这样的:

import {Entity, PrimaryGeneratedColumn, Column} from "typeorm";
@Entity()
export class User {
 @PrimaryGeneratedColumn()
 id: number;
 @Column()
 firstName: string;
 @Column()
 lastName: string;
 @Column()
 age: number;
}

你的域逻辑是这样的:

const user = new User();
user.firstName = "Timber";
user.lastName = "Saw";
user.age = 25;
await repository.save(user);
const allUsers = await repository.find();
const firstUser = await repository.findOneById(1);
const timber = await repository.findOne({ firstName: "Timber", lastName: "Saw" });
await repository.remove(timber);

或者,你如果你喜欢使用"ActiveRecord"实现,你也可以使用它:

import {Entity, PrimaryGeneratedColumn, Column, BaseEntity} from "typeorm";
@Entity()
export class User extends BaseEntity {
 @PrimaryGeneratedColumn()
 id: number;
 @Column()
 firstName: string;
 @Column()
 lastName: string;
 @Column()
 age: number;
}

你的域逻辑是这样的:

const user = new User();
user.firstName = "Timber";
user.lastName = "Saw";
user.age = 25;
await user.save();
const allUsers = await User.find();
const firstUser = await User.findOneById(1);
const timber = await User.findOne({ firstName: "Timber", lastName: "Saw" });
await timber.remove();

请注意

这个文档可能不是最新的。 可以去官网查看最新的英文文档。 非常欢迎你的贡献。

安装

  1. 安装TypeORM:

    npm install typeorm --save

  2. 需要安装依赖模块 reflect-metadata :

    npm install reflect-metadata --save

    在应用里全局引用一下:

    • 比如在app.ts的入口处 require("reflect-metadata")
  3. 你可能需要安装node类型:

    npm install @types/node --save

  4. 安装数据库驱动:

    • MySQLMariaDB

      npm install mysql --save

    • PostgreSQL

      npm install pg --save

    • SQLite

      npm install sqlite3 --save

    • Microsoft SQL Server

      npm install mssql --save

    • sql.js

      npm install sql.js --save

    • Oracle (experimental)

      npm install oracledb --save

    可以根据你的数据库选择安装上面的任意一个.

    使用oracle驱动需要参考安装说明:地址.

TypeScript配置

确保你的TypeScript编译器的版本大于2.3,并且在tsconfig.json开启下面设置:

"emitDecoratorMetadata": true,
"experimentalDecorators": true,

同时需要开启编译选项里的lib下的es6或者从@typings安装es6-shim

快速开始

开始使用TypeORM的最快方法是使用它的CLI命令生成一个初始项目。 快速开始只有在NodeJS应用程序中使用TypeORM才可以使用。 如果你正在使用其他平台,请看分步指南

首先全局安装TypeORM:

npm install typeorm -g

然后转到新项目的目录并运行该命令:

typeorm init --name MyProject --database mysql

name即项目的名称,database是你将使用的数据库。数据库可以是下列值之一:mysqlmariadbpostgressqlitemssqloracle,websqlmongodb

该命令将在MyProject目录中生成一个新项目,其中包含以下文件:

MyProject
├── src // 放你的 TypeScript 代码
│ ├── entity // 放实体(数据库模型)的目录
│ │ └── User.ts // 实体的案例
│ ├── migration // 迁移文件目录
│ └── index.ts // 应用程序入口
├── .gitignore // 标准git忽略文件
├── ormconfig.json // ORM和数据连接配置
├── package.json // node模块依赖
├── README.md // 简单的说明文件
└── tsconfig.json // TypeScript编译配置

你也可以在现有的node项目目录执行typeorm init,但是一定要小心 - 它可能会覆盖你已经有的一些文件。

下一步是安装项目依赖

cd MyProject
npm install

在安装过程中,修改 ormconfig.json 文件将自己的数据库连接配置选项放在其中:

{
 "type": "mysql",
 "host": "localhost",
 "port": 3306,
 "username": "test",
 "password": "test",
 "database": "test",
 "synchronize": true,
 "logging": false,
 "entities": [
 "src/entity/**/*.ts"
 ],
 "migrations": [
 "src/migration/**/*.ts"
 ],
 "subscribers": [
 "src/subscriber/**/*.ts"
 ]
}

通常来说,大多数时候你只需要配置host,username,password,database 或者 port 选项。

配置和模块安装都完成之后,就可以运行应用程序了:

npm start

就是这样,你的应用程序应该成功地运行并将一个新用户插入到数据库中。 你可以继续这个项目,集成你需要的其他模块,并创建更多的实体。

运行typeorm init --name MyProject --database mysql --express命令可以安装express,生成一个更高级的项目。

分步指南

你对ORM的期望是什么? 首先,你预期它将为你创建数据库表,并查找/插入/更新/删除你的数据,而不必编写大量难以维护的SQL查询。 本指南将向你展示如何从头开始设置TypeORM,并让它按照你所期望的ORM进行。

创建模型

与数据库一起工作从创建表开始。 如何告诉TypeORM创建一个数据库表? 答案是 - 通过模型。 你的应用程序中的模型就是你的数据库中的表。

例如你有一个 Photo 模型:

export class Photo {
 id: number;
 name: string;
 description: string;
 filename: string;
 views: number;
}

你想在你的数据库中存储照片。 要在数据库中存储东西,首先需要一个数据库表,并从模型创建数据库表。 不是所有的模型,而仅仅是那些你定义为实体

创建实体

实体是你用 @Entity 装饰的模型。 将为这些模型创建一个数据库表。 使用TypeORM你将在任何地方使用实体。 你可以使用他们加载/插入/更新/删除并执行其他操作。

让我们把Photo模型变成一个实体:

import {Entity} from "typeorm";
@Entity()
export class Photo {
 id: number;
 name: string;
 description: string;
 filename: string;
 views: number;
 isPublished: boolean;
}

现在,将会为 Photo 实体创建一个数据库表,我们能够在应用程序的任何地方使用它。 我们已经创建了一个数据库表,然而没有列的表示不存在的。 让我们在数据库表中创建一些列吧。

添加数据库表列

要添加数据库列,只需要将生成的实体的属性用 @Column 装饰。

import {Entity, Column} from "typeorm";
@Entity()
export class Photo {
 @Column()
 id: number;
 @Column()
 name: string;
 @Column()
 description: string;
 @Column()
 filename: string;
 @Column()
 views: number;
 @Column()
 isPublished: boolean;
}

现在 id,name,description,filename,viewsisPublished 列将会被添加 photo 表。 数据库中的列类型是从你使用的属性类型推断出来的,例如:number 将会被转成 integer,string 转为 varchar,boolean 转为 bool,等。 但是你可以通过隐式在 @Column 装饰器传入类型将列类型指定为任何你数据库支持的类型。

我们生成了一个带有列的数据库表,但是还剩下一件事。 每个数据库表必须有一个带有主键的列。

创建一个主键列

每个表都必须至少有一个主键列。这是一个要求,你不能避免。要使列成为主键,你需要使用 @PrimaryColumn 修饰符。

import {Entity, Column, PrimaryColumn} from "typeorm";
@Entity()
export class Photo {
 @PrimaryColumn()
 id: number;
 @Column()
 name: string;
 @Column()
 description: string;
 @Column()
 filename: string;
 @Column()
 views: number;
 @Column()
 isPublished: boolean;
}

创建一个自动生成的列

现在,假设你希望将id列自动生成(这就是所谓的自动递增/按顺序/连续的/生成唯一标识列)。 要做到这一点,你需要将 @PrimaryColumn 修饰符更改为 @PrimaryGeneratedColumn 修饰符:

import {Entity, Column, PrimaryGeneratedColumn} from "typeorm";
@Entity()
export class Photo {
 @PrimaryGeneratedColumn()
 id: number;
 @Column()
 name: string;
 @Column()
 description: string;
 @Column()
 filename: string;
 @Column()
 views: number;
 @Column()
 isPublished: boolean;
}

列数据类型

接下来,让我们修复数据类型。默认情况下,字符串被映射到一个varchar(255)类型(取决于数据库类型)。 数字被映射到一个integer类型(取决于数据库类型)。 我们不希望所有的列都是有限的varchars或整数。 让我们设置正确的数据类型:

import {Entity, Column, PrimaryGeneratedColumn} from "typeorm";
@Entity()
export class Photo {
 @PrimaryGeneratedColumn()
 id: number;
 @Column({
 length: 100
 })
 name: string;
 @Column("text")
 description: string;
 @Column()
 filename: string;
 @Column("double")
 views: number;
 @Column()
 isPublished: boolean;
}

列类型取决于数据库支持的类型。 可以设置数据库支持的任何列类型。 更多关于支持的列类型信息可以在这里找到这里

创建数据库连接

现在实体已经有了,让我们新建一个 index.ts (或 app.ts 不管你叫它什么)的文件,并配置数据库连接:

import "reflect-metadata";
import {createConnection} from "typeorm";
import {Photo} from "./entity/Photo";
createConnection({
 type: "mysql",
 host: "localhost",
 port: 3306,
 username: "root",
 password: "admin",
 database: "test",
 entities: [
 Photo
 ],
 synchronize: true,
 logging: false
}).then(connection => {
 // 这里可以写实体操作相关的代码 
}).catch(error => console.log(error));

在例子里使用的是mysql,你也可以选择其他数据库,只需要简单修改driver选项里的数据库的类型就可以了,比如: mysql, mariadb, postgres, sqlite, mssql or oracle. 同样可以修改host, port, username, password 以及database等设置.

把Photo实体加到数据连接的实体列表中,所有需要在这个连接下使用的实体都必须加到这个列表中。

synchronize选项可以在应用启动时确保你的实体和数据库保持同步。

引用目录下的所有实体

接下来我们可能会创建更多的实体并把它们一一加到配置当中。 不过这样会比较麻烦,好在可以直接写上实体的目录,这样这个目录下的所有实体都可以在当前连接中被使用:

import {createConnection} from "typeorm";
createConnection({
 driver: {
 type: "mysql",
 host: "localhost",
 port: 3306,
 username: "root",
 password: "admin",
 database: "test"
 },
 entities: [
 __dirname + "/entity/*.js"
 ],
 synchronize: true,
}).then(connection => {
 // here you can start to work with your entities
}).catch(error => console.log(error));

启动应用

现在可以启动app.ts,启动后可以发现数据库自动被初始化,并且Photo这个表也会创建出来。

+-------------+--------------+----------------------------+
| photo |
+-------------+--------------+----------------------------+
| id | int(11) | PRIMARY KEY AUTO_INCREMENT |
| name | varchar(500) | |
| description | text | |
| filename | varchar(255) | |
| views | int(11) | |
| isPublished | boolean | |
+-------------+--------------+----------------------------+

添加和插入photo

现在创建一个新的photo然后存到数据库:

import {createConnection} from "typeorm";
createConnection(/*...*/).then(connection => {
 let photo = new Photo();
 photo.name = "Me and Bears";
 photo.description = "I am near polar bears";
 photo.filename = "photo-with-bears.jpg";
 photo.views = 1;
 photo.isPublished = true;
 connection.manager
 .save(photo)
 .then(photo => {
 console.log("Photo has been saved");
 });
}).catch(error => console.log(error));

使用async/await语法

现在利用TypeScript的async/await语法来实现同样的功能:

import {createConnection} from "typeorm";
import {Photo} from "./entity/Photo";
createConnection(/*...*/).then(async connection => {
 let photo = new Photo();
 photo.name = "Me and Bears";
 photo.description = "I am near polar bears";
 photo.filename = "photo-with-bears.jpg";
 photo.views = 1;
 photo.isPublished = true;
 await connection.manager.save(photo);
 console.log("Photo has been saved");
}).catch(error => console.log(error));

使用EntityManager

刚刚我们创建了一个新的photo并且存进数据库。使用EntityManager可以操作实体,现在用EntityManager来把photo从数据库中取出来。

import {createConnection} from "typeorm";
import {Photo} from "./entity/Photo";
createConnection(/*...*/).then(async connection => {
 /*...*/
 let savedPhotos = await connection.manager.find(Photo);
 console.log("All photos from the db: ", savedPhotos);
}).catch(error => console.log(error));

savedPhotos 会从数据库中取到的是一个Photo对象的数组

使用Repositories

现在重构下代码,使用Repository来代替EntityManage。每个实体都有自己的repository,可以对这个实体进行任何操作。 如果要对实体做很多操作,Repositories会比EntityManager更加方便。

import {createConnection} from "typeorm";
import {Photo} from "./entity/Photo";
createConnection(/*...*/).then(async connection => {
 let photo = new Photo();
 photo.name = "Me and Bears";
 photo.description = "I am near polar bears";
 photo.filename = "photo-with-bears.jpg";
 photo.views = 1;
 photo.isPublished = true;
 let photoRepository = connection.getRepository(Photo);
 await photoRepository.save(photo);
 console.log("Photo has been saved");
 let savedPhotos = await photoRepository.find();
 console.log("All photos from the db: ", savedPhotos);
}).catch(error => console.log(error));

从数据库中取photos

现在来尝试用Repository做一些取数据方面的操作:

import {createConnection} from "typeorm";
import {Photo} from "./entity/Photo";
createConnection(/*...*/).then(async connection => {
 /*...*/
 let allPhotos = await photoRepository.find();
 console.log("All photos from the db: ", allPhotos);
 let firstPhoto = await photoRepository.findOneById(1);
 console.log("First photo from the db: ", firstPhoto);
 let meAndBearsPhoto = await photoRepository.findOne({ name: "Me and Bears" });
 console.log("Me and Bears photo from the db: ", meAndBearsPhoto);
 let allViewedPhotos = await photoRepository.find({ views: 1 });
 console.log("All viewed photos: ", allViewedPhotos);
 let allPublishedPhotos = await photoRepository.find({ isPublished: true });
 console.log("All published photos: ", allPublishedPhotos);
 let [allPhotos, photosCount] = await photoRepository.findAndCount();
 console.log("All photos: ", allPublishedPhotos);
 console.log("Photos count: ", allPublishedPhotos);
}).catch(error => console.log(error));

更新photo

现在来从数据库中取出一个photo,修改并更新到数据库。

import {createConnection} from "typeorm";
import {Photo} from "./entity/Photo";
createConnection(/*...*/).then(async connection => {
 /*...*/
 let photoToUpdate = await photoRepository.findOneById(1);
 photoToUpdate.name = "Me, my friends and polar bears";
 await photoRepository.save(photoToUpdate);
}).catch(error => console.log(error));

这个id = 1的photo在数据库中就成功更新了.

删除photo

再来,从数据库中删除我们的photo:

import {createConnection} from "typeorm";
import {Photo} from "./entity/Photo";
createConnection(/*...*/).then(async connection => {
 /*...*/
 let photoToRemove = await photoRepository.findOneById(1);
 await photoRepository.remove(photoToRemove);
}).catch(error => console.log(error));

这个id = 1的photo就在数据库中被移除了。

一对一关系

来创建与另一个类的一对一关系。 新建PhotoMetadata.ts用来存photo的元信息。

import {Entity, Column, PrimaryGeneratedColumn, OneToOne, JoinColumn} from "typeorm";
import {Photo} from "./Photo";
@Entity()
export class PhotoMetadata {
 @PrimaryGeneratedColumn()
 id: number;
 @Column("int")
 height: number;
 @Column("int")
 width: number;
 @Column()
 orientation: string;
 @Column()
 compressed: boolean;
 @Column()
 comment: string;
 @OneToOne(type => Photo)
 @JoinColumn()
 photo: Photo;
}

这里我们用到了一个新的装饰器@OneToOne,它可以用来在两个实体之间创建一对一关系。 type => Photo指示了我们想要连接的实体类名,这里因为TypeScript语言的支持原因不能直接用类名。 当然也可以使用() => Photo,但是type => Photo显得更有可读性。 Type变量本身并不包含任何东西。

我们同样使用了@JoinColumn装饰器,这个装饰器可以指定一对一关系的拥有者。 关系可以是单向的或双向的,但是只有一方是拥有者,加个这个装饰器就表示关系是给这个表服务的。

现在运行app,会新创建一个table,这个table有一个连接photo的外键:

+-------------+--------------+----------------------------+
| photo `译者注:应该是PhotoMetadata` |
+-------------+--------------+----------------------------+
| id | int(11) | PRIMARY KEY AUTO_INCREMENT |
| height | int(11) | |
| width | int(11) | |
| comment | varchar(255) | |
| compressed | boolean | |
| orientation | varchar(255) | |
| photo | int(11) | FOREIGN KEY |
+-------------+--------------+----------------------------+

存一个有一对一关系的对象

现在来创建一个photo,一个photo的元信息,并把它们已经连接起来。

import {createConnection} from "typeorm";
import {Photo} from "./entity/Photo";
import {PhotoMetadata} from "./entity/PhotoMetadata";
createConnection(/*...*/).then(async connection => {
 // 创建一个photo
 let photo = new Photo();
 photo.name = "Me and Bears";
 photo.description = "I am near polar bears";
 photo.filename = "photo-with-bears.jpg"
 photo.isPublished = true;
 // 创建一个photo的元信息
 let metadata = new PhotoMetadata();
 metadata.height = 640;
 metadata.width = 480;
 metadata.compressed = true;
 metadata.comment = "cybershoot";
 metadata.orientation = "portait";
 metadata.photo = photo; // 这里把两者连起来
 // 获取实体repositories
 let photoRepository = connection.getRepository(Photo);
 let metadataRepository = connection.getRepository(PhotoMetadata);
 // 先来把photo存到数据库
 await photoRepository.save(photo);
 // photo存完了,再存下photo的元信息
 await metadataRepository.save(metadata);
 // 搞定
 console.log("metadata is saved, and relation between metadata and photo is created in the database too");
}).catch(error => console.log(error));

双向关系

关系可以是单向的或是双向的. 现在PhotoMetadata和Photo的关系是单向的,关系拥有者是PhotoMetadata,Photo并不知道PhotoMetadata,这样如果要想从Photo里得到PhotoMetadata的数据会比较麻烦。 现在来改变一下,把单向改成双向:

import {Entity, Column, PrimaryGeneratedColumn, OneToOne, JoinColumn} from "typeorm";
import {Photo} from "./Photo";
@Entity()
export class PhotoMetadata {
 /* ... 其他列 */
 @OneToOne(type => Photo, photo => photo.metadata)
 @JoinColumn()
 photo: Photo;
}
import {Entity, Column, PrimaryGeneratedColumn, OneToOne} from "typeorm";
import {PhotoMetadata} from "./PhotoMetadata";
@Entity()
export class Photo {
 /* ... 其他列 */
 @OneToOne(type => PhotoMetadata, photoMetadata => photoMetadata.photo)
 metadata: PhotoMetadata;
}

photo => photo.metadata 是用来指定反向关系的字段名字,photo.metadata就指出了Photo里的metadata字段名字。 当然也可以使用@OneToOne('metadata')来达到同样的目的,不过这种对于以后的代码重构不友好。

按上面说的,@JoinColumn只能在关系的一边使用来使这边做为关系的拥有者,关系拥有者在数据库里的表现就是拥有一个外键列。

取出关系对象的数据

现在来用一个查询来取出photo以及它的元信息。 有两种方式,一是用FindOptions,另一个是使用QueryBuilder。 先试下FindOptions,通过指定FindOptions接口作为参数来使用Repository.find方法可以完成非常复杂的查询。

import {createConnection} from "typeorm";
import {Photo} from "./entity/Photo";
import {PhotoMetadata} from "./entity/PhotoMetadata";
createConnection(/*...*/).then(async connection => {
 /*...*/
 let photoRepository = connection.getRepository(Photo);
 let photos = await photoRepository.find({
 alias: "photo",
 innerJoinAndSelect: {
 "metadata": "photo.metadata"
 }
 });
}).catch(error => console.log(error));

返回的photos是从数据库里取回的photo的数组,每个photo都包含它的元信息。

alias 是FindOptions的一个必需选项,这是你自己在select里定义的别名,然后需要用在接下来的 where, order by, group by, join 以及其他表达式.

这里还用到了innerJoinAndSelect,表示内联查询photo.metadata的数据。 "photo.metadata"里"photo"是一个别名,"metadata"则是你想查询的那个对象的属性名。 "metadata": 是内联返回数据的新的别名.

下面来尝试第二种方式:QueryBuilder来达到同样的目的. 使用QueryBuilder可以优雅完成复杂的查询:

import {createConnection} from "typeorm";
import {Photo} from "./entity/Photo";
import {PhotoMetadata} from "./entity/PhotoMetadata";
createConnection(/*...*/).then(async connection => {
 /*...*/
 let photoRepository = connection.getRepository(Photo);
 let photos = await photoRepository.createQueryBuilder("photo")
 .innerJoinAndSelect("photo.metadata", "metadata")
 .getMany();
}).catch(error => console.log(error));

使用 cascade 选项来自动保存关系着的对象

上面要保存关系对象需要一个一个来保存,略显麻烦。 如果我们需要当关系对象中的一个被保存后,另一个也同样被保存,则可以使用cascade选项来做到。 稍微改下@OneToOne装饰:

export class Photo {
 /// ... 其他列
 @OneToOne(type => PhotoMetadata, metadata => metadata.photo, {
 cascadeInsert: true,
 cascadeUpdate: true,
 cascadeRemove: true
 })
 metadata: PhotoMetadata;
}
  • cascadeInsert - 如果表中没有关系中的metadata,则自动insert,即我们不需要再手动insert一个新的photoMetadata对象。
  • cascadeUpdate - 如果metadata有变化,则自动update。
  • cascadeRemove - 如果把photo里的metadata移除了,也就是为空,则会自动remove表中的这条metadata数据。

使用cascadeInsert就可以不需要像上面那边先存photo再存metadata了。 现在我们来单单存photo对象,由于cascade的作用,metadata也会自动存上。

createConnection(options).then(async connection => {
 // 创建photo对象
 let photo = new Photo();
 photo.name = "Me and Bears";
 photo.description = "I am near polar bears";
 photo.filename = "photo-with-bears.jpg"
 photo.isPublished = true;
 // 创建photo metadata 对象
 let metadata = new PhotoMetadata();
 metadata.height = 640;
 metadata.width = 480;
 metadata.compressed = true;
 metadata.comment = "cybershoot";
 metadata.orientation = "portait";
 
 photo.metadata = metadata; // 连接起来
 // 得到repository
 let photoRepository = connection.getRepository(Photo);
 // 存photo
 await photoRepository.save(photo);
 // photo metadata也自动存上了
 console.log("Photo is saved, photo metadata is saved too.")
}).catch(error => console.log(error));

多对一/一对多关系

接下来显示多对一/一对多关系。 假设一个photo会有一个author,并且每个author可以有很多photo。 先创建Author实体:

import {Entity, Column, PrimaryGeneratedColumn, OneToMany, JoinColumn} from "typeorm";
import {Photo} from "./Photo";
@Entity()
export class Author {
 @PrimaryGeneratedColumn()
 id: number;
 @Column()
 name: string;
 @OneToMany(type => Photo, photo => photo.author) // 备注:下面会为Photo创建author属性
 photos: Photo[];
}

Author包含一个反向的关系,OneToMany总是反向的,并且总是与ManyToOne成对出现。

现在来为Photo加上关系拥有者。

import {Entity, Column, PrimaryGeneratedColumn, ManyToOne} from "typeorm";
import {PhotoMetadata} from "./PhotoMetadata";
import {Author} from "./Author";
@Entity()
export class Photo {
 /* ... 其他列 */
 @ManyToOne(type => Author, author => author.photos)
 author: Author;
}

ManyToOne/OneToMany关系中,拥有者一边总是ManyToOne译者注:拥有外键者即关系拥有者 也就是ManyToOne的那个字段存的是另一个对象的id。译者注:也就是上面的author虽然属性是Author,但在数据库中类型是Author id的类型,存的也是id

执行上面的代码将会自动创建author表,如下:

+-------------+--------------+----------------------------+
| author |
+-------------+--------------+----------------------------+
| id | int(11) | PRIMARY KEY AUTO_INCREMENT |
| name | varchar(255) | |
+-------------+--------------+----------------------------+

因为photo表已经存在,所以不是增加而是修改photo表 - 添加一个新外键列author:

+-------------+--------------+----------------------------+
| photo |
+-------------+--------------+----------------------------+
| id | int(11) | PRIMARY KEY AUTO_INCREMENT |
| name | varchar(255) | |
| description | varchar(255) | |
| filename | varchar(255) | |
| isPublished | boolean | |
| author | int(11) | FOREIGN KEY |
+-------------+--------------+----------------------------+

多对多关系

假设photo可以存在多个相册中,并且相册里可以包含多个photo。 先创建一个Album

import {Entity, PrimaryGeneratedColumn, Column, ManyToMany, JoinTable} from "typeorm";
@Entity()
export class Album {
 @PrimaryGeneratedColumn()
 id: number;
 @Column()
 name: string;
 @ManyToMany(type => Photo, photo => photo.albums, { // 备注: 会在下面的Photo类里添加"albums"属性
 cascadeInsert: true, // 在添加Album时,会自动添加相册里的Photo
 cascadeUpdate: true, // 在更新Album时,会自动更新相册里的Photo 
 cascadeRemove: true // 在移除Album时,会自动移除相册里的Photo
 })
 @JoinTable()
 photos: Photo[];
}

@JoinTable多对多关系拥有者必须指定的。

接着给Photo实体加个反向关系:

export class Photo {
 /// ... 其他列
 @ManyToMany(type => Album, album => album.photos, {
 cascadeInsert: true, // 在添加Album时,会自动添加相册里的Photo
 cascadeUpdate: true, // 在更新Album时,会自动更新相册里的Photo 
 cascadeRemove: true // 在移除Album时,会自动移除相册里的Photo
 })
 albums: Album[];
}

执行上面的代码后会自动创建一个叫 album_photos_photo_albums联接表:

+-------------+--------------+----------------------------+
| album_photos_photo_albums |
+-------------+--------------+----------------------------+
| album_id_1 | int(11) | PRIMARY KEY FOREIGN KEY |
| photo_id_2 | int(11) | PRIMARY KEY FOREIGN KEY |
+-------------+--------------+----------------------------+

记得把Album实体加到ConnectionOptions中:

const options: CreateConnectionOptions = {
 // ... 其他配置
 entities: [Photo, PhotoMetadata, Author, Album]
};

现在来往数据库里插入albums和photos

let connection = await createConnection(options);
// 创建两个albums
let album1 = new Album();
album1.name = "Bears";
let album2 = new Album();
album2.name = "Me";
// 创建两个photos
let photo1 = new Photo();
photo1.name = "Me and Bears";
photo1.description = "I am near polar bears";
photo1.filename = "photo-with-bears.jpg";
photo1.albums = [album1];
let photo2 = new Photo();
photo2.name = "Me and Bears";
photo2.description = "I am near polar bears";
photo2.filename = "photo-with-bears.jpg";
photo2.albums = [album2];
// 获取Photo的repository
let photoRepository = connection.getRepository(Photo);
// 依次存储photos,由于cascade,albums也同样会自动存起来
await photoRepository.save(photo1);
await photoRepository.save(photo2);
console.log("Both photos have been saved");

使用QueryBuilder

可以利用QueryBuilder来构建一个非常复杂的查询,例如:

let photoRepository = connection.getRepository(Photo);
let photos = await photoRepository
 .createQueryBuilder("photo") // 别名,必填项,用来指定本次查询
 .innerJoinAndSelect("photo.metadata", "metadata")
 .leftJoinAndSelect("photo.albums", "albums")
 .where("photo.isPublished=true")
 .andWhere("(photo.name=:photoName OR photo.name=:bearName)")
 .orderBy("photo.id", "DESC")
 .skip(5)
 .take(10)
 .setParameters({ photoName: "My", bearName: "Mishka" })
 .getMany();

这个查询会查找已经published的,并且name是"My"或"Mishka", 得到的结果会从第5个开始(分页偏移决定的), 并且只会得到10个结果(分页每页个数决定的), 所得结果是以id的倒序排序的, Photo的albums是左联接,photo的metadata是内联接。

你将在应用程序中大量使用QueryBuilder。 了解更多QueryBuilder这里.

样例

看看样例里这些例子的用法

这些仓库,你可以克隆下来帮助你开始:

扩展

这几个扩展可以简化TypeORM的使用,并将其与其他模块集成:

37 回复

虽然装饰器有时候挺好用,但是这样滥用真的好吗

js社区好强大👍👍👍

👍 强大 各种炫酷的玩法,长见识

@hpgt 没有太理解你所说的滥用

@hpgt 看看这个: https://docs.nestjs.cn 不觉得是滥用

@huanz 搜狗截图20180402142247.png 这是nest.js的截图,红框部分这些注解就是增加了负责度。原本nodejs都是简单明了,现在学起了java spring框架,搞出一堆装饰器

@zuohuadong 甚至连dto这种东西也搬过来了,跟着java的步伐又怎么能超越java

违反广告法。

const some = ctx.body.some

(@Body some) { // some }

各有千秋吧。前端以前没有装饰器,喜欢用用找找快感也没啥问题啊。

@hpgt 没啥超越不超越的,java 在大型项目架构上积累的经验,值得很多语言去学习。 node 跟 java 也从来不是敌人,没必要动不动就超越〜 但就 express koa 来说,连跟 spring 动手的资格都不够

@hpgt 我也觉得没必要,就像路由,可能因为最早接触的是laravel的关系,我个人更倾向集中化管理

来自酷炫的 CNodeMD

@hpgt 这里怎么说是滥用呢?我们随便来看下express的应用例子:

nodeclub/controllers/topic.js

exports.update = function (req, res, next) {
 var topic_id = req.params.tid;
 var title = req.body.title;
 var tab = req.body.tab;
 var content = req.body.t_content;
 ...

再来看下eggjs的 cnodejs/egg-cnode/app/controller/topic.js

 async update() {
 const { ctx, service, config } = this;
 const topic_id = ctx.params.tid;
 let { title, tab, content } = ctx.request.body;
	...
}

回头看看 nestjs 的

@Controller('cats')
export class CatsController {
 @Post()
 create(@Res() res, @Body() createCatDto: CreateCatDto) {
 // TODO: Add some logic here
 res.status(HttpStatus.CREATED).send();
 }
 @Get()
 findAll(@Res() res) {
 res.status(HttpStatus.OK).json([]);
 }
}

每个方法对应的请求类型一目了然,需要用到什么方法,什么参数,就注入什么。而且实现的也很优雅。天生拥抱ts

@hpgt 另外你仍然没有回答你所说的 typeorm 装饰器滥用的问题

@huanz 哥们,不要太较真,每个框架都有自己的风格,有些地方好用了必然会有过度封装的情况。滥用也是我自己这么觉得,你要是觉得好那就好

@liuzhiguo11 路由集中管理没问题,但是一些前置、后置处理使用装饰器来声明调用还是很方便啊

最近出现的一些框架,ORM,让我也觉得装饰器被滥用了

我也在写spring boot。但是真心不喜欢这种注解满天飞的风格。

很好很强大... npm i sequelize --save

@hpgt 其实只是想一起探讨下,并不是说随便凭感觉发个言论丢在这里就完了

@huanz 既然是探讨的话那我就说点比较烦的

import {Entity, PrimaryGeneratedColumn, Column} from "typeorm";
@Entity()
export class User {
 @PrimaryGeneratedColumn()
 id: number;
 @Column()
 firstName: string;
 @Column()
 lastName: string;
 @Column()
 age: number;
}

和sequelize比较,原本好好的属性,非要加个@Entity(),@Column()这玩意儿来说明吗? 再说nest.js

@Controller('cats')
export class CatsController {
 @Post()
 create(@Res() res, @Body() createCatDto: CreateCatDto) {
 // TODO: Add some logic here
 res.status(HttpStatus.CREATED).send();
 }
 @Get()
 findAll(@Res() res) {
 res.status(HttpStatus.OK).json([]);
 }
}

create(req:any, res:any)这样有什么不好,非要create(@Res() res, @Body() createCatDto: CreateCatDto)加注解,乍看之下,我要获取ip,url怎么办,或许它有自己的方法,但是又不知道需要怎么整才行。我是觉得保持简单最好,不要徒增学习成本

@hpgt 我来贴下sequelize的话一个模型是怎么定义:

const User = sequelize.define('user', {
 id: {
 	type: Sequelize.INTEGER,
 primaryKey: true,
 autoIncrement: true,
 },
 firstName: Sequelize.TEXT,
 lastName: Sequelize.DATE
})

而在 typeorm 里面,这些可能都变成了一个装饰器(所以只从这里来看我觉得两者差不多的吧?)。

如果我们一些表有些共同字段,在 typeorm 可以这样:

import { Column, PrimaryGeneratedColumn, CreateDateColumn, UpdateDateColumn } from 'typeorm';
export abstract class Base {
 @PrimaryGeneratedColumn()
 id: number;
 @CreateDateColumn()
 createAt: Date;
 @UpdateDateColumn()
 updateAt: Date; // 该字段会自动在update的时候更新
}
@Entity()
export class Comment extends Base {
}

还有 embedded-entities

除此之外,你可以继续看下对与各种关系的处理等等,我觉得都是十分优雅的

ps: 我对 sequelize 并不是非常了解,只是过了一下它的文档,所以并不是十分了解它的特性

@hpgt 其次对于 nestjs,我觉得 res \ req \ body 都是很明确的吧? 你要获取 ip url 肯定是在请求里面啊,不应该你用 express 你知道去 req 里面取数据,在nestjs你不知道?

有讨论组什么的吗,这样翻译方便分配工作,不然大家各自翻各自的很容易重复

@fwgood 添加issue,名称为翻译的文件名,添加label 正在翻译,如:https://github.com/huanz/typeorm.github.io/issues/2 翻译完成后发pr

因为使用nestjs而关注到typeorm,目前已经在3个项目中实践了。总得来说我是非常喜欢注解的这种方式。因为更贴合Spring的风格。在用了React之后,更加觉得注解是优化代码、提升可读性的不二法门。

404 There isn’t a GitHub Pages site here.

@yuedun 那你可能 没见过 hibernate 里面疯狂使用annotation

学习java和hibernate的模式,我就已经没一点兴趣了。

真的没必要加太多的注解,express、koa里本来就一目了然。

所以说你们不懂, 注解, 依赖注入等, 是为了解耦, 关注点分离, 你们盖鸟窝当然怎么直接怎么来了

@chunjiu 我估计你无法理解, 为什么要 关注点分离, 依赖注入, 面向切面, 解耦. 当然, 如果只是盖个鸟窝, 根本不需要理解这些

@151263 我发现你有些上冈上线,说话就好好说话,喜欢带枪?nodejs本来就是快速、简单开发,谁用nodejs来写很大型项目?全中国称得上大项目的又能有多少个,那么多从业人员当中做大项目的人员比例又是多少?

非得每个字段都 colum 声明一下么

@hxh1246996371 colum 声明一下, 启动的时候它就会帮你自动建表, 自动修改字段, 增加,删除修改字段, 索引 等

nest的确是滥用了,typeorm还好

@151263 说的很实清楚了;说滥用的,说白了就是不懂,或者嘛合适你的场景真的不要这么多。 注解信息是一个高度抽象化的描述; 当然相对来说 typeorm、nestjs 都是牛刀级别的了,如果你只想削削你的水果,express 、sequelize 很适合你。 typeorm、nestjs 对node 来说是在重型应用方面一个很好的补充,借用@151263 的话来说你想 ‘ 关注点分离, 依赖注入, 面向切面, 解耦.(例如你想积累一些通用的基础代码库、你想多人合作开发、你想组件化开发、你想业务分离、插件化等)’ 那请用上,一点也不滥用。

@151263 建表不应该是migration的事吗

回到顶部

AltStyle によって変換されたページ (->オリジナル) /