IOS开发之异步加载网络图片并缓存本地实现瀑布流(二)

  1. /*
  2. * @brief 图片加载通用函数
  3. * @parma imageName 图片名
  4. */
  5. - (void)imageStartLoading:(NSString *)imageName{
  6. NSURL *url = [NSURL URLWithString:imageName];
  7. if([_fileUtil hasCachedImage:url]){
  8. UIImageView *imageView = [[UIImageView alloc] init];
  9. NSString *path = [_fileUtil pathForUrl:url];
  10. imageView = [_imageLoad compressImage:MY_WIDTH/3 imageView:nil imageName:path flag:NO];
  11. [self addImage:imageView name:path];
  12. [self adjustContentSize:NO];
  13. }else{
  14. UIImageView *imageView = [[UIImageView alloc] init];
  15. NSDictionary *dic = [NSDictionary dictionaryWithObjectsAndKeys:url, @"URL",
  16. imageView, @"imageView", nil nil];
  17. [NSThread detachNewThreadSelector:@selector(cacheImage:) toTarget:[ImageCacher shareInstance] withObject:dic];
  18. }
  19. }

这个函数的作用是为每一张网络图片开启一个下载线程,但是因为该程序用到了图片缓存的技术,所以在每次开线程下载图片的时候都会去本地缓存目录查找一下,

该图片是否已经存在,如果存在则直接加载在视图中。一般OC的线程函数有三个,NSThread, Cocoa Operations,和GCD,(想要了解三者的异同点可查看:点击打开链接),

这里我用了比较轻量级的NSThread,detachNewThreadSelector函数中所传的函数名:
cacheImage是类ImageCache中得函数,这里通过iOS开发中使用的比较多的单例模式,

得到了ImageCache的句柄,参数dic中主要存放了图片的网络地址以及imageView用来add图片进视图以及根据图片的大小压缩成合适的大小.

接下来是cacheImage函数:

[objc] view plain copy

  1. - (void)cacheImage:(NSDictionary*)dic{
  2. NSURL *url = [dic objectForKey:@"URL"];
  3. NSFileManager *fileManage = [NSFileManager defaultManager];
  4. NSData *data = [NSData dataWithContentsOfURL:url];
  5. NSString *fileName = [_fileUtil pathForUrl:url];
  6. if(data){
  7. [fileManage createFileAtPath:fileName contents:data attributes:nil];
  8. }
  9. UIImageView *imageView = [dic objectForKey:@"imageView"];
  10. imageView.image = [UIImage imageWithData:data];
  11. imageView = [_imageLoader compressImage:MY_WIDTH/3 imageView:imageView imageName:nil flag:YES];
  12. [self.myDelegate addImage:imageView name:fileName];
  13. [self.myDelegate adjustContentSize:NO];
  14. }

该函数用来将下载下来的图片缓存进入文件沙盒中(缓存文件可以自己定义并指定),并且按照图片的大小进行等比例压缩,固定宽度是屏幕的三分之一大小,这样一来,

图片显示就不会出现不全或失真的现象。由于ImageCache和MyScrollView是两个独立的类,所以这里通过使用ios的delegate(代理)来进行图片在scrollView上的加载,

(什么是代理模式:点击打开链接).

下面我们来看如何在沙盒中建立缓存文件夹,其实缓存文件夹跟普通的文件夹一样,只是该文件夹是专门用来存放缓存文件的而已。类代码如下所示:

[objc] view plain copy

  1. //
  2. //  FileUtil.m
  3. //  Test515
  4. //
  5. //  Created by silicon on 14-5-30.
  6. //  Copyright (c) 2014年 silicon. All rights reserved.
  7. //
  8. #import "FileUtil.h"
  9. @implementation FileUtil
  10. + (FileUtil *)shareInstance{
  11. static FileUtil *instance;
  12. static dispatch_once_t onceToken;
  13. dispatch_once(&onceToken, ^{
  14. instance = [[self alloc] init];
  15. });
  16. return instance;
  17. }
  18. /*
  19. @breif 创建缓存文件夹
  20. */
  21. - (void)createPathInDocumentDirectory{
  22. NSArray *paths = NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES);
  23. NSString *diskCachePath = [[[paths objectAtIndex:0] stringByAppendingPathComponent:@"ImageCache"] retain];
  24. NSLog(@"%@", diskCachePath);
  25. if(![[NSFileManager defaultManager] fileExistsAtPath:diskCachePath]){
  26. NSError *error = nil;
  27. [[NSFileManager defaultManager] createDirectoryAtPath:diskCachePath
  28. withIntermediateDirectories:YES
  29. attributes:nil
  30. error:&error];
  31. }
  32. }
  33. /*
  34. @breif     获取沙盒中文档目录
  35. @param     fileName:文件名字
  36. */
  37. - (NSString *)pathInDocumentDirectory:(NSString *)fileName{
  38. NSArray *fileArray = NSSearchPathForDirectoriesInDomains(NSCachesDirectory,
  39. NSUserDomainMask, YES);
  40. NSString *cacheDirectory = [fileArray objectAtIndex:0];
  41. return [cacheDirectory stringByAppendingPathComponent:fileName];
  42. }
  43. /*
  44. @breif     获取沙盒中缓存文件目录
  45. @param     fileName:文件名字
  46. */
  47. - (NSString *)pathInCacheDirectory:(NSString *)fileName{
  48. NSArray *fileArray = NSSearchPathForDirectoriesInDomains(NSCachesDirectory,
  49. NSUserDomainMask, YES);
  50. NSString *cacheDirectory = [fileArray objectAtIndex:0];
  51. return [cacheDirectory stringByAppendingPathComponent:fileName];
  52. }
  53. /*
  54. @breif     判断是否已经缓存
  55. @param     url:图片名称
  56. */
  57. - (BOOL)hasCachedImage:(NSURL *)url{
  58. NSFileManager *fileManager = [NSFileManager defaultManager];
  59. if([fileManager fileExistsAtPath:[self pathForUrl:url]]){
  60. return YES;
  61. }else{
  62. return NO;
  63. }
  64. }
  65. /*
  66. @breif     根据URL的給图片命名
  67. @param     url:图片url
  68. */
  69. - (NSString *)pathForUrl:(NSURL *)url{
  70. return [self pathInCacheDirectory:[NSString stringWithFormat:@"qiaoqiao-%u", [[url description] hash]]];
  71. }
  72. @end

在这次的demo中,我新加入了用户可以点击图片放大 并可以左右滑动的功能,其实实现起来很简单,我一开始为每一个ScrollView 中得ImageView都设置了tag值,并且添加了

手势(UITapGestureRecognizer),当用户点击图片时,程序可以根据点击视图的tag值来获得相应的图片是哪一张,从而可以加载。支持左右滑动的功能在新的界面中增加了

一个ScrollView,然后将下载下来的图片添加到scrollView中。代码如下

[objc] view plain copy

    1. //
    2. //  PhotoViewController.m
    3. //  Test515
    4. //
    5. //  Created by silicon on 14-5-22.
    6. //  Copyright (c) 2014年 silicon. All rights reserved.
    7. //
    8. #import "PhotoViewController.h"
    9. #import "ImageLoader.h"
    10. @interface PhotoViewController ()
    11. @end
    12. @implementation PhotoViewController
    13. @synthesize scrollView = _scrollView;
    14. @synthesize imageArray = _imageArray;
    15. @synthesize page = _page;
    16. - (id)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil
    17. {
    18. self = [super initWithNibName:nibNameOrNil bundle:nibBundleOrNil];
    19. if (self) {
    20. // Custom initialization
    21. }
    22. return self;
    23. }
    24. - (void)viewDidLoad
    25. {
    26. [super viewDidLoad];
    27. // Do any additional setup after loading the view.
    28. [self.view setFrame:CGRectMake(0, 0, MY_WIDTH, MY_HEIGHT)];
    29. [self.view setBackgroundColor:[UIColor blackColor]];
    30. self.scrollView = [[UIScrollView alloc] initWithFrame:CGRectMake(0, 20, MY_WIDTH, MY_HEIGHT)];
    31. _scrollView.delegate = self;
    32. _scrollView.contentSize = CGSizeMake(MY_WIDTH * [_imageArray count], MY_HEIGHT);
    33. _scrollView.showsVerticalScrollIndicator = NO;
    34. _scrollView.showsHorizontalScrollIndicator = NO;
    35. _scrollView.backgroundColor = [UIColor blackColor];
    36. _scrollView.bounces = YES;
    37. _scrollView.pagingEnabled = YES;
    38. [self.view addSubview:_scrollView];
    39. //图片添加事件响应
    40. UITapGestureRecognizer *tapRecognizer = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(closePhotoView)];
    41. tapRecognizer.delegate = self;
    42. _scrollView.userInteractionEnabled = YES;
    43. [_scrollView addGestureRecognizer:tapRecognizer];
    44. [tapRecognizer release];
    45. [self loadingImages];
    46. }
    47. - (void)viewWillAppear:(BOOL)animated{
    48. [_scrollView setContentOffset:CGPointMake([_imageArray indexOfObject:_imageName] * MY_WIDTH, 0)];
    49. }
    50. - (void)didReceiveMemoryWarning
    51. {
    52. [super didReceiveMemoryWarning];
    53. // Dispose of any resources that can be recreated.
    54. }
    55. //关闭
    56. - (void)closePhotoView{
    57. [self.view removeFromSuperview];
    58. }
    59. - (void)dealloc{
    60. [_scrollView release];
    61. [super dealloc];
    62. }
    63. - (void)loadingImages{
    64. //加载图片
    65. for(int i = 0; i < [_imageArray count]; i++){
    66. NSString *picName = [_imageArray objectAtIndex:i];
    67. UIImageView *imageV = [[ImageLoader shareInstance] compressImage:MY_WIDTH imageView:nil imageName:picName flag:NO];
    68. float width = imageV.image.size.width;
    69. float height = imageV.image.size.height;
    70. float new_width = MY_WIDTH;
    71. float new_height = (MY_WIDTH * height)/width;
    72. imageV.frame = CGRectMake(MY_WIDTH * i, 0, new_width, new_height);
    73. [_scrollView addSubview:imageV];
    74. [imageV release];
    75. }
    76. }
    77. - (void)scrollViewDidScroll:(UIScrollView *)_scrollView{
    78. }
    79. @end
时间: 2024-12-16 18:48:50

IOS开发之异步加载网络图片并缓存本地实现瀑布流(二)的相关文章

Android之ListView异步加载网络图片(优化缓存机制)【转】

网上关于这个方面的文章也不少,基本的思路是线程+缓存来解决.下面提出一些优化: 1.采用线程池 2.内存缓存+文件缓存 3.内存缓存中网上很多是采用SoftReference来防止堆溢出,这儿严格限制只能使用最大JVM内存的1/4 4.对下载的图片进行按比例缩放,以减少内存的消耗 具体的代码里面说明.先放上内存缓存类的代码MemoryCache.java: public class MemoryCache { private static final String TAG = "MemoryCa

iOS开发swift版异步加载网络图片(带缓存和缺省图片)

iOS开发之swift版异步加载网络图片 与SDWebImage异步加载网络图片的功能相似,只是代码比较简单,功能没有SD的完善与强大,支持缺省添加图片,支持本地缓存. 异步加载图片的核心代码如下:  func setZYHWebImage(url:NSString?, defaultImage:NSString?, isCache:Bool){         var ZYHImage:UIImage?         if url == nil {             return   

(BUG已修改,最优化)安卓ListView异步加载网络图片与缓存软引用图片,线程池,只加载当前屏之说明

原文:http://blog.csdn.net/java_jh/article/details/20068915 迟点出更新的.这个还有BUG.因为软引应不给力了.2.3之后 前几天的原文有一个线程管理与加载源过多,造成浪费流量的问题.下面对这进下改进的一些说明(红色为新加) 这两天一直在优化这个问题.google也很多种做法.但发现都是比较不全面. 比如: 一些只实现了异步加载,却没有线程池与软引用. 一些是用AsynTast的, 一些有了线程池但加载所有的图片,这样造成具大资源浪费 一些是用

Android 异步加载网络图片并缓存到本地

在android应用开发的时候,加载网络图片是一个非常重要的部分,很多图片不可能放在本地,所以就必须要从服务器或者网络读取图片. 软引用是一个现在非常流行的方法,用户体验比较好,不用每次都需要从网络下载图片,如果下载后就存到本地,下次读取时首先查看本地有没有,如果没有再从网络读取. 记得2月份在和爱奇艺公司的项目总监一起搞联通的OTT盒子的时候他就提了一下软引用,奇艺做的手机客户端就是采用这种方法,所以你会发现奇艺客户端占用很大的空间,下面就分享一下异步加载网络图片的方法吧. FileCache

Android之ListView异步加载网络图片(优化缓存机制)

网上关于这个方面的文章也不少,基本的思路是线程+缓存来解决.下面提出一些优化: 1.采用线程池 2.内存缓存+文件缓存 3.内存缓存中网上很多是采用SoftReference来防止堆溢出,这儿严格限制只能使用最大JVM内存的1/4 4.对下载的图片进行按比例缩放,以减少内存的消耗 具体的代码里面说明.先放上内存缓存类的代码MemoryCache.java: [java] view plain copy public class MemoryCache { private static final

ios开发多线程篇--异步加载网络图片

一.异步加载网络图片 1.ATS (1)简介 从iOS9.0开始,如果按照以前的方式写代码,在访问网络的时候 ,会报以下警告信息: App Transport Security has blocked a cleartext HTTP (http://) resource load since it is insecure. Temporary exceptions can be configured via your app's Info.plist file. 原因:iOS9.0引入了新特性

扩展于RCLabel的支持异步加载网络图片的富文本引擎的设计

扩展于RCLabel的支持异步加载网络图片的富文本引擎的设计 在iOS开发中,图文混排一直都是UI编程的一个核心点,也有许多优秀的第三方引擎,其中很有名的一套图文混排的框架叫做DTCoreText.但是在前些日的做的一个项目中,我并没有采用这套框架,原因有二,一是这套框架体积非常大,而项目的需求其实并不太高:二是要在这套框架中修改一些东西,难度也非常大,我最终采用的是一个叫做RCLabel的第三方控件,经过一些简单的优化和完善,达到了项目的要求. 先来介绍一下我项目中的图文混排的需求:首先我从服

iOS_第3方类库_EGOImageView异步加载网络图片

异步加载网络图片是很常见的需求 下载,导入,-fno-objc-arc 1.导入头文件 2.创建EGOImageView,并指定占位图,设置url即可 最终效果图:(后补) 附EGO原文件地址:

ios UIImageView异步加载网络图片

方法1:在UI线程中同步加载网络图片 UIImageView *headview = [[UIImageView alloc] initWithFrame:CGRectMake(0, 0, 40, 40)]; NSURL *photourl = [NSURL URLWithString:@"http://www.exampleforphoto.com/pabb/test32.png"]; //url请求实在UI主线程中进行的 UIImage *images = [UIImage ima