2012-01-07 13 views
17

तो मैं आईपैड 5 और आईपैड 2 पर क्रैशलिटिक्स में क्रैशलिटिक्स में अक्सर इस क्रैश को देख रहा हूं, ऐसा लगता है कि यह आईओएस 5 चला रहा है, लेकिन ऐसा लगता है कि यह स्मृति चेतावनी के कारण होता है, लेकिन स्टैक ट्रेस ' टी संदर्भ अपने आवेदन कोड के किसी भी, बस आईओएस चौखटे:UIViewController purgeMemoryForReason: आईओएस पर क्रैशिंग 5

0 libobjc.A.dylib objc_msgSend + 15 
1 UIKit   -[UIViewController purgeMemoryForReason:] + 64 
2 Foundation  __57-[NSNotificationCenter addObserver: selector: name: object:]_block_invoke_0 + 18 
3 CoreFoundation  ___CFXNotificationPost_block_invoke_0 + 70 
4 CoreFoundation  _CFXNotificationPost + 1406 
5 Foundation  -[NSNotificationCenter postNotificationName: object: userInfo:] + 66 
6 Foundation  -[NSNotificationCenter postNotificationName: object:] + 30 
7 UIKit   -[UIApplication _performMemoryWarning] + 80 
8 UIKit   -[UIApplication _receivedMemoryNotification] + 174 
9 libdispatch.dylib _dispatch_source_invoke + 516 
10 libdispatch.dylib _dispatch_queue_invoke + 50 
11 libdispatch.dylib _dispatch_main_queue_callback_4CF + 156 
12 CoreFoundation  __CFRunLoopRun + 1268 
13 CoreFoundation  CFRunLoopRunSpecific + 300 
14 CoreFoundation  CFRunLoopRunInMode + 104 
15 GraphicsServices GSEventRunModal + 156 
16 UIKit   UIApplicationMain + 1090 
17 500px iOS  main.m line 12 

मैं उच्च और निम्न googled है, लेकिन इस के लिए किसी भी समाधान नहीं मिल रहा। ऐसा लगता है कि यह UIViewController इंस्टेंस को ओवर-रिलीज़ करने के कारण होता है, लेकिन मैं एआरसी का उपयोग कर रहा हूं, इसलिए मुझे नहीं लगता कि यह मामला कैसा हो सकता है।

मुझे इस बात से भी नुकसान उठाना है कि इस तक कैसे पहुंचे। मैं यह भी नहीं बता सकता कि कौन सा UIViewController उपclass समस्या उत्पन्न कर रहा है। मैंने सिम्युलेटर और डिवाइस पर समस्या को पुन: उत्पन्न करने का प्रयास किया है, लेकिन मुझे यह नहीं मिल रहा है कि इसका कारण क्या है। क्या किसी ने इस तरह कुछ देखा है या इस मुद्दे को पुन: उत्पन्न करने के तरीके पर सुझाव हैं?

+2

दिलचस्प। आम तौर पर, स्टैक पर अगला चरण उस दृश्य नियंत्रक पर एक unloadViewIfReloadable कॉल होगा। जैसा कि हम अब एक दुर्घटना देखते हैं, इसका मतलब यह है कि इस विधि को भी नहीं पहुंचाया गया है या हम पहले से ही उस चरण के पीछे आ गए हैं। बाद के लिए, अपने दृश्य DidUnload विधि कार्यान्वयन की जांच करें। यह अगला कदम होगा जिसे मैं मानता हूं। एक मानक सिफारिश के रूप में, लाश सक्षम करें और सिम्युलेटर पर स्मृति चेतावनी ट्रिगर करें। – Till

+0

मैं इसे एक शॉट दूंगा और आपको वापस ले जाऊंगा। धन्यवाद! –

उत्तर

17

मुझे लगता है कि मैंने इस मुद्दे को हल किया है। मैं इसके बारे में सोच रहा था, और समस्या UIViewController दृश्य को उतारना नहीं है, यह वास्तविक कम स्मृति चेतावनी अधिसूचना की पोस्टिंग है। मेरे कोड में कई उदाहरण हैं जहां मैं [[NSNotificationCenter defaultCenter] removeObserver:self] पर कॉल करता हूं। यह dealloc विधि में ठीक है, लेकिन viewDidUnload विधियों में इसके दो उदाहरण थे।

मैंने यह देखा जब UIViewController में से किसी एक के didReceiveMemory में मेरा ब्रेकपॉइंट हिट नहीं हो रहा था। viewDidUnload में कोड self को अन्य सिस्टम सिस्टम अधिसूचनाओं के साथ-साथ here विस्तृत रूप से अनधिकृत कर रहा था।

मैं इसे स्वीकार किए गए उत्तर के रूप में चिह्नित नहीं कर रहा हूं जब तक कि मैं सत्यापित नहीं करता कि क्रैश नए अपडेट के साथ रुक गया है।

अपडेट: मैंने क्रैशलिटिक्स के साथ सत्यापित किया है कि समस्या ठीक हो गई है!

+2

हाँ, अधिसूचना का सुनहरा नियम देख रहा है: केवल उन अधिसूचनाओं के लिए पर्यवेक्षकों को हटाएं जिनके लिए आपने पंजीकरण किया है। मैंने कुछ समय पहले इसके बारे में पढ़ा है, लेकिन परिणामों को कभी भी स्पष्ट नहीं किया है। खुशी है कि आपने अपनी समस्या तय की है। – iHunter

+0

@AshFurrow दृश्य पर उन अधिसूचनाओं से स्वयं को हटा देंDidUnload _only_ अगर आप उन लोगों के लिए पंजीकरण कर रहे हैं जो DidLoad में हैं। – Till

+0

@ समस्या होने तक, UIViewController प्रारंभिकरण पर अधिसूचनाओं के लिए पंजीकरण कर रहा प्रतीत होता है कि मैं देखने के लिए पंजीकरण रद्द कर रहा हूं DidUnload। मैंने अपना निकालना ऑब्सर्वर बदल दिया है: निकालने के लिए ऑब्सर्वर: नाम: ऑब्जेक्ट: ताकि मैं विशिष्ट सूचनाओं के लिए * केवल * अनधिकृत हूं। –

8

मैं उपकरणों के लिए HockeyApp द्वारा रिपोर्ट दुर्घटनाओं में ठीक उसी स्टैक ट्रेस देखा

मैं never called[[NSNotificationCenter defaultCenter] removeObserver:self] अंदर dealloc छोड़कर आईओएस 5 पर चल रहा है, इसलिए इस दुर्घटना के कारण नहीं हो सकता।

यहां बताया गया है कि मैं क्रैश को पुन: उत्पन्न करने में सक्षम था: MasterViewController से मैं DetailViewController दबाता हूं, फिर बैक बटन टैप करके इसे पॉप करता हूं। अंत में, मैं एक स्मृति चेतावनी ट्रिगर करता हूं और दुर्घटना होती है (केवल आईओएस 5 पर)।

ऐसा लगता है कि DetailViewController उदाहरण एक चक्र को बनाए रखने की वजह से पॉप जा रहा है जब SVPullToRefresh उपयोग करने के बाद जारी नहीं किया जाता है:

@implementation DetailViewController 

- (void) viewDidLoad 
{ 
    [super viewDidLoad]; 

    [self.scrollView addPullToRefreshWithActionHandler:^{ 
     [self refresh]; 
    }]; 
} 

@end 

DetailViewController के बाद से जारी नहीं किया गया है यह अभी भी स्मृति चेतावनी सूचनाओं के लिए पंजीकृत है और यह क्या है होता है:

frame #0: 0x0004d61b MyApp`-[DetailViewController dealloc](self=0x089a5150, _cmd=0x024d2738) + 27 at DetailViewController.m:103 
frame #1: 0x0227ae3d libobjc.A.dylib`_objc_rootRelease + 47 
frame #2: 0x0227ae00 libobjc.A.dylib`objc_release + 48 
frame #3: 0x0227c047 libobjc.A.dylib`objc_storeStrong + 39 
frame #4: 0x0004e44c MyApp`__destroy_helper_block_ + 44 at DetailViewController.m:157 
frame #5: 0x029b555d libsystem_sim_blocks.dylib`_Block_release + 166 
frame #6: 0x0227ae00 libobjc.A.dylib`objc_release + 48 
frame #7: 0x0227c047 libobjc.A.dylib`objc_storeStrong + 39 
frame #8: 0x00084c8d MyApp`-[SVPullToRefreshView .cxx_destruct](self=0x08bf3af0, _cmd=0x00000001) + 525 at UIScrollView+SVPullToRefresh.m:121 
frame #9: 0x0226630d libobjc.A.dylib`object_cxxDestructFromClass + 104 
frame #10: 0x02270fde libobjc.A.dylib`objc_destructInstance + 38 
frame #11: 0x02271015 libobjc.A.dylib`object_dispose + 20 
frame #12: 0x0247a9a1 CoreFoundation`-[NSObject dealloc] + 97 
frame #13: 0x00a8cdc7 UIKit`-[UIView dealloc] + 748 
frame #14: 0x0227ae3d libobjc.A.dylib`_objc_rootRelease + 47 
frame #15: 0x00a90b73 UIKit`-[UIView(Hierarchy) removeFromSuperview] + 194 
frame #16: 0x00a8cc10 UIKit`-[UIView dealloc] + 309 
frame #17: 0x00a9d6ff UIKit`-[UIScrollView dealloc] + 405 
frame #18: 0x013ab36c Foundation`NSKVODeallocate + 105 
frame #19: 0x0227ae3d libobjc.A.dylib`_objc_rootRelease + 47 
frame #20: 0x00b21c12 UIKit`-[UIViewController setView:] + 447 
frame #21: 0x00b21885 UIKit`-[UIViewController unloadViewForced:] + 117 
frame #22: 0x00b2180b UIKit`-[UIViewController unloadViewIfReloadable] + 41 
frame #23: 0x00b256ff UIKit`-[UIViewController purgeMemoryForReason:] + 75 
frame #24: 0x00b2563b UIKit`-[UIViewController didReceiveMemoryWarning] + 41 
frame #25: 0x00b2560d UIKit`-[UIViewController _didReceiveMemoryWarning:] + 33 
frame #26: 0x0141ca29 Foundation`__57-[NSNotificationCenter addObserver:selector:name:object:]_block_invoke_0 + 40 
frame #27: 0x02443855 CoreFoundation`___CFXNotificationPost_block_invoke_0 + 85 
frame #28: 0x02443778 CoreFoundation`_CFXNotificationPost + 1976 
frame #29: 0x0136119a Foundation`-[NSNotificationCenter postNotificationName:object:userInfo:] + 98 
frame #30: 0x0136db03 Foundation`-[NSNotificationCenter postNotificationName:object:] + 55 
frame #31: 0x00a64cf4 UIKit`-[UIApplication _performMemoryWarning] + 91 
frame #32: 0x00a64e00 UIKit`-[UIApplication _receivedMemoryNotification] + 180 
frame #33: 0x00a64f98 UIKit`__block_global_0 + 36 
frame #34: 0x029f1450 libdispatch.dylib`_dispatch_source_invoke + 719 
frame #35: 0x029edcc4 libdispatch.dylib`_dispatch_queue_invoke + 66 
frame #36: 0x029ee4cf libdispatch.dylib`_dispatch_main_queue_callback_4CF + 295 
frame #37: 0x023af803 CoreFoundation`__CFRunLoopRun + 2003 
frame #38: 0x023aed84 CoreFoundation`CFRunLoopRunSpecific + 212 
frame #39: 0x023aec9b CoreFoundation`CFRunLoopRunInMode + 123 
frame #40: 0x038d07d8 GraphicsServices`GSEventRunModal + 190 
frame #41: 0x038d088a GraphicsServices`GSEventRun + 103 
frame #42: 0x00a5a626 UIKit`UIApplicationMain + 1163 
frame #43: 0x00002b82 MyApp`main(argc=1, argv=0xbffff318) + 178 at main.m:15 

या अंग्रेजी: SVPullToRefreshView उदाहरण देखने का एक परिणाम अनलोड के रूप में जारी किया गया है। चूंकि SVPullToRefreshView उदाहरण DetailViewController का संदर्भ रखने के लिए अंतिम वस्तु है, इसे जारी किया जाता है, फिर इसे हटा दिया जाता है। लेकिन purgeMemoryForReason: अभी भी चीजों को (यानी इंस्टेंस वेरिएबल्स एक्सेस करना) बस डिलीओटेड व्यू कंट्रोलर के साथ कर रहा था, इसलिए क्रैश।

एक बार निदान का निदान बहुत आसान था: बस पहले स्थान पर बनाए रखने के चक्र से बचें।

@implementation DetailViewController 

- (void) viewDidLoad 
{ 
    [super viewDidLoad]; 

    __typeof__(self) __weak weakSelf = self; 
    [self.scrollView addPullToRefreshWithActionHandler:^{ 
     [weakSelf refresh]; 
    }]; 
} 

@end 
+0

यहां एक बग purgeMemoryForReason अपने काम को करने से पहले स्वयं को बनाए रखता है और फिर इसे समाप्त होने के बाद स्वयं को छोड़ देता है। एक और बात यह है कि दृश्य पदानुक्रम के सफल टियरडाउन के लिए अनुमानित आदेश पर भरोसा करते हुए वीसी के विचारों के साथ कोई मजबूत संबंध नहीं था। –

संबंधित मुद्दे