Създадох просто приложение, което получава отчетите от 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: