当前位置: 首页 > 文档资料 > Sequelize ORM 实践 >

3.7 关系嵌套

优质
小牛编辑
121浏览
2023-12-01

在之前,创建一个有依赖其他模型的模型的时候,我们需要提前把依赖的模型给建立好,这样就非常麻烦,是不是有那么一种方法,在一个 create 里面把其他的依赖模型一起新建出来呢?

这一小节就来解决这个问题。这其中有一个坑(关于命名),让我调试的头皮发麻。

Book 与 User 的 belongs 关系

1. 保存关系

来到 book.ts , 首先保存一下 this.belongsTo 的返回值,在 d.ts 中的定义中写的返回的 void,其实并不是你可以log 一下,是一个包含 source target option 的对象。

(Book as any).associate = function(this: typeof Book, models: any){
   (Book as any).User = this.belongsTo(models.User);
}

2.添加代码提示

export interface BookAttributes {
  status: "inSale" | "noSale";
  description: string;
  title: string;
  author: string;

  User?: UserAttributes | Sequelize.BelongsToGetAssociationMixin<UserInstance>;
}

特别注意这里是 User,为什么是 User? 因为这个值要跟你 define 传入的第一个参数保持一致。 笔者一直认为是小写,自己写了很多例子验证就是没有正常运行,花了2天时间去找到这个问题。

这里的 User 跟实例的 User 重名了所有我们要用一个联合类型来保持父类接口的兼容性。

User 与 Tag 的 belongsToMany

belongsToMany、hasMany、hasOne 都是这样使用的。

1.保存关系

来到 user.ts

(User as any).associate = function(this: typeof User,models){
   // this.hasOne(models.Book);
   (User as any).Book = this.hasMany(models.Book);
   (User as any).Tag = this.belongsToMany(models.Tag, { through: 'user_tag'});
}

注意,这里的 this.hasMany 跟之前的第一个没有任何关系,因为此时的主体是 User,这是为了让 User.create 可以新建 Book 对象。

即:(实验请根据后面内容自行加上代码提示)

User.create({
    .....,
    Book: {
        .....,
    }
}, { include: [User.Book] })

2.代码提示

export interface UserAttributes {
  email: string;
  name: string;
  Tags?: TagAttributes[];
}

在 Tag 模型里面的define函数的第一个参数是 Tag,所以这里是 Tags,因为是多的一方,所以这里数组,并且与实例属性没有冲突,所以就不需要再做什么其他的。

验证

index.ts

await Book.create({
 author: '213',
 description: '213',
 status: "inSale",
 title: '12322',
 User: {
   email: '1733996866@qq.com',
   name: 'some',
   Tags: [
     {name: '123'},
     {name: '22'}
   ]
 }
},{
 include: [{
   association: (Book as any).User,
   include: [(User as any).Tag]
 }]
});

运行之后便可看到数据库中有数据,记得同步数据库表哦。

假如只有一项相关联的,这样写即可。假如是嵌套,像上面这样写即可,可以一直嵌套。

include: [(User as any).Tag]