# 实体监听器和订阅者

# 监听器

任何实体都可以使用具有侦听特定实体事件的自定义逻辑的方法。 你必须使用特殊装饰器标记这些方法,具体取决于要侦听的事件。

# @AfterLoad

你可以在实体中定义具有任何名称的方法,并使用@AfterLoad标记它,TypeORM 将在每次实体时调用它 使用QueryBuilder或 repository/manager 查找方法加载。 例如:

@Entity()
export class Post {
  @AfterLoad()
  updateCounters() {
    if (this.likesCount === undefined) this.likesCount = 0;
  }
}
1
2
3
4
5
6
7

# @BeforeInsert

你可以在实体中定义具有任何名称的方法,并使用@BeforeInsert标记它,并且 TypeORM 将在使用 repository/manager save插入实体之前调用它。 例如:

@Entity()
export class Post {
  @BeforeInsert()
  updateDates() {
    this.createdDate = new Date();
  }
}
1
2
3
4
5
6
7

# @AfterInsert

你可以在实体中定义具有任何名称的方法,并使用@AfterInsert标记它,并且在使用 repository/manager save插入实体后,TypeORM 将调用它。

@Entity()
export class Post {
  @AfterInsert()
  resetCounters() {
    this.counters = 0;
  }
}
1
2
3
4
5
6
7

# @BeforeUpdate

你可以在实体中定义具有任何名称的方法,并使用@BeforeUpdate标记它,并且 TypeORM 将在使用 repository/manager save更新现有实体之前调用它。 但请记住,只有在模型中更改信息时才会出现这种情况。 如果运行save而不修改模型中的任何内容,@BeforeUpdate@AfterUpdate将不会运行。 例如:

@Entity()
export class Post {
  @BeforeUpdate()
  updateDates() {
    this.updatedDate = new Date();
  }
}
1
2
3
4
5
6
7

# @AfterUpdate

你可以在实体中定义具有任何名称的方法,并使用@AfterUpdate标记它,并且在使用 repository/manager save更新现有实体后,TypeORM 将调用它。 例如:

@Entity()
export class Post {
  @AfterUpdate()
  updateCounters() {
    this.counter = 0;
  }
}
1
2
3
4
5
6
7

# @BeforeRemove

你可以在实体中定义具有任何名称的方法,并使用@BeforeRemove标记它,并且 TypeORM 将在使用 repository/manager remove删除实体之前调用它。 例如:

@Entity()
export class Post {
  @BeforeRemove()
  updateStatus() {
    this.status = "removed";
  }
}
1
2
3
4
5
6
7

# @AfterRemove

你可以在实体中定义一个具有任何名称的方法,并使用@AfterRemove标记它,TypeORM 将在使用 repository/manager remove删除实体后调用它。 例如:

@Entity()
export class Post {
  @AfterRemove()
  updateStatus() {
    this.status = "removed";
  }
}
1
2
3
4
5
6
7

# @BeforeSoftRemove

你可以在实体中定义具有任何名称的方法,并使用@BeforeSoftRemove标记它,并且 TypeORM 将在使用 repository/manager softRemove删除实体之前调用它。 例如:

@Entity()
export class Post {
  @BeforeSoftRemove()
  updateStatus() {
    this.status = "soft-removed";
  }
}
1
2
3
4
5
6
7

# @AfterSoftRemove

你可以在实体中定义一个具有任何名称的方法,并使用@AfterSoftRemove标记它,TypeORM 将在使用 repository/manager softRemove删除实体后调用它。 例如:

@Entity()
export class Post {
  @AfterSoftRemove()
  updateStatus() {
    this.status = "soft-removed";
  }
}
1
2
3
4
5
6
7

# @BeforeRecover

你可以在实体中定义具有任何名称的方法,并使用@BeforeRecover标记它,并且 TypeORM 将在使用 repository/manager recover删除实体之前调用它。 例如:

@Entity()
export class Post {
  @BeforeRecover()
  updateStatus() {
    this.status = "recovered";
  }
}
1
2
3
4
5
6
7

# @AfterRecover

你可以在实体中定义一个具有任何名称的方法,并使用@AfterRecover标记它,TypeORM 将在使用 repository/manager recover删除实体后调用它。 例如:

@Entity()
export class Post {
  @AfterRecover()
  updateStatus() {
    this.status = "recovered";
  }
}
1
2
3
4
5
6
7

# 订阅者

将类标记为可以侦听特定实体事件或任何实体事件的事件订阅者。 使用QueryBuilder和存储库/管理器方法触发事件。 例如:

@EventSubscriber()
export class PostSubscriber implements EntitySubscriberInterface<Post> {
  /**
   * 表示此订阅者仅侦听Post事件。
   */
  listenTo() {
    return Post;
  }

  /**
   * 插入post之前调用。
   */
  beforeInsert(event: InsertEvent<Post>) {
    console.log(`BEFORE POST INSERTED: `, event.entity);
  }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16

你可以从EntitySubscriberInterface实现任何方法。 要监听任何实体,你只需省略listenTo方法并使用any

@EventSubscriber()
export class PostSubscriber implements EntitySubscriberInterface {
  /**
   * 在实体插入之前调用。
   */
  beforeInsert(event: InsertEvent<any>) {
    console.log(`BEFORE ENTITY INSERTED: `, event.entity);
  }
}
1
2
3
4
5
6
7
8
9