亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 系統 > iOS > 正文

舉例講解iOS中延遲加載和上拉刷新/下拉加載的實現

2020-07-26 03:34:17
字體:
來源:轉載
供稿:網友

lazy懶加載(延遲加載)UITableView
舉個例子,當我們在用網易新聞App時,看著那么多的新聞,并不是所有的都是我們感興趣的,有的時候我們只是很快的滑過,想要快速的略過不喜歡的內容,但是只要滑動經過了,圖片就開始加載了,這樣用戶體驗就不太好,而且浪費內存.
             這個時候,我們就可以利用lazy加載技術,當界面滑動或者滑動減速的時候,都不進行圖片加載,只有當用戶不再滑動并且減速效果停止的時候,才進行加載.
              剛開始我異步加載圖片利用SDWebImage來做,最后試驗的時候出現了重用bug,因為雖然SDWebImage實現了異步加載緩存,當加載完圖片后再請求會直接加載緩存中的圖片,注意注意注意,關鍵的來了,如果是lazy加載,滑動過程中是不進行網絡請求的,cell上的圖片就會發生重用,當你停下來能進行網絡請求的時候,才會變回到當前Cell應有的圖片,大概1-2秒的延遲吧(不算延遲,就是沒有進行請求,也不是沒有緩存的問題).怎么解決呢?這個時候我們就要在Model對象中定義個一個UIImage的屬性,異步下載圖片后,用已經緩存在沙盒中的圖片路徑給它賦值,這樣,才cellForRowAtIndexPath方法中,判斷這個UIImage對象是否為空,若為空,就進行網絡請求,不為空,就直接將它賦值給cell的imageView對象,這樣就能很好的解決圖片短暫重用問題.
              @下面我的代碼用的是自己寫的異步加載緩存類,SDWebImage的加載圖片的懶加載,會在后面的章節給出.(為什么不同呢,因為SDWebImage我以前使用重來不關心它將圖片存儲在沙盒中的名字和路徑,但是要實現懶加載的話,一定要得到圖片路徑,所以在找SDWebImage如何存儲圖片路徑上花了點時間)

復制代碼 代碼如下:

@model類 
#import <Foundation/Foundation.h> 
 
@interface NewsItem : NSObject 
 
@property (nonatomic,copy) NSString * newsTitle; 
@property (nonatomic,copy) NSString * newsPicUrl; 
@property (nonatomic,retain) UIImage * newsPic; //  存儲每個新聞自己的image對象 
 
- (id)initWithDictionary:(NSDictionary *)dic; 
 
//  處理解析 
+ (NSMutableArray *)handleData:(NSData *)data; 
@end 
 
 
#import "NewsItem.h" 
#import "ImageDownloader.h" 
 
@implementation NewsItem 
 
- (void)dealloc 

    self.newsTitle = nil; 
    self.newsPicUrl = nil; 
    self.newsPic = nil; 
    [super dealloc]; 

 
- (id)initWithDictionary:(NSDictionary *)dic 

    self = [super init]; 
    if (self) { 
 
 
        self.newsTitle = [dic objectForKey:@"title"]; 
        self.newsPicUrl = [dic objectForKey:@"picUrl"]; 
         
        //從本地沙盒加載圖像 
        ImageDownloader * downloader = [[[ImageDownloader alloc] init] autorelease]; 
        self.newsPic = [downloader loadLocalImage:_newsPicUrl]; 
 
    } 
 
    return self; 

 
+ (NSMutableArray *)handleData:(NSData *)data; 

 
        //解析數據 
        NSError * error = nil; 
        NSDictionary * dic = [NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingMutableContainers error:&error]; 
        NSMutableArray * originalArray = [dic objectForKey:@"news"]; 
 
        //封裝數據對象 
        NSMutableArray * resultArray = [NSMutableArray array]; 
     
        for (int i=0 ;i<[originalArray count]; i++) { 
            NSDictionary * newsDic = [originalArray objectAtIndex:i]; 
            NewsItem * item = [[NewsItem alloc] initWithDictionary:newsDic]; 
            [resultArray addObject:item]; 
            [item release]; 
        } 
 
        return resultArray; 
 

 
@end 

復制代碼 代碼如下:

@圖片下載類 
#import <Foundation/Foundation.h> 
 
 
@class NewsItem; 
 
 
@interface ImageDownloader : NSObject 
 
 
@property (nonatomic,copy) NSString * imageUrl; 
@property (nonatomic,retain) NewsItem * newsItem; //下載圖像所屬的新聞 
 
 
//圖像下載完成后,使用block實現回調 
@property (nonatomic,copy) void (^completionHandler)(void); 
 
 
//開始下載圖像 
- (void)startDownloadImage:(NSString *)imageUrl; 
 
 
//從本地加載圖像 
- (UIImage *)loadLocalImage:(NSString *)imageUrl; 
 
 
@end 
 
 
 
 
#import "ImageDownloader.h" 
#import "NewsItem.h" 
 
 
@implementation ImageDownloader 
 
 
- (void)dealloc 

    self.imageUrl = nil; 
    Block_release(_completionHandler); 
    [super dealloc]; 

 
 
 
 
#pragma mark - 異步加載 
- (void)startDownloadImage:(NSString *)imageUrl 

 
 
    self.imageUrl = imageUrl; 
 
 
    // 先判斷本地沙盒是否已經存在圖像,存在直接獲取,不存在再下載,下載后保存 
    // 存在沙盒的Caches的子文件夾DownloadImages中 
    UIImage * image = [self loadLocalImage:imageUrl]; 
 
 
    if (image == nil) { 
 
 
        // 沙盒中沒有,下載 
        // 異步下載,分配在程序進程缺省產生的并發隊列 
        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{ 
 
 
            // 多線程中下載圖像 
            NSData * imageData = [NSData dataWithContentsOfURL:[NSURL URLWithString:imageUrl]]; 
 
 
            // 緩存圖片 
            [imageData writeToFile:[self imageFilePath:imageUrl] atomically:YES]; 
 
 
            // 回到主線程完成UI設置 
            dispatch_async(dispatch_get_main_queue(), ^{ 
 
 
                //將下載的圖像,存入newsItem對象中 
                UIImage * image = [UIImage imageWithData:imageData]; 
                self.newsItem.newsPic = image; 
 
 
                //使用block實現回調,通知圖像下載完成 
                if (_completionHandler) { 
                    _completionHandler(); 
                } 
                 
            }); 
             
        }); 
    } 
     

 
#pragma mark - 加載本地圖像 
- (UIImage *)loadLocalImage:(NSString *)imageUrl 

 
    self.imageUrl = imageUrl; 
 
 
    // 獲取圖像路徑 
    NSString * filePath = [self imageFilePath:self.imageUrl]; 
 
 
    UIImage * image = [UIImage imageWithContentsOfFile:filePath]; 
 
 
    if (image != nil) { 
        return image; 
    } 
 
    return nil; 

 
#pragma mark - 獲取圖像路徑 
- (NSString *)imageFilePath:(NSString *)imageUrl 

    // 獲取caches文件夾路徑 
    NSString * cachesPath = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject]; 
 
 
    // 創建DownloadImages文件夾 
    NSString * downloadImagesPath = [cachesPath stringByAppendingPathComponent:@"DownloadImages"]; 
    NSFileManager * fileManager = [NSFileManager defaultManager]; 
    if (![fileManager fileExistsAtPath:downloadImagesPath]) { 
 
 
        [fileManager createDirectoryAtPath:downloadImagesPath withIntermediateDirectories:YES attributes:nil error:nil]; 
    } 
 
 
#pragma mark 拼接圖像文件在沙盒中的路徑,因為圖像URL有"/",要在存入前替換掉,隨意用"_"代替 
    NSString * imageName = [imageUrl stringByReplacingOccurrencesOfString:@"/" withString:@"_"]; 
    NSString * imageFilePath = [downloadImagesPath stringByAppendingPathComponent:imageName]; 
 
 
    return imageFilePath; 

 
@end 

復制代碼 代碼如下:

@這里只給出關鍵代碼,網絡請求,數據處理,自定義cell自行解決 
 
#pragma mark - Table view data source 
 
- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView 

    // Return the number of sections. 
    return 1; 

 
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section 

    // Return the number of rows in the section. 
    if (_dataArray.count == 0) { 
        return 10; 
    } 
    return [_dataArray count]; 

 
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath 

    static NSString *cellIdentifier = @"Cell"; 
    NewsListCell *cell = [tableView dequeueReusableCellWithIdentifier:cellIdentifier ]; 
    if (!cell) { 
        cell = [[[NewsListCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:cellIdentifier] autorelease]; 
    } 
 
    NewsItem * item = [_dataArray objectAtIndex:indexPath.row]; 
 
    cell.titleLabel.text = item.newsTitle; 
 
    //判斷將要展示的新聞有無圖像 
 
    if (item.newsPic == nil) { 
        //沒有圖像下載 
        cell.picImageView.image = nil; 
         
        NSLog(@"dragging = %d,decelerating = %d",self.tableView.dragging,self.tableView.decelerating); 
        // ??執行的時機與次數問題 
        if (self.tableView.dragging == NO && self.tableView.decelerating == NO) { 
            [self startPicDownload:item forIndexPath:indexPath]; 
        } 
 
    }else{ 
        //有圖像直接展示 
        NSLog(@"1111"); 
        cell.picImageView.image = item.newsPic; 
 
    } 
     
    cell.titleLabel.text = [NSString stringWithFormat:@"indexPath.row = %ld",indexPath.row]; 
 
    return cell; 

 
- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath 

    return [NewsListCell cellHeight]; 

 
//開始下載圖像 
- (void)startPicDownload:(NewsItem *)item forIndexPath:(NSIndexPath *)indexPath 

    //創建圖像下載器 
    ImageDownloader * downloader = [[ImageDownloader alloc] init]; 
 
    //下載器要下載哪個新聞的圖像,下載完成后,新聞保存圖像 
    downloader.newsItem = item; 
 
    //傳入下載完成后的回調函數 
    [downloader setCompletionHandler:^{ 
 
        //下載完成后要執行的回調部分,block的實現 
        //根據indexPath獲取cell對象,并加載圖像 
#pragma mark cellForRowAtIndexPath-->沒看到過 
        NewsListCell * cell = (NewsListCell *)[self.tableView cellForRowAtIndexPath:indexPath]; 
        cell.picImageView.image = downloader.newsItem.newsPic; 
 
    }]; 
 
    //開始下載 
    [downloader startDownloadImage:item.newsPicUrl]; 
 
    [downloader release]; 

 
 
- (void)loadImagesForOnscreenRows 

#pragma mark indexPathsForVisibleRows-->沒看到過 
    //獲取tableview正在window上顯示的cell,加載這些cell上圖像。通過indexPath可以獲取該行上需要展示的cell對象 
    NSArray * visibleCells = [self.tableView indexPathsForVisibleRows]; 
    for (NSIndexPath * indexPath in visibleCells) { 
        NewsItem * item = [_dataArray objectAtIndex:indexPath.row]; 
        if (item.newsPic == nil) { 
            //如果新聞還沒有下載圖像,開始下載 
            [self startPicDownload:item forIndexPath:indexPath]; 
        } 
    } 

 
#pragma mark - 延遲加載關鍵 
//tableView停止拖拽,停止滾動 
- (void)scrollViewDidEndDragging:(UIScrollView *)scrollView willDecelerate:(BOOL)decelerate 

    //如果tableview停止滾動,開始加載圖像 
    if (!decelerate) { 
 
        [self loadImagesForOnscreenRows]; 
    } 
     NSLog(@"%s__%d__|%d",__FUNCTION__,__LINE__,decelerate); 

 
- (void)scrollViewDidEndDecelerating:(UIScrollView *)scrollView 

    //如果tableview停止滾動,開始加載圖像 
    [self loadImagesForOnscreenRows]; 
 
}

下拉刷新和上拉加載的原理
很多App中,新聞或者展示類都存在下拉刷新和上拉加載的效果,網上提供了實現這種效果的第三方類(詳情請見MJRefresh和EGOTableViewPullRefresh),用起來很方便,但是閑暇之余,我們可以思考下,這種效果實現的原理是什么,我以前說過,只要是動畫都是騙人的,只要不是硬件問題大部分效果都能在系統UI的基礎上做出來.
下面是關鍵代碼分析:

復制代碼 代碼如下:

// 下拉刷新的原理 
- (void)scrollViewWillBeginDecelerating:(UIScrollView *)scrollView 

    if (scrollView.contentOffset.y < - 100) { 
         
        [UIView animateWithDuration:1.0 animations:^{ 
             
            //  frame發生偏移,距離頂部150的距離(可自行設定) 
            self.tableView.contentInset = UIEdgeInsetsMake(150.0f, 0.0f, 0.0f, 0.0f); 
        } completion:^(BOOL finished) { 
             
            /**
             *  發起網絡請求,請求刷新數據
             */ 
 
        }]; 
    } 

 
// 上拉加載的原理 
- (void)scrollViewDidEndDragging:(UIScrollView *)scrollView willDecelerate:(BOOL)decelerate 

     
    NSLog(@"%f",scrollView.contentOffset.y); 
    NSLog(@"%f",scrollView.frame.size.height); 
    NSLog(@"%f",scrollView.contentSize.height); 
    /**
     *  關鍵-->
     *  scrollView一開始并不存在偏移量,但是會設定contentSize的大小,所以contentSize.height永遠都會比contentOffset.y高一個手機屏幕的
     *  高度;上拉加載的效果就是每次滑動到底部時,再往上拉的時候請求更多,那個時候產生的偏移量,就能讓contentOffset.y + 手機屏幕尺寸高大于這
     *  個滾動視圖的contentSize.height
     */ 
    if (scrollView.contentOffset.y + scrollView.frame.size.height >= scrollView.contentSize.height) { 
         
        NSLog(@"%d %s",__LINE__,__FUNCTION__); 
        [UIView commitAnimations]; 
         
        [UIView animateWithDuration:1.0 animations:^{ 
            //  frame發生的偏移量,距離底部往上提高60(可自行設定) 
            self.tableView.contentInset = UIEdgeInsetsMake(0, 0, 60, 0); 
        } completion:^(BOOL finished) { 
             
            /**
             *  發起網絡請求,請求加載更多數據
             *  然后在數據請求回來的時候,將contentInset改為(0,0,0,0)
             */ 
        }]; 
 
    } 

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产欧美最新羞羞视频在线观看| 中文字幕久久久av一区| 成人淫片在线看| 久久久久久久久久av| 国内精品免费午夜毛片| 欧美国产极速在线| 正在播放国产一区| 亚洲人成网7777777国产| 日韩av毛片网| zzjj国产精品一区二区| 成人亚洲欧美一区二区三区| 国产精品久久久久999| 全球成人中文在线| 精品久久国产精品| 欧美午夜片欧美片在线观看| 97色在线视频观看| 伦理中文字幕亚洲| …久久精品99久久香蕉国产| 国产自产女人91一区在线观看| 亚洲国产精彩中文乱码av在线播放| 日韩一区二区av| 九九热r在线视频精品| 国产日韩欧美日韩| 亚洲欧美国产日韩天堂区| 成人写真视频福利网| 精品久久国产精品| 欧美体内谢she精2性欧美| 国产精品第一页在线| 欧美性猛交xxxx乱大交| 欧美另类极品videosbest最新版本| 亚洲最大av在线| 久久99热精品| 色综合伊人色综合网| 97婷婷涩涩精品一区| 亚洲成人黄色在线观看| 国产69久久精品成人看| 91久久国产综合久久91精品网站| 亚洲视频欧洲视频| www.日韩av.com| 国产精品第2页| 亚洲精品视频网上网址在线观看| 亚洲a级在线观看| 97香蕉超级碰碰久久免费软件| 国产香蕉精品视频一区二区三区| 2024亚洲男人天堂| 国产一区二区三区久久精品| 日韩一区av在线| 欧美午夜丰满在线18影院| 在线看日韩av| 精品久久久在线观看| 狠狠色香婷婷久久亚洲精品| 亚洲影院在线看| 国产成人拍精品视频午夜网站| 久久久久久久久爱| 久久久久久久久网站| 日韩欧美主播在线| 日产精品久久久一区二区福利| 日韩专区在线播放| 久久国产精品电影| 97在线免费观看视频| 45www国产精品网站| 亚洲国产三级网| 精品国产欧美成人夜夜嗨| 日韩高清a**址| 在线免费观看羞羞视频一区二区| 欧美性猛交xxxx乱大交极品| www.美女亚洲精品| 91禁外国网站| 国内精品久久久| 91亚洲精品一区二区| 国产精品主播视频| 2025国产精品视频| 亚洲欧美日韩中文在线制服| 精品呦交小u女在线| 精品久久久久久久久久| 欧美另类99xxxxx| 精品成人久久av| 视频直播国产精品| 欧美激情在线一区| 日韩电影免费在线观看| 欧美精品videosex牲欧美| 欧美专区第一页| 精品国产欧美一区二区三区成人| 久久久精品999| 日本久久久久久| 亚洲天堂免费观看| 911国产网站尤物在线观看| 国产精品国产三级国产aⅴ浪潮| 4438全国亚洲精品在线观看视频| www.午夜精品| 国产成人精品综合久久久| 亚洲欧美综合区自拍另类| 在线日韩精品视频| 欧美成人精品xxx| 欧美成人免费全部观看天天性色| 亚洲xxx视频| 国产精品香蕉在线观看| 久久噜噜噜精品国产亚洲综合| 亚洲精品国产精品乱码不99按摩| 亚洲精品www久久久| 一个人看的www欧美| 欧美影院在线播放| 91亚洲国产成人精品性色| 这里只有精品视频| 欧美xxxx综合视频| 亚洲第一精品电影| 国产精品专区第二| 色狠狠av一区二区三区香蕉蜜桃| 欧美区在线播放| 国产日韩av在线| 久久精品久久久久| 美女扒开尿口让男人操亚洲视频网站| 成人免费直播live| 久久中国妇女中文字幕| 国产精品成人观看视频国产奇米| 亚洲日本中文字幕| 亚洲片国产一区一级在线观看| 91香蕉嫩草神马影院在线观看| 日韩欧美第一页| 欧美在线视频a| 97视频人免费观看| 91国产视频在线播放| 欧美在线观看网址综合| 精品国内自产拍在线观看| 欧美做爰性生交视频| 国产女人精品视频| 97精品国产91久久久久久| 久久人人97超碰精品888| 久久久久久伊人| 日韩欧美国产激情| 国产精品日韩在线观看| 日韩黄色高清视频| 日本精品一区二区三区在线播放视频| 国产日韩精品在线观看| 久久久久国产精品免费网站| 5566日本婷婷色中文字幕97| 九九热r在线视频精品| 欧美刺激性大交免费视频| 久久国产精品久久久久| 国产精品美女www爽爽爽视频| 国产激情久久久久| 亚洲精品一区在线观看香蕉| 国产精品久久婷婷六月丁香| 精品露脸国产偷人在视频| 日韩在线视频国产| 国产精品中文字幕久久久| 国产精品旅馆在线| 欧美性猛交xxxx| 亚洲激情视频在线观看| 97精品伊人久久久大香线蕉| 伊人久久精品视频| 亚洲国产精品va在线观看黑人| 日韩在线视频中文字幕| 成人免费福利视频| 91精品国产自产在线老师啪| 色婷婷av一区二区三区在线观看| 欧美中文字幕在线播放| 日韩欧美国产免费播放| 亚洲成人精品视频在线观看| 一区二区中文字幕| 性欧美激情精品| 97视频在线观看亚洲| 亚洲图片制服诱惑| 久久久国产成人精品|