0

私のアプリでは、巨大な(200MB以上)ファイルをディスクにダウンロードしています。
ダウンロードが完了した後、私はこのエラーが発生しています:

[NSURLConnectionInternalConnection _withConnectionDisconnectFromConnection]:  
message sent to deallocated instance 0x8cb0b40  

info malloc-history 0x8cb0b40に電話をかけようとしましたが、取得しています

Undefined info command: "malloc-history 0x8cb0b40".  Try "help info".  

私は(gdb)を使用していますが、それでもこの情報を取得しています。ここでエラーを見つけることができません。


この問題を解決するにはどうすればよいですか?

これが私のコード(NSURLメソッド)です:

#pragma mark - NSURL methods


- (void)loadingProgress:(NSNumber *)nProgress {
[self.progressView setProgress:[nProgress floatValue]];
}


- (void) connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response {
XLog(@"");

[self.receivedData setLength:0];
XLog(@"response: %@", [response description]);


self.path = [[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) objectAtIndex:0] stringByAppendingPathComponent:@"vox.zip"];
[[NSFileManager defaultManager] createFileAtPath:self.path contents:nil attributes:nil];
self.file = [[NSFileHandle fileHandleForUpdatingAtPath:self.path] retain];// Here file is object of NSFileHandle and its declare in .h File

[self.file seekToEndOfFile];


if ([response isKindOfClass:[NSHTTPURLResponse self]]) {
    NSDictionary *headers = [(NSHTTPURLResponse *)response allHeaderFields];
    lengthOfXMLContent = [[headers objectForKey:@"Content-Length"] intValue];
    XLog(@"session: %@", [headers objectForKey:@"Set-Cookie"]);
    
    [self.prefs setValue:[headers objectForKey:@"Set-Cookie"] forKey:@"keySession"];
    
    XLog(@"headers: %@", headers);
    XLog(@"Incoming length: %i", lengthOfXMLContent);
    
}
}


- (void) connection:(NSURLConnection *)connection didReceiveData:(NSData *)data {

//XLog(@"didReceiveData entered");

self.dataCount = self.dataCount + [data length];

if (self.receivedData) {
    
    self.paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
    self.documentsDirectory = [self.paths objectAtIndex:0];
    
            
    self.responseString = [[[NSString alloc] initWithData:self.receivedData encoding:NSUTF8StringEncoding] autorelease];

}

XLog(@"------------------------------------------: %d", lengthOfXMLContent);


double progress = (double)self.dataCount / (double)lengthOfXMLContent;


XLog(@"--------------: %i", self.dataCount);
XLog(@"--------------: %d", lengthOfXMLContent);
XLog(@"--------------: ");
XLog(@"--------------: %f \n", progress);


[self.file seekToEndOfFile];
[self.file writeData:data];
//XLog(@"data--------------: %i", [data length]);

[self performSelectorInBackground:@selector(loadingProgress:) withObject:[NSNumber numberWithFloat:progress]];
}


- (void) connection:(NSURLConnection *)connection didFailWithError:(NSError *)error {
XLog(@"");
XLog(@"error: %@", error);


self.receivedData = nil;
self.progressView.progress = 0.00;
lengthOfXMLContent = 0;
//recievedData = NO;

UIAlertView *alertCanceled = [[UIAlertView alloc] initWithTitle:@"Verbindungsfehler" message:@"Verbindung wurde unterbrochen. \nPruefen Sie Ihre Verbindung und versuchen Sie es noch einmal." delegate:self cancelButtonTitle:@"OK" otherButtonTitles:nil];
[alertCanceled show];
[alertCanceled release];

viewDownload.hidden = YES;

//[self removeActivityViewer];
[connection release];
[self.file closeFile];
}


- (void) connectionDidFinishLoading:(NSURLConnection *)connection {
XLog(@"");


[self.file closeFile];
//NSError *error = nil;
XLog(@"Succeeded! Received %d bytes of data", [self.receivedData length]);


[connection release];

//self.lblInfo.text = @"Datei wurde heruntergeladen.\nDatei wird entpackt, bitte Warten..";
self.progressView.hidden = YES;
self.btnDownloadVOX_DE.hidden = YES;
self.btnDownloadVOX_EN.hidden = YES;


[UIApplication sharedApplication].networkActivityIndicatorVisible = NO;




ZipArchive *za = [[[ZipArchive alloc] init] autorelease];
[za UnzipOpenFile:[NSString stringWithFormat:@"%@/vox.zip", self.documentsDirectory]];
[za UnzipFileTo:self.documentsDirectory overWrite:YES];


viewDownload.hidden = YES;

}  
4

2 に答える 2

4

connectionDidFinishLoadingで、デリゲートにサービスを提供している間に接続を解放します。

クラスにそれを保持するプロパティまたはivarがあると仮定します(これを「myConnection」と呼びます)。あなたがすべきことは、そのdidFinishLoadingデリゲートでこれです:

connection.delegate = nil; // insure no more delegate calls

に続く:

// either this
[self performSelectorOnMainThread:@selectior(setMyConnection:) withObject:nil waitUntilDone:NO];

// or using GCD (my preference)
dispatch_asynch(dispatch_get_main_queue(), ^ { myConnection = nil; } );

これでクラッシュが止まるに違いない。[位置情報サービスでも同様の問題がありました!]

于 2012-07-14T15:07:58.180 に答える
2

AFNetworkingFrameworkを使用することをお勧めします。このフレームワークは処理がはるかに簡単で、接続を手動で処理する多くの利点があります。たとえば、再開可能なダウンロード、詳細なアップ/dowlnoadの進行状況など。

GithubのAFNetworking

于 2012-07-13T09:40:16.390 に答える