2012-10-12 7 views
38

Jestem na Xcode 4.5.1. Myślę, że jest to stosunkowo nowe zachowanie, które widzę.Skąd wiadomo, kiedy można zignorować __cxa_throw w punkcie przerwania wszystkich wyjątków?

Wolę tworzyć i testować moje projekty z włączonym punktem przerwania "Wszystkie wyjątki".

Uruchomiłem scenariusz, w którym ładuję obrazy miniatur do komórek w widoku tabeli, w którym otrzymuję wyjątek __cxa_throw. Kiedy klikam przycisk "Kontynuuj wykonywanie programu", Xcode kontynuuje swoją wesołą drogę. Dostaję miniatury. Aplikacja wydaje się działać dobrze. Szukam wskazówek, w jaki sposób mogę określić, czy jest to coś bezpiecznego do zignorowania. Może jakieś wskazówki dotyczące zrozumienia śladu stosu? Albo coś innego?

Oto fragment kodu:

NSString *imageStr = item.thumbURL; 
    imageStr = [imageStr stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding]; 
    NSURL *imageURL; 

    if (![imageStr isEqualToString:@""]) { 
     imageURL = [NSURL URLWithString:imageStr]; 
     NSLog(@"imageURL: %@",imageURL); 
     if (imageURL == nil) { 
      NSLog(@"imageUrl was nil for string: %@",imageStr); 
     } else { 
      UIActivityIndicatorView *spinner = [[UIActivityIndicatorView alloc] initWithActivityIndicatorStyle:UIActivityIndicatorViewStyleWhiteLarge]; 
      //spinner.frame = CGRectMake(cell.imageView.frame.size.width/2,cell.imageView.frame.origin.y + cell.imageView.frame.size.height/2,cell.imageView.frame.size.width,cell.imageView.frame.size.height); 
      spinner.frame = CGRectMake(10,20,40,40); 
      [spinner startAnimating]; 
      [cell addSubview:spinner]; 

      dispatch_queue_t downloadQueue = dispatch_queue_create("thumbnail downloader", NULL); 
      dispatch_async(downloadQueue, ^{ 
       NSLog(@"Network request for tour_thumb image: %@",imageStr); 
       UIImage *img = [[UIImage alloc] initWithData:[NSData dataWithContentsOfURL:imageURL]]; 
       dispatch_async(dispatch_get_main_queue(), ^{ 
        [spinner removeFromSuperview]; 
        UIImageView *imgView=[[UIImageView alloc] initWithFrame:CGRectMake(8, 8, cell.contentView.frame.size.width/2 - 16, cell.contentView.frame.size.height - 16)]; 
        imgView.image = img; 
        [cell.contentView addSubview:imgView]; 
       }); 
      }); 
      dispatch_release(downloadQueue); 
     } 
    } 

Oto co widzę na ślad stosu:

#0 0x34a9c524 in __cxa_throw() 
#1 0x387014ce in AddChildNode(XMP_Node*, XML_Node const&, char const*, bool)() 
#2 0x38700d22 in RDF_LiteralPropertyElement(XMP_Node*, XML_Node const&, bool)() 
#3 0x3870094e in RDF_PropertyElementList(XMP_Node*, XML_Node const&, bool)() 
#4 0x38700608 in ProcessRDF(XMP_Node*, XML_Node const&, unsigned int)() 
#5 0x3871480a in XMPMeta::ParseFromBuffer(char const*, unsigned int, unsigned int)() 
#6 0x387095c0 in WXMPMeta_ParseFromBuffer_1() 
#7 0x38725578 in TXMPMeta<std::string>::ParseFromBuffer(char const*, unsigned int, unsigned int)() 
#8 0x387254de in TXMPMeta<std::string>::TXMPMeta(char const*, unsigned int)() 
#9 0x38722b70 in CreateMetadataFromXMPDataInternal(char const*, unsigned long, unsigned int)() 
#10 0x38739a50 in readXMPProperties() 
#11 0x386a01fc in readXMPData() 
#12 0x3868cec8 in initImageJPEG() 
#13 0x3868c2ee in _CGImagePluginInitJPEG() 
#14 0x3867e274 in makeImagePlus() 
#15 0x3867ddde in CGImageSourceCreateImageAtIndex() 
#16 0x38e117b6 in _UIImageRefFromData() 
#17 0x38e116c6 in -[UIImage initWithData:]() 
#18 0x0004cb0a in __57-[ViewController tableView:cellForRowAtIndexPath:]_block_invoke_0 at ViewController.m:335 
#19 0x313fc792 in _dispatch_call_block_and_release() 
#20 0x313ffb3a in _dispatch_queue_drain() 
#21 0x313fd67c in _dispatch_queue_invoke() 
#22 0x31400612 in _dispatch_root_queue_drain() 
#23 0x314007d8 in _dispatch_worker_thread2() 
#24 0x394767f0 in _pthread_wqthread() 
#25 0x39476684 in start_wqthread() 
+1

To pytanie i odpowiedź poniżej (to znaczy tylko wyjątki od standardowych wyjątków od Celu C) dotyczą również wersji Xcode 7.2.1 i prawdopodobnie także wersji późniejszych. – bluebinary

Odpowiedz

57

To jest oczywiście głęboko w realizacji i wyjątków C++ może być częścią normalny przepływ, jeśli zdecydowali się użyć wyjątków do wewnętrznego sygnalizowania błędów.

Jeśli nie piszesz C++ samemu, to jest bezpieczne do zignorowania.

Wystarczy pułapka średnia Objective-C wyjątek:

objc_exception_throw 
+20

Dzięki, trojanfoe. Edytowałem punkt przerwania, aby zmienić "Wyjątek: wszystko" na "Wyjątek: Cel-C". To ma sens. –

+5

Na wszelki wypadek, gdy ktoś to osiągnie i nie wiesz, jak to zrobić :) http://stackoverflow.com/a/14767076/2298217 – iamdavidlam

29

Go tutaj:

enter image description here

I to zrobić:

enter image description here

Aby to włączyć:

enter image description here

w tym:

enter image description here

Będziesz nadal uzyskać wiele korzyści dodanie punktu przerwania, ale nie będzie miał swoją aplikację katastrofę dla rzeczy, które prawdopodobnie nie są w każdym razie odpowiedzialny. Teraz, jeśli pracujesz z C++, lepiej się tym przejmuj.

+2

Pomocne, dziękuję. –

Powiązane problemy