Angular Signals 是 Angular 16 引入的新功能,专注于响应式编程,它通过显式的依赖追踪来提高性能和开发体验。与之前的变更检测机制不同,Signals 提供了一种更简洁且高效的方式来处理数据流和组件更新。
Signals 的核心概念
- Signal (信号)
Signal 是一个持有某个值的对象,当这个值改变时,会通知所有依赖于该值的计算函数或组件进行更新。可以将 Signal 理解为一个"可观察的值"。
javascript
import { signal } from '@angular/core';
const count = signal(0); // 创建一个信号,初始值为0
count(); // 获取信号的值,输出0
count.set(1); // 设置新的值,所有依赖此信号的地方都会更新
count.update(value => value + 1); // 对现有值进行更新,输出2
- Computed (计算属性)
Computed 是一种基于其他信号值计算出来的信号。它依赖其他信号,当依赖的信号发生变化时,Computed 信号会自动重新计算。
javascript
const doubleCount = computed(() => count() * 2); // 当count变化时,doubleCount会自动更新
- Effect (副作用)
Effect 是一段响应式代码,它依赖一个或多个信号。当这些信号发生变化时,Effect 会自动运行。这类似于 RxJS 中的 subscribe,但不需要手动订阅或取消订阅。
javascript
effect(() => {
console.log('Count changed:', count());
});
Signals 与现有 Angular 状态管理的对比
- 与 RxJS 的区别:RxJS 的 observable 是异步流数据,而 Signals 更类似于同步的、单一值的容器。Signals 更轻量,主要用于局部状态管理,而 RxJS 在处理异步流或复杂操作时依然占据优势。
- 与 Angular 变更检测机制的区别:传统的 Angular 依赖 Zone.js 进行变更检测,当输入属性或服务中的数据改变时,Angular 会重新检查整个组件树。但 Signals 通过精确追踪数据依赖关系,仅更新受影响的部分,从而减少不必要的组件重渲染,提高性能。
Angular Signals 的实际应用
- 管理组件内部状态
Signals 可以用来管理组件的本地状态,而无需使用复杂的服务或共享状态管理库。
javascript
@Component({
selector: 'app-counter',
template: `
<button (click)="increment()">Increment</button>
<p>{{ count() }}</p>
`
})
export class CounterComponent {
count = signal(0);
increment() {
this.count.update(c => c + 1);
}
}
- 实现局部响应式逻辑
Signals 和 Effects 可以用来处理局部的响应式逻辑,比如在表单控件中实时计算值。
javascript
@Component({
selector: 'app-form',
template: `
<input [ngModel]="value()" (ngModelChange)="value.set($event)" />
<p>Value length: {{ valueLength() }}</p>
`
})
export class FormComponent {
value = signal('');
valueLength = computed(() => this.value().length);
}
- 替代服务中的行为对象
在服务中管理共享状态时,Signal 也可以取代传统的 BehaviorSubject 或 ReplaySubject,无需手动订阅或更新视图。
javascript
@Injectable({ providedIn: 'root' })
export class CounterService {
private count = signal(0);
getCount() {
return this.count();
}
increment() {
this.count.update(c => c + 1);
}
}
何时使用 Signals
- 组件局部状态管理:如果一个状态只在一个组件或局部范围内使用,Signals 是非常合适的选择。
- 简单的同步响应式数据流:当你的状态更新逻辑不涉及复杂的异步操作时,使用 Signals 可以简化代码。
- 性能优化:对于一些频繁更新的局部状态(如表单输入或小部件),使用 Signals 可以减少不必要的变更检测,提高应用性能。
总结
Angular Signals 提供了一种轻量级的响应式状态管理方式,尤其适用于局部状态或简单的同步数据流。相比 RxJS 和传统的 Angular 变更检测,Signals 更加直接和高效,减少了对全局变更检测的依赖。