TypeORM 一对一关联篇:掌握实体关系与 JoinColumn

用户和身份证是一对一的关系,即一个用户对应一个身份证。

一对一是通过外键来存储这种关系的

创建数据库并初始化 TypeORM 项目

ts 复制代码
# 创建数据库
create database typeorm_one_to_one_relation_mapping;
​
# 初始化 TypeORM 项目
npx typeorm@latest init --name typeorm_one_to_one_relation_mapping --database mysql

修改数据库连接配置

具体参数配置可查看上一篇文章: TypeORM 基础篇:项目初始化与增删改查全流程

ts 复制代码
import "reflect-metadata"
import {DataSource} from "typeorm"
import {User} from "./entity/User"
​
export const AppDataSource = new DataSource({
  type: "mysql",
  host: "localhost",
  port: 3306,
  username: "root",
  password: "admin",
  database: "typeorm_one_to_one_relation_mapping",
  synchronize: true,
  logging: true,
  entities: [User],
  migrations: [],
  subscribers: [],
  poolSize: 10,
  connectorPackage: "mysql2",
  extra: {
    authPlugins: 'sha256_password'
  }
})
​

安装 mysql2 驱动包

bash 复制代码
npm i mysql2 --save

运行项目

bash 复制代码
npm run start

可以看到建表和插入数据的 sql 执行日志。


创建身份证表

创建 IdCard Entity

ts 复制代码
import {Column, Entity, PrimaryGeneratedColumn} from "typeorm";
​
@Entity({
  name: "id_card"
})
export class IdCard {
  @PrimaryGeneratedColumn()
  id: number;
​
  @Column({
    length: 50,
    comment: "身份证号"
  })
  cardNumber: string;
}

在 DataSource 的 entities 中引入

ts 复制代码
import "reflect-metadata"
import {DataSource} from "typeorm"
import {User} from "./entity/User"
import {IdCard} from "./entity/IdCard";
​
export const AppDataSource = new DataSource({
  // ...
  entities: [User, IdCard],
  // ...
})
​

重新执行 npm run start 可以看到建表 sql


user 表和 IdCard 表建立一对一关联

IdCard 表中维护外键

在 IdCard Entity 中添加 user 列

  • @OneToOne() 指定它和 User 是一对一的关系
  • @JoinColum() 指定外键列在 IdCard 表中维护

重新执行 npm run start 可以看到日志:

设置外键级联关系

通过主表中如何访问到从表

如何在 user 表中访问 idCard 表?同样通过 @OneToOne 装饰器。

通过第二个参数告诉 typeorm,外键是另一个 Entity 的哪个属性。

查询主表带出从表数据

ts 复制代码
const users = await AppDataSource.manager.find(User, {
  relations: {
    idCard: true
  }
})
console.log(users)

查询结果:


增删改查

新增

ts 复制代码
import {AppDataSource} from "./data-source"
import {User} from "./entity/User"
import {IdCard} from "./entity/IdCard";
​
AppDataSource.initialize().then(async () => {
​
  const user = new User()
  user.firstName = "San"
  user.lastName = "Shi"
  user.age = 30
  await AppDataSource.manager.save(user)
​
  const idCard = new IdCard()
  idCard.cardNumber = "1234567890"
  idCard.user = user
​
  await AppDataSource.manager.save(user)
  await AppDataSource.manager.save(idCard)
​
}).catch(error => console.log(error))
  • 创建 user 和 idCard 对象
  • 设置 idCard.user 为 user,也就是建立关联关系

分别保存 user 和 idCard。

再次执行 npm run start。

查询数据库中可以看到输入插入成功

优化

上边先保存了 user 又保存了 idCard,能不能让它自动按照关联关系来保存呢?

可以通过 @OneToOne() 指定 cascade 为 true

cascade 用来告诉 typeorm 当增删改一个 Entity 时,是否级联增删改它关联的 Entity。

这样只需要保存 idCard 就可以了。


查询

在 find 方法中通过 relations 来声明关联查询,反之不进行关联查询。


修改

指定 id 就是修改,不指定 id 就是新增。


删除

因为前面设置的级联关系是 cascade,所以只要删除了 user,那关联的 idCard 就会跟着被删除。

ts 复制代码
const users = await AppDataSource.manager.find(User)
  console.log('删除前查询: ', users)
  await AppDataSource.manager.delete(User, 1)
  const usersAfterDelete = await AppDataSource.manager.find(User)
  console.log('删除后查询: ', usersAfterDelete)

执行日志

相关推荐
马克学长5 小时前
SSM青岛恒星科技学院机房管理系统0k0u9(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面
数据库·科技·ssm 框架·教育信息化·机房管理系统·青岛恒星科技学院
7***68437 小时前
Spring Boot 从 2.7.x 升级到 3.3注意事项
数据库·hive·spring boot
L***d6707 小时前
Spring Boot 各种事务操作实战(自动回滚、手动回滚、部分回滚)
java·数据库·spring boot
java_logo7 小时前
MySQL Server Docker 容器化部署指南
linux·运维·数据库·docker·容器
likuolei7 小时前
XSL-FO 软件
java·开发语言·前端·数据库
p***95007 小时前
Springboot3 Mybatis-plus 3.5.9
数据库·oracle·mybatis
CS_浮鱼7 小时前
【MySQL】InnoDB存储引擎
数据库·mysql
合作小小程序员小小店7 小时前
桌面开发,在线%信息管理%系统,基于vs2022,c#,winform,sql server数据。
开发语言·数据库·sql·microsoft·c#
q***18847 小时前
解决phpstudy无法启动MySQL服务
数据库·mysql·adb
e***95647 小时前
【HTML+CSS】使用HTML与后端技术连接数据库
css·数据库·html