ios - UITableView和UINavigationController中的内存管理

标签 ios uitableview memory uinavigationcontroller

更新

我已经更新了代码,发现我可以使用nil iconDownLoader或downloadArray,但不能同时使用两者。

我认为卸载中的nil与dealloc中的发布不同吗?并且self.xxx = nil是必要的吗?我已经更新了代码。它工作正常(尽管它不会释放所有内存),但我不太了解。

你能解释为什么吗?谢谢

我已经搜索了很多,但仍然无法解决此内存管理问题。希望有人可以帮助我修复它并理解其原理。

我有一个UIViewController,其中有一个UITableView。 tableview加载自定义单元格和图像。每次它消耗大量内存,并且在我popViewControllerAnimated之后不会释放内存。多次按下并弹出viewController后,内存将增加到200m。

这是一些代码,任何帮助将不胜感激。

- (void)didReceiveMemoryWarning
{
    [super didReceiveMemoryWarning];
}

#pragma mark - View lifecycle

- (void)viewDidUnload
{
    self.newsArray = nil;
    self.newsTableView = nil;
    self.indicatorView = nil;
//  self.iconDownLoader = nil;
    self.downloadArray = nil;

    [super viewDidUnload];
}

... 

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
    static NSString *CellIdentifier = @"NewsCellIdentifier";

    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier];
    if (cell == nil) {
        [[NSBundle mainBundle] loadNibNamed:@"NewsTableViewCell" owner:self options:nil];
        cell = self.newsTableViewCell;
        self.newsTableViewCell = nil;
    }

    // read from newsModel
    NewsModel *news = [newsArray objectAtIndex:indexPath.row];

    UILabel *label;
    label = (UILabel *)[cell viewWithTag:10];
    label.text = [NSString stringWithString:news.title];
    label = nil;
    label = (UILabel *)[cell viewWithTag:11];
    label.text = [NSString stringWithString:news.description];
    UIImageView *imageView = (UIImageView *)[cell viewWithTag:12];
    imageView.image = news.image;

    if (news.image == nil)
    {
        imageView.image = [UIImage imageNamed:IconPlaceHolder];

        IconDownLoader *iconDownLoader = [[IconDownLoader alloc] init];
        iconDownLoader.url = news.imageUrl;
        iconDownLoader.delegate = self;
        iconDownLoader.indexPath = indexPath;
        if (appDelegate.ip4 == YES)
        {
            iconDownLoader.width = 300;
            iconDownLoader.height = 150;
        }
        else
        {
            iconDownLoader.width = 150;
            iconDownLoader.height = 75;
        }
        [downloadArray addObject:iconDownLoader];
        [iconDownLoader start];
    }
    return cell;
}

#pragma mark - IconDownLoaderDelegate

- (void)iconDownLoadFinsh:(NSData *)imageData row:(NSIndexPath *)indexPath {

    UITableViewCell *cell = [newsTableView cellForRowAtIndexPath:indexPath];
    UIImageView *imageView = (UIImageView *)[cell viewWithTag:12];
    if (imageData != 0)
    {
        imageView.image = [UIImage imageWithData:imageData];
    }
    else
    {
        imageView.image = [UIImage imageNamed:@"icon57"];
    }
    NewsModel *newsModel = [newsArray objectAtIndex:indexPath.row];
    newsModel.image = [UIImage imageWithData:imageData];
}

这是IconDownloader
#import "IconDownLoader.h"
#import "ASIHTTPRequest.h"

@implementation IconDownLoader

@synthesize delegate = _delegate;
@synthesize url = _url;
@synthesize indexPath = _indexPath;
@synthesize width = _width;
@synthesize height = _height;
@synthesize request = _request;

- (void)start {

    NSString *originalString = @"width=%s&height=%s";
    NSString *newString = [NSString stringWithFormat:@"width=%d&height=%d&type=jpg", self.width, self.height];

    NSString *resizedURL = [self.url stringByReplacingOccurrencesOfString:originalString withString:newString];

    NSURL *url = [NSURL URLWithString:[resizedURL stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding]];
    _request = [ASIHTTPRequest requestWithURL:url];
    if (_indexPath) {
        _request.userInfo = [NSDictionary dictionaryWithObject:_indexPath forKey:@"indexPath"];
    }
    [_request setDelegate:self];
    [_request startAsynchronous];
}

- (void)requestFinished:(ASIHTTPRequest *)request {

    NSInteger statusCode = request.responseStatusCode;
    switch (statusCode) {
        case 401: // Not Authorized: either you need to provide authentication credentials, or the credentials provided aren't valid.
            break;

        case 200: {
            NSData *responseData = [request responseData];
            if (!responseData) {
                UIAlertView *alertView;
                alertView = [[UIAlertView alloc] initWithTitle:@"Oops" message:[NSString stringWithFormat:@"Download failed in row %d", _indexPath.row] delegate:nil cancelButtonTitle:@"OK" otherButtonTitles:nil];
                return;
            }
            [_delegate iconDownLoadFinsh:responseData row:[request.userInfo objectForKey:@"indexPath"]];
        }
            break;

        case 304: // Not Modified: there was no new data to return.
            break;
        case 400: // Bad Request: your request is invalid, and we'll return an error message that tells you why. This is the status code returned if you've exceeded the rate limit
            break;
        case 403: // Forbidden: we understand your request, but are refusing to fulfill it.  An accompanying error message should explain why.
            break;

        case 404: // Not Found: either you're requesting an invalid URI or the resource in question doesn't exist (ex: no such user). 
        {
            UIAlertView *alertView;
            alertView = [[UIAlertView alloc] initWithTitle:@"Oops" message:@"Classical 404, please contact with your administer" delegate:nil cancelButtonTitle:@"OK" otherButtonTitles:nil];
        }
            break;
        case 500: // Internal Server Error: we did something wrong.  Please post to the group about it and the Weibo team will investigate.
        case 502: // Bad Gateway: returned if Weibo is down or being upgraded.
        case 503: // Service Unavailable: the Weibo servers are up, but are overloaded with requests.  Try again later.
        {
            UIAlertView *alertView;
            alertView = [[UIAlertView alloc] initWithTitle:@"Oops" message:@"503 error, service unavailable" delegate:nil cancelButtonTitle:@"OK" otherButtonTitles:nil];
        }
        default:{

        }
    }
}

- (void)dealloc {
    if (_request != nil) {
        [_request clearDelegatesAndCancel];
    }
}

@end

最佳答案

您没有发布IconDownLoader

关于ios - UITableView和UINavigationController中的内存管理,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9155611/

相关文章:

ios - 在 UIView 上上下滑动工作但 ScrollView 阻碍它快速?

ios - 检测单元格编辑中的单元格点击,显示重新排序控制 + 披露控制作为 Safari 书签编辑

ios - 在 Swift 中加速 TableView 加载 JSON

c - C中的内存消耗

c++ - C/C++ 字符串内存泄漏?

ios - 如何将文本字段数据保存在字典中然后保存在数组中?

objective-c - RestKit 多个 RKObjectManagers 对象映射

ios - 更新应用程序时如何强制删除 NSUserDefaults?

ios - 如何列出本地 WiFi/LAN 网络上的所有共享设备?

memory - 您可以获得多接近 GPU 理论内存带宽?