An-observeValueForKeyPath:ofObjectを変更コンテキスト:したなら、受け取ったメッセージが取り扱っておりません
-
29-09-2019 - |
質問
私は比較的新しいKVOがあるので良い機会になると私は違反するいくつかの基本的な規則です。を使用している.
私のアプリがクラッシュ以下のメッセージ:何かの理解はなぜCGImage関わってもらってい観測値を設定しMeasurementPointerオブジェクトです。
*** Terminating app due to uncaught exception 'NSInternalInconsistencyException', reason: '<CGImage 0x276fc0>: An -observeValueForKeyPath:ofObject:change:context: message was received but not handled.
Key path: measurementDescriptor
Observed object: <MeasurementPointer: 0x8201640> (entity: MeasurementPointer; id: 0x8200410 <x-coredata://EBEE0687-D67D-4B03-8C95-F4C60CFDC20F/MeasurementPointer/p75> ; data: {
measurementDescriptor = "0x260fd0 <x-coredata://EBEE0687-D67D-4B03-8C95-F4C60CFDC20F/MeasurementDescriptor/p22>";
})
Change: {
kind = 1;
new = "<MeasurementDescriptor: 0x262530> (entity: MeasurementDescriptor; id: 0x260fd0 <x-coredata://EBEE0687-D67D-4B03-8C95-F4C60CFDC20F/MeasurementDescriptor/p22> ; data: {\n measurementName = Temperature;\n measurementUnits = \"\\U00b0C\";\n sortString = nil;\n})";
}
Context: 0x0'
*** Call stack at first throw:
(
0 CoreFoundation 0x30897ed3 __exceptionPreprocess + 114
1 libobjc.A.dylib 0x3002f811 objc_exception_throw + 24
2 CoreFoundation 0x30897d15 +[NSException raise:format:arguments:] + 68
3 CoreFoundation 0x30897d4f +[NSException raise:format:] + 34
4 Foundation 0x34a13779 -[NSObject(NSKeyValueObserving) observeValueForKeyPath:ofObject:change:context:] + 60
5 Foundation 0x349b6acd NSKeyValueNotifyObserver + 216
6 Foundation 0x349b6775 NSKeyValueDidChange + 236
7 Foundation 0x349ae489 -[NSObject(NSKeyValueObserverNotification) didChangeValueForKey:] + 76
8 CoreData 0x3165b577 _PF_ManagedObject_DidChangeValueForKeyIndex + 102
9 CoreData 0x3165ac51 _sharedIMPL_setvfk_core + 184
10 CoreData 0x3165dc83 _svfk_0 + 10
11 SPARKvue 0x000479f1 -[MeasurementViewController doneAction:] + 152
12 CoreFoundation 0x3083f719 -[NSObject(NSObject) performSelector:withObject:withObject:] + 24
13 UIKit 0x31eb1141 -[UIApplication sendAction:to:from:forEvent:] + 84
14 UIKit 0x31f08315 -[UIBarButtonItem(UIInternal) _sendAction:withEvent:] + 92
15 CoreFoundation 0x3083f719 -[NSObject(NSObject) performSelector:withObject:withObject:] + 24
16 UIKit 0x31eb1141 -[UIApplication sendAction:to:from:forEvent:] + 84
17 UIKit 0x31eb10e1 -[UIApplication sendAction:toTarget:fromSender:forEvent:] + 32
18 UIKit 0x31eb10b3 -[UIControl sendAction:to:forEvent:] + 38
19 UIKit 0x31eb0e05 -[UIControl(Internal) _sendActionsForEvents:withEvent:] + 356
20 UIKit 0x31eb1453 -[UIControl touchesEnded:withEvent:] + 342
21 UIKit 0x31eafddd -[UIWindow _sendTouchesForEvent:] + 368
22 UIKit 0x31eaf757 -[UIWindow sendEvent:] + 262
23 UIKit 0x31eaa9ff -[UIApplication sendEvent:] + 298
24 UIKit 0x31eaa337 _UIApplicationHandleEvent + 5110
25 GraphicsServices 0x31e4504b PurpleEventCallback + 666
26 CoreFoundation 0x3082cce3 __CFRUNLOOP_IS_CALLING_OUT_TO_A_SOURCE1_PERFORM_FUNCTION__ + 26
27 CoreFoundation 0x3082cca7 __CFRunLoopDoSource1 + 166
28 CoreFoundation 0x3081f56d __CFRunLoopRun + 520
29 CoreFoundation 0x3081f277 CFRunLoopRunSpecific + 230
30 CoreFoundation 0x3081f17f CFRunLoopRunInMode + 58
31 GraphicsServices 0x31e445f3 GSEventRunModal + 114
32 GraphicsServices 0x31e4469f GSEventRun + 62
33 UIKit 0x31e51123 -[UIApplication _run] + 402
34 UIKit 0x31e4f12f UIApplicationMain + 670
35 SPARKvue 0x000031ff main + 70
36 SPARKvue 0x000031b4 start + 40
)
terminate called after throwing an instance of 'NSException'
Program received signal: “SIGABRT”.
すべてのことが起こっているトリガーす:
[[self measurementPointer] setMeasurementDescriptor:descriptor];
このことから、
[[meterDisplay measurementPointer] addObserver:self
forKeyPath:@"measurementDescriptor"
options:NSKeyValueObservingOptionNew
context:nil];
基本的には、MeasurementPointerブポイントMeasurementDescriptor物とともにNSManagedObjectサブクラス.MeasurementDescriptor物を記述する特定の"計測"と"ユニットの組み合わせ(例えば、"温度(°C)"または"風速(mph)").MeasurementDescriptorsているようなものsingletonsる範囲がそれぞれにユニークな計測ユニットが含まれます。
MeasurementPointers参照している他の物両方のモデルオブジェクトとコントローラオブジェクト。A MeasurementPointer参照MeasurementDescriptor.たくさんの興味を知ることがMeasurementPointer開始を参照し、新たな異なるMeasurementDescriptor.ような変化を引き起こす可能性のあのグラフ表示の軸変わります。または、上記のコードが原因となるタスが表示された異なるサンプルから選択されたセットアップしました。
いと思う根本的な問題があることでCGImageを受けるメッセージを意図したものではないで...残念ながら、この間欠的になっていないのを見つけることができるようなパターンをトリガします。
解決
しているオブジェクトのたdealloc投稿内容の投稿者とだけにとどまりませんでした観察を別のオブジェクトです。歩いてすべての -addObserver...
呼びているか確認してください致し -removeObserver...
通話が少なくとも -dealloc
となる可能性があり、 -viewDidUnload
ご用途に応じて構造です。
他のヒント
またこのエラーが送信したの observeValueForKeyPath
方法 super
, 私として登録されていないオブザーバーに変化します。 Apple docs "ということを必ずお電話にスーパークラスの実施の observeValueForKeyPath
] を行った場合においてもこ."
私の固定変更:
- (void)observeValueForKeyPath:(NSString *)keyPath
ofObject:(id)object
change:(NSDictionary *)change
context:(void *)context {
if ([keyPath isEqualToString:kPropertyThatChanges]) {
...
}
[super observeValueForKeyPath:keyPath ofObject:object change:change context:context];
}
:
- (void)observeValueForKeyPath:(NSString *)keyPath
ofObject:(id)object
change:(NSDictionary *)change
context:(void *)context {
if ([keyPath isEqualToString:kPropertyThatChanges]) {
...
}
}
では迅速:
func observeValue(forKeyPath keyPath: String?, of object: Any?, change: [NSKeyValueChangeKey : Any]?, context: UnsafeMutableRawPointer?) {
if (keyPath == kPropertyThatChanges) {
}
super.observeValue(forKeyPath: keyPath, of: object, change: change, context: context)
}
To:
func observeValue(forKeyPath keyPath: String?, of object: Any?, change: [NSKeyValueChangeKey : Any]?, context: UnsafeMutableRawPointer?) {
if (keyPath == kPropertyThatChanges) {
}
}
またこの問題による偶発の目標としてのオブザーバーの代わりに、自己のような:
[self.someView addObserver:self.someView forKeyPath:@"key" options:0 context:nil];
エラーメッセージはすべてのものをこなった後の場合には誰もが同じことをしているのです。
また、同じ問題が、私の場合は観測と異なるコンテキスト.その後はすべて、同じコンテキストのクラッシュが無くなりました。ということで自信がない。
私にとって忘れてしまったので追加 override func observeValue(forKeyPath keyPath: String?, of object: Any?, change: [NSKeyValueChangeKey : Any]?, context: UnsafeMutableRawPointer?)
かのkeyPathった。
ここでは迅速なバージョン。追加のオブザーバー viewDidLoad
として再生することに deinit
:
lazy var firstNameTextField: UITextField = {
let textField = UITextField()
// configure your textField
return textField
}()
lazy var lastNameTextField: UITextField = {
let textField = UITextField()
// configure your textField
return textField
}()
override func viewDidLoad() {
super.viewDidLoad()
// 1. add your observers here
firstNameTextField.addObserver(self, forKeyPath: "text", options: [.old, .new], context: nil)
lastNameTextField.addObserver(self, forKeyPath: "text", options: [.old, .new], context: nil)
}
// 2. ***IMPORTANT you must add this function or it will crash***
override func observeValue(forKeyPath keyPath: String?, of object: Any?, change: [NSKeyValueChangeKey : Any]?, context: UnsafeMutableRawPointer?) {
if keyPath == "text" {
print("do something when the textField's text your observing changes")
}
}
// 3. remove them in deinit
deinit {
firstNameTextField.removeObserver(self, forKeyPath: "text", context: nil)
lastNameTextField.removeObserver(self, forKeyPath: "text", context: nil)
print("DEINIT")
}