- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我正在尝试熟悉 Angular 2 的 ChangeDetectionStrategy.OnPush
性能提升(如 here 所述)。但我这里有古玩盒。
我有父 AppComponent
:
@Component({
selector: 'my-app',
template: `<h1>
<app-literals [title]="title" [dTitle]="dTitle"></app-literals>
<input [value]="title.name"/>
</h1>
`
})
export class AppComponent implements OnInit {
title = { name: 'original' };
dTitle = { name: "original" };
constructor(private changeDetectorRef : ChangeDetectorRef) {
}
ngOnInit(): void {
setTimeout(() => {
alert("About to change");
this.title.name = "changed";
this.dTitle = { name: "changed" };
}, 1000);
}
}
和子 LiteralsComponent
组件:
@Component({
selector: 'app-literals',
template: ` {{title.name}}
{{dTitle.name}}`,
changeDetection: ChangeDetectionStrategy.OnPush
})
export class LiteralsComponent implements OnInit {
@Input('title') title;
@Input('dTitle') dTitle;
constructor() { }
ngOnInit() {
}
}
我认为将策略设置为 OnPush
会使 Angular 仅反射(reflect)引用的更改,但在示例中我尝试更改(变异)对象的属性, Angular 仍然反射(reflect)它。
this.title.name = "changed";
不应被检测到(因此 UI 不应不反射(reflect)更改)。
这是关于 plunker 的案例
怎么来的?怎样做才是对的?
最佳答案
如果我没理解错,你问的是为什么绑定(bind)值在 LiteralsComponent
模板中更新,即使你不修改引用 title
,而是改变对象.
简短的回答是因为您修改了两者:
this.title.name = "changed";
this.dTitle = {name: "changed"};
在 AppComponent.ngOnInit
中。如果您只修改 this.title.name = "changed"
,您将看到该模板没有更新。
然而,这是一个非常有趣的问题,需要详细探讨
让我们首先从没有 this.dTitle
的 this.title
开始。
首先要了解的是,当您在模板中指定以下内容时:
{{title.name}}
这是 Angular 所做的。它尝试在当前组件实例上找到 title
对象,然后从中获取 name
属性并将其反射(reflect)在 DOM 中。但具有以下配置:
class AppComponent {
title = { name: 'original' }
ngOnInit(): void {
setTimeout(() => {
alert("About to change");
this.title.name = "changed";
}, 1000);
}
}
class LiteralsComponent {
@Input() title;
}
title
对象在两个组件中相同(指向相同的内存位置)。
因此,当 Angular 为 LiteralsComponent
组件运行更改检测时,它会访问您在此处在 AppComponent
中更改的同一对象:
ngOnInit(): void {
setTimeout(() => {
alert("About to change");
this.title.name = "changed";
}, 1000);
}
这里有趣的观察是变化未被检测到,无论是使用 OnPush
还是没有它:
class LiteralsComponent {
@Input() title;
ngOnChanges(changes) {
// will be triggered only for the first CD cycle,
// and won't be triggered when `title` is updated
}
}
现在,最后一件事是了解 DOM 何时更新。根据this article , 它在当前组件的 CD 期间更新。这意味着如果未选中当前组件,则不会更新 DOM。所以我们为 LiteralsComponent
指定 onPush
:
changeDetection: ChangeDetectionStrategy.OnPush,
View 不会更新。
但是,它已在您的问题中更新。为什么?
这就是 dTitle
发挥作用的地方。使用此属性,您实际上是在修改引用,Angular 检测绑定(bind)更改 并为 LiteralsComponent
组件运行 CD。我们在上面了解到,当 CD 运行时,DOM 会更新。所以 Angular 也会更新 {{title.name}}
,因为它指向 AppComponent
中的同一个对象,虽然它没有检测到它被改变了.
关于angular - ChangeDetectionStrategy.OnPush 不符合我的预期,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43539254/
我们如何让 SwiftUI 对象,尤其是 Image,符合 Hashable 协议(protocol)? 我知道它们符合 Equatable 协议(protocol),所以主要问题是如何获取哈希值,或
我遇到了一些符合 AVAudioPlayerDelegate 的奇怪问题。以下正是我在一个全新的 Xcode 项目中所拥有的: import UIKit import AVFoundation cla
我一辈子都弄不明白为什么我会收到此类不符合 NSCoding 协议(protocol)的错误。也许另一双眼睛会有所帮助。我试图添加注释以明确每个函数在做什么。 import Foundation im
关闭。这个问题需要details or clarity .它目前不接受答案。 想改进这个问题吗? 通过 editing this post 添加细节并澄清问题. 关闭 9 年前。 Improve t
所有现代浏览器都理解 HTML,所以除了在键盘最右侧编写更多字符之外,兼容 XHTML 的意义何在。 最佳答案 没有一点我能想到的。 W3C 已经取消了 XHTML 2.0,尽管应该有一个 XHTML
我正在设计一个订单系统,状态设计模式似乎很合适,因为订单可以更改其状态,从而更改订单允许的功能。下面是我的基本类图: 我不喜欢这种方法,因为客户端无法查看某个方法是否受支持并且违反了里氏原则。我在下面
我正在考虑使用图形数据库来存储 IFC数据。理想情况下,数据库应该提供一种方法来定义 IFC 架构中定义的所有规则类型。但是,我不认为有任何这样的数据库,因为 IFC 中的某些规则类型非常复杂并且需要
我所在的组织必须满足 FISMA 对启用 FIPS 的系统的要求。我正在尝试做的一件事是为我们的密码实现哈希算法。我对此有很多选择:SHA-2、MD5、bcrypt(使用 Blowfish)、RIPE
我正在尝试实现我的自定义 CoreData Carpark 实体以符合 MKAnnotation,就像我们如何使 class 对象符合 >MKAnnotation. 我根据以下帖子调整了我的实现:th
我在 project-Swift.h 文件中收到名为“CBCentralManagerDelegate”的 No 类型或协议(protocol)。不知道我在这里哪里出错了。我认为这与 swift.h
我正在尝试读取之前写入 NVM 闪存的变量的值。 我的代码是: uintptr_t address = getAddress(); //[MISRA C++ Rule 5-2-8] cast from
所以我有这个练习要解决。我必须创建第一个。一个名为 Printable 的接口(interface),它有一个 put() 方法,该方法将接受实现 Comparable 的对象。 完成 interfa
我的问题涉及 IEEE 754 单精度数字。假设我有一个结构: typedef struct __ieee754 { int sign; int exponent; int mant
我需要使用 map,键为 uint32_t,值为 Meshes。我希望将网格布局在连续的内存中以 boost 性能,因为它们将经常被连续访问。 我想知道有哪些内存分配器库可以提供以下内容; 分配给连续
在处理小对象时,有哪些分配器可用于 STL。我已经尝试过使用 Boost 的池分配器,但没有得到任何性能提升(实际上,在某些情况下性能下降相当大)。 最佳答案 你没有说你使用的是什么编译器,但它可能带
我想做什么 我有一种划分事物的方法。此方法不会对数组进行完全排序;它只是简单地对数组进行分区,以便一侧的所有元素(某些预先确定的“中心”或“中点值”——但它不必导致均匀拆分)小于“中心”和另一侧的所有
假设我需要开发一个 REST 银行应用程序允许创建/销毁银行账户以及对帐户进行以下操作:withdraw/credit/getBalance。 创建帐户 PUT/银行/约翰 这里我使用 PUT 而不是
假设我有一个 struct 符合我的模型的 Equatable,如下所示: struct Model: Equatable { var a: Int = 0 var b: String
我目前正在研究 Decodable、Encodable 和 friend ,试图理解它背后的“魔法”。 以下是我发现不寻常的一件事: class Person: Decodable { var n
在 Swift 书中,枚举的例子很好用 enum CompassPoint: String { case north, south, east, west } var northCom = C
我是一名优秀的程序员,十分优秀!