Я создал простое приложение, которое получает отчеты из HockeyApp. Однако когда я запускаю приложение с инструментами утечки памяти, оно показало, что произошла утечка памяти, когда я выполняю действие getReport. Я не мог понять всю информацию, отображаемую в приборе.
Вот метод действия кнопки, который вызывает утечку памяти:
- (IBAction)getReports:(id)sender {
//initialize url that is going to be fetched.
NSURL *url = [NSURL URLWithString:@"https://rink.hockeyapp.net/api/2/apps/APP_ID/crash_reasons"];
//initialize a request from url
NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
[request addValue:tokenReceived forHTTPHeaderField:@"X-HockeyAppToken"];
[request setHTTPMethod:@"GET"];
[request setValue:@"application/x-www-form-urlencoded" forHTTPHeaderField:@"Content-Type"];
//initialize a connection from request
NSURLConnection *connection = [[NSURLConnection alloc] initWithRequest:request delegate:self];
self.getReportConnection = connection;
}
- (void)connection:(NSURLConnection *)connection didReceiveData:(NSData*)data{
if (connection==getReportConnection) {
[self.receivedData appendData:data];
NSLog(@"data is %@",data);
NSString *responseString = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
NSError *e = nil;
NSData *jsonData = [responseString dataUsingEncoding:NSUTF8StringEncoding];
NSDictionary *JSON = [NSJSONSerialization JSONObjectWithData:jsonData options: NSJSONReadingMutableContainers error: &e];
NSLog(@"login json is %@",JSON);
NSLog(@"reason json is %@",JSON[@"reason"]);
[JSON[@"crash_reasons"] enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) {
[reportArray addObject:obj[@"reason"]];
NSLog(@"index = %lu, Object For title Key = %@", (unsigned long)idx, obj[@"reason"]);
}];
NSError *error = nil;
NSArray *jsonArray = [NSJSONSerialization JSONObjectWithData:jsonData
options:kNilOptions error:&error];
if (error != nil) {
NSLog(@"Error parsing JSON.");
}
else {
NSLog(@"Array: %@,array count is %d", jsonArray,jsonArray.count);
}
// [reportArray addObject:[jsonArray objectAtIndex:0]];
if (JSON!=NULL) {
UIAlertView *alert=[[UIAlertView alloc]initWithTitle:@"Reports succesfully retrieved" message:@"" delegate:self cancelButtonTitle:@"Ok" otherButtonTitles: nil];
[alert show];
}
}
}
// This method receives the error report in case of connection is not made to server.
- (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error{
UIAlertView *errorAlert=[[UIAlertView alloc]initWithTitle:@"Wrong Login" message:nil delegate:self cancelButtonTitle:@"ok" otherButtonTitles: nil];
[errorAlert show];
NSLog(@"error is %@",error);
}
// This method is used to process the data after connection has made successfully.
- (void)connectionDidFinishLoading:(NSURLConnection *)connection{
}
Я вижу, что утечка памяти происходит незадолго до появления окна предупреждения в методе didRecieveData
.
Вот снимок экрана прибора утечки памяти, показывающий утечку памяти:
Я не мог понять, какая часть кода вызывает утечку памяти. Может ли кто-нибудь сказать мне, как определить часть кода, вызывающую утечку памяти, с помощью инструмента утечки?
Изменить: когда я запускаю приложение на симуляторе, прибор не обнаруживает утечки памяти:
Вот скриншот:
Снова, когда я запускаю приложение на устройстве, прибор показал мне утечку памяти:
Я заглянул в раздел утечек и обнаружил, что NSmutableArray
вызывает утечку:
Я использовал в своем коде только один NSMutableArray
. Я объявил это в .h
файле:
@property (nonatomic,strong) NSMutableArray *reportArray;
и разместил его в viewDidLoad
:
reportArray=[[NSMutableArray alloc]init];
и загрузил в didRecieveData
:
[reportArray addObject:obj[@"reason"]];
Снимки Stacktrace: