[TypeORM]官方示例详解-Vol.1 创建数据库连接和第一个实体映射类

前言

本系列将对TypeORM中官方提供的33个示例逐一进行说明讲解。
本期是第一期,官方示例地址https://github.com/typeorm/typeorm/tree/master/sample/sample1-simple-entity
本期主要用于演示如何编写一个数据库连接和如何利用TypeORM的注解(装饰器)编写数据库表的实体映射。

一、 创建数据库连接

利用TypeORM创建一个数据库连接十分容易,只需要使用createConnections函数并传入必要的连接配置的参数便可以对数据库进行连接和操作。

export async function createConnections(options?: ConnectionOptions[]): Promise<Connection[]>

官方示例中使用了MySQL数据库的连接配置:

const options: ConnectionOptions = {
    "name": "mysql",
    "type": "mysql",
    "host": "localhost",
    "port": 3306,
    "username": "test",
    "password": "test",
    "database": "test",
    synchronize: true,
    entities: [Post]
};

其中name,type,host,port,passowrd,database参数基本上是现在所有数据库配置都有的,比较特殊的是synchronizeentities这两个参数。

synchronize与migration

在解释说明synchronize,我们先引入一个概念叫Migration即迁移。在数据库的实体映射发生变化的时候,TypeORM会根据变化对现有的视图Schema进行修改、增加、删除调整至当前实体映射所期望的样子。我们可以把每个数据库视图之间的变化称为Migration
synchronize 则标识了是否需要在每次应用启动后自动对数据库的视图Schema进行自动的Migration。通常这个开关在我们开发的时候都是默认打开的,但是在生产环境推荐关闭,免得某一个实体定义写错,直接就把一些表给删除了。

entities

entities则表明要引入的实体文件,一般来说更推荐大家使用路径的形式进行引入,一个一个引入实在会手疼……

    __dirname + '/entity/*.{ts,js}',

这里有一个很大的坑,便是如果通过路径文件引入一定记得后缀要同时写上js和ts两个后缀格式,因为ts在编译后会变成js文件,但是数据库连接启动读取的部分并不会变化从而会导致读不到编译后的js文件导致错误。

创建连接

了解了主要的配置文件参数后,我们便可调用createConnection(options)进行数据库连接十分简单。

createConnection(options)
    .then(async connection => {
        // 成功连接后通过connection变量操作数据库
    }, error => console.log("Cannot connect: ", error));

二、 实体映射类定义

官方示例中给出了一个Post即我们常见的博客发帖的帖子的类定义。
我们通过TypeORM的实体Entity去映射数据库中的对应的表Table。
首先我们来看下Post Entity的代码。

@Entity("sample01_post")
export class Post {

    @PrimaryColumn("integer")
    @Generated()
    id: number;

    @Column()
    title: string;

    @Column()
    text: string;

    @Column({ nullable: false })
    likesCount: number;

}

首先,我们说下最重要的装饰器@Entity,一个class如果被@Entity装饰了,则表示这个class是一个实体类,如果我们在数据库连接的时候正确的将它放入了entitis数组中那么TypeORM便会将其纳入实体管理中。
@Entity()如果我们不指定表命,一般这种无参形式就可以了,TypeORM会自动将类名变为小写加下划线形式创建表名。不如这个例子中的Post便会变为改名post创建一个表。

其次,每一个类的属性上的注解@Column(),则是用于标注这个属性需要被吃持久化为一个表中的列,其无参情况下命名方式与@Entity()是一样的。

最后,说一下@PrimaryColumn("integer"),@Generated(),这两个主键上的注解。@PrimaryColumn("integer")是表示id这一列是主键列且主键的类型是整形,这边在装饰器中的类型通常使用的是数据库类型,而映射成ts的属性id类型则是number。@Generated()则表示列是一个主键的创建策略。下面我简单的介绍下TypeORM中关于主键创建策略的说明。

@Generated的主键创建策略

TypeORM中提供的主键创建策略主要有几种:

  • increment:自增,一般默认都使用这个
  • uuid: string类型的uuid,类似Oracle没有自增便需要使用其他策略了
  • rowid: number类型的rowid

当我们完成了Post实体的定义,并且在应用启动时、创建数据库连接时并正确的将Post的加载到TypeORM的配置中去后,在应用启动时,系统便会自动创建一个名为sample01_post表。

三、通过connection创建一条记录

首先让我们回顾下,创建连接后的Promise如果进行操作。

createConnection(options)
    .then(async connection => {
        // 成功连接后通过connection变量操作数据库
    }, error => console.log("Cannot connect: ", error));

可以在then方法内获取连接成功后的connection变量,并进行操作。
我们尝试像数据库中新增一个Post记录。
跟其他语言的ORM一样,我们并不直接通过INSERT语句去操作数据库,而是创建一个实体类,并对其进行保存操作。
首先,我们通过代码创建一个Post对象。

    let post = new Post();
    post.text = "Hello how are you?";
    post.title = "hello";
    post.likesCount = 100;

接着,我们通过通过数据库连接获取Repository仓库类,可能有些同学对Repository比较陌生,其实Repository可以简单理解成我们操作Entity使用的工具类,类似DAO层。

    let postRepository = connection.getRepository(Post);

最后,最后我们通过Repositorysave方法保存刚刚创建的Post对象。

    postRepository
        .save(post)
        .then(post => console.log("Post has been saved: ", post))
        .catch(error => console.log("Cannot save. Error: ", error));

完成的代码如下:

createConnection(options).then(async connection => {
    let post = new Post();
    post.text = "Hello how are you?";
    post.title = "hello";
    post.likesCount = 100;

    let postRepository = connection.getRepository(Post);

    postRepository
        .save(post)
        .then(post => console.log("Post has been saved: ", post))
        .catch(error => console.log("Cannot save. Error: ", error));

}, error => console.log("Cannot connect: ", error));

这样我们便完成了第一个基于TypeORM的数据库操作的编码工作。通过TyepORM可以避免了通过编写SQL语句模板操作数据库的工作,大大提高了编码效率。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,271评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,275评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,151评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,550评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,553评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,559评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,924评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,580评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,826评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,578评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,661评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,363评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,940评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,926评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,156评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,872评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,391评论 2 342

推荐阅读更多精彩内容