IOS开发之异步加载网络图片并缓存本地实现瀑布流(二)
- /*
- * @brief 图片加载通用函数
- * @parma imageName 图片名
- */
- - (void)imageStartLoading:(NSString *)imageName{
- NSURL *url = [NSURL URLWithString:imageName];
- if([_fileUtil hasCachedImage:url]){
- UIImageView *imageView = [[UIImageView alloc] init];
- NSString *path = [_fileUtil pathForUrl:url];
- imageView = [_imageLoad compressImage:MY_WIDTH/3 imageView:nil imageName:path flag:NO];
- [self addImage:imageView name:path];
- [self adjustContentSize:NO];
- }else{
- UIImageView *imageView = [[UIImageView alloc] init];
- NSDictionary *dic = [NSDictionary dictionaryWithObjectsAndKeys:url, @"URL",
- imageView, @"imageView", nil nil];
- [NSThread detachNewThreadSelector:@selector(cacheImage:) toTarget:[ImageCacher shareInstance] withObject:dic];
- }
- }
这个函数的作用是为每一张网络图片开启一个下载线程,但是因为该程序用到了图片缓存的技术,所以在每次开线程下载图片的时候都会去本地缓存目录查找一下,
该图片是否已经存在,如果存在则直接加载在视图中。一般OC的线程函数有三个,NSThread, Cocoa Operations,和GCD,(想要了解三者的异同点可查看:点击打开链接),
这里我用了比较轻量级的NSThread,detachNewThreadSelector函数中所传的函数名:
cacheImage是类ImageCache中得函数,这里通过iOS开发中使用的比较多的单例模式,
得到了ImageCache的句柄,参数dic中主要存放了图片的网络地址以及imageView用来add图片进视图以及根据图片的大小压缩成合适的大小.
接下来是cacheImage函数:
- - (void)cacheImage:(NSDictionary*)dic{
- NSURL *url = [dic objectForKey:@"URL"];
- NSFileManager *fileManage = [NSFileManager defaultManager];
- NSData *data = [NSData dataWithContentsOfURL:url];
- NSString *fileName = [_fileUtil pathForUrl:url];
- if(data){
- [fileManage createFileAtPath:fileName contents:data attributes:nil];
- }
- UIImageView *imageView = [dic objectForKey:@"imageView"];
- imageView.image = [UIImage imageWithData:data];
- imageView = [_imageLoader compressImage:MY_WIDTH/3 imageView:imageView imageName:nil flag:YES];
- [self.myDelegate addImage:imageView name:fileName];
- [self.myDelegate adjustContentSize:NO];
- }
该函数用来将下载下来的图片缓存进入文件沙盒中(缓存文件可以自己定义并指定),并且按照图片的大小进行等比例压缩,固定宽度是屏幕的三分之一大小,这样一来,
图片显示就不会出现不全或失真的现象。由于ImageCache和MyScrollView是两个独立的类,所以这里通过使用ios的delegate(代理)来进行图片在scrollView上的加载,
(什么是代理模式:点击打开链接).
下面我们来看如何在沙盒中建立缓存文件夹,其实缓存文件夹跟普通的文件夹一样,只是该文件夹是专门用来存放缓存文件的而已。类代码如下所示:
- //
- // FileUtil.m
- // Test515
- //
- // Created by silicon on 14-5-30.
- // Copyright (c) 2014年 silicon. All rights reserved.
- //
- #import "FileUtil.h"
- @implementation FileUtil
- + (FileUtil *)shareInstance{
- static FileUtil *instance;
- static dispatch_once_t onceToken;
- dispatch_once(&onceToken, ^{
- instance = [[self alloc] init];
- });
- return instance;
- }
- /*
- @breif 创建缓存文件夹
- */
- - (void)createPathInDocumentDirectory{
- NSArray *paths = NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES);
- NSString *diskCachePath = [[[paths objectAtIndex:0] stringByAppendingPathComponent:@"ImageCache"] retain];
- NSLog(@"%@", diskCachePath);
- if(![[NSFileManager defaultManager] fileExistsAtPath:diskCachePath]){
- NSError *error = nil;
- [[NSFileManager defaultManager] createDirectoryAtPath:diskCachePath
- withIntermediateDirectories:YES
- attributes:nil
- error:&error];
- }
- }
- /*
- @breif 获取沙盒中文档目录
- @param fileName:文件名字
- */
- - (NSString *)pathInDocumentDirectory:(NSString *)fileName{
- NSArray *fileArray = NSSearchPathForDirectoriesInDomains(NSCachesDirectory,
- NSUserDomainMask, YES);
- NSString *cacheDirectory = [fileArray objectAtIndex:0];
- return [cacheDirectory stringByAppendingPathComponent:fileName];
- }
- /*
- @breif 获取沙盒中缓存文件目录
- @param fileName:文件名字
- */
- - (NSString *)pathInCacheDirectory:(NSString *)fileName{
- NSArray *fileArray = NSSearchPathForDirectoriesInDomains(NSCachesDirectory,
- NSUserDomainMask, YES);
- NSString *cacheDirectory = [fileArray objectAtIndex:0];
- return [cacheDirectory stringByAppendingPathComponent:fileName];
- }
- /*
- @breif 判断是否已经缓存
- @param url:图片名称
- */
- - (BOOL)hasCachedImage:(NSURL *)url{
- NSFileManager *fileManager = [NSFileManager defaultManager];
- if([fileManager fileExistsAtPath:[self pathForUrl:url]]){
- return YES;
- }else{
- return NO;
- }
- }
- /*
- @breif 根据URL的給图片命名
- @param url:图片url
- */
- - (NSString *)pathForUrl:(NSURL *)url{
- return [self pathInCacheDirectory:[NSString stringWithFormat:@"qiaoqiao-%u", [[url description] hash]]];
- }
- @end
在这次的demo中,我新加入了用户可以点击图片放大 并可以左右滑动的功能,其实实现起来很简单,我一开始为每一个ScrollView 中得ImageView都设置了tag值,并且添加了
手势(UITapGestureRecognizer),当用户点击图片时,程序可以根据点击视图的tag值来获得相应的图片是哪一张,从而可以加载。支持左右滑动的功能在新的界面中增加了
一个ScrollView,然后将下载下来的图片添加到scrollView中。代码如下
- //
- // PhotoViewController.m
- // Test515
- //
- // Created by silicon on 14-5-22.
- // Copyright (c) 2014年 silicon. All rights reserved.
- //
- #import "PhotoViewController.h"
- #import "ImageLoader.h"
- @interface PhotoViewController ()
- @end
- @implementation PhotoViewController
- @synthesize scrollView = _scrollView;
- @synthesize imageArray = _imageArray;
- @synthesize page = _page;
- - (id)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil
- {
- self = [super initWithNibName:nibNameOrNil bundle:nibBundleOrNil];
- if (self) {
- // Custom initialization
- }
- return self;
- }
- - (void)viewDidLoad
- {
- [super viewDidLoad];
- // Do any additional setup after loading the view.
- [self.view setFrame:CGRectMake(0, 0, MY_WIDTH, MY_HEIGHT)];
- [self.view setBackgroundColor:[UIColor blackColor]];
- self.scrollView = [[UIScrollView alloc] initWithFrame:CGRectMake(0, 20, MY_WIDTH, MY_HEIGHT)];
- _scrollView.delegate = self;
- _scrollView.contentSize = CGSizeMake(MY_WIDTH * [_imageArray count], MY_HEIGHT);
- _scrollView.showsVerticalScrollIndicator = NO;
- _scrollView.showsHorizontalScrollIndicator = NO;
- _scrollView.backgroundColor = [UIColor blackColor];
- _scrollView.bounces = YES;
- _scrollView.pagingEnabled = YES;
- [self.view addSubview:_scrollView];
- //图片添加事件响应
- UITapGestureRecognizer *tapRecognizer = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(closePhotoView)];
- tapRecognizer.delegate = self;
- _scrollView.userInteractionEnabled = YES;
- [_scrollView addGestureRecognizer:tapRecognizer];
- [tapRecognizer release];
- [self loadingImages];
- }
- - (void)viewWillAppear:(BOOL)animated{
- [_scrollView setContentOffset:CGPointMake([_imageArray indexOfObject:_imageName] * MY_WIDTH, 0)];
- }
- - (void)didReceiveMemoryWarning
- {
- [super didReceiveMemoryWarning];
- // Dispose of any resources that can be recreated.
- }
- //关闭
- - (void)closePhotoView{
- [self.view removeFromSuperview];
- }
- - (void)dealloc{
- [_scrollView release];
- [super dealloc];
- }
- - (void)loadingImages{
- //加载图片
- for(int i = 0; i < [_imageArray count]; i++){
- NSString *picName = [_imageArray objectAtIndex:i];
- UIImageView *imageV = [[ImageLoader shareInstance] compressImage:MY_WIDTH imageView:nil imageName:picName flag:NO];
- float width = imageV.image.size.width;
- float height = imageV.image.size.height;
- float new_width = MY_WIDTH;
- float new_height = (MY_WIDTH * height)/width;
- imageV.frame = CGRectMake(MY_WIDTH * i, 0, new_width, new_height);
- [_scrollView addSubview:imageV];
- [imageV release];
- }
- }
- - (void)scrollViewDidScroll:(UIScrollView *)_scrollView{
- }
- @end
IOS开发之异步加载网络图片并缓存本地实现瀑布流(二)的更多相关文章
- wemall app商城源码Android之ListView异步加载网络图片(优化缓存机制)
wemall-mobile是基于WeMall的android app商城,只需要在原商城目录下上传接口文件即可完成服务端的配置,客户端可定制修改.本文分享wemall app商城源码Android之L ...
- (BUG已修改,最优化)安卓ListView异步加载网络图片与缓存软引用图片,线程池,只加载当前屏之说明
原文:http://blog.csdn.net/java_jh/article/details/20068915 迟点出更新的.这个还有BUG.因为软引应不给力了.2.3之后 前几天的原文有一个线程管 ...
- android官方开源的高性能异步加载网络图片的Gridview例子
这个是我在安卓安卓巴士上看到的资料,放到这儿共享下.这个例子android官方提供的,其中讲解了如何异步加载网络图片,以及在gridview中高效率的显示图片此代码很好的解决了加载大量图片时,报OOM ...
- Libgdx实现异步加载网络图片并保存到SD卡或者data/data目录下边
Libgdx实现异步加载网络图片并保存到SD卡或者data/data目录下边,当本地有图片的时候,直接从本地读取图片,如果本地没有图片,将从服务器异步加载图片 package com.example. ...
- ios UIImageView异步加载网络图片
方法1:在UI线程中同步加载网络图片 UIImageView *headview = [[UIImageView alloc] initWithFrame:CGRectMake(0, 0, 40, 4 ...
- UIImageView异步加载网络图片
在iOS开发过程中,经常会遇到使用UIImageView展现来自网络的图片的情况,最简单的做法如下: 去下载https://github.com/rs/SDWebImage放进你的工程里,加入头文件# ...
- Android批量图片加载经典系列——采用二级缓存、异步加载网络图片
一.问题描述 Android应用中经常涉及从网络中加载大量图片,为提升加载速度和效率,减少网络流量都会采用二级缓存和异步加载机制,所谓二级缓存就是通过先从内存中获取.再从文件中获取,最后才会访问网络. ...
- Android批量图片加载经典系列——使用xutil框架缓存、异步加载网络图片
一.问题描述 为提高图片加载的效率,需要对图片的采用缓存和异步加载策略,编码相对比较复杂,实际上有一些优秀的框架提供了解决方案,比如近期在git上比较活跃的xutil框架 Xutil框架提供了四大模块 ...
- IOS中UITableView异步加载图片的实现
本文转载至 http://blog.csdn.net/enuola/article/details/8639404 最近做一个项目,需要用到UITableView异步加载图片的例子,看到网上有一个E ...
随机推荐
- 未能加载文件或程序集“Newtonsoft.Json, Version=4.0.0.0, Culture=neutral, PublicKeyToken=30a [问题点数:40分,结帖人u010259408]
未能加载文件或程序集“Newtonsoft.Json, Version=4.0.0.0, Culture=neutral, PublicKeyToken=30a [问题点数:40分,结帖人u01025 ...
- 四则运算之C++实现篇
对四则运算的一些要求如下: 1.题目避免重复:2.可定制(数量/打印方式):3.可以控制下列参数: 是否有乘除法.数值范围.加减有无负数.除法有无余数.否支持分数 (真分数, 假分数, …): 一 ...
- Struts2的流程(三)
Struts的流程图如下(需要完全理解):
- buddy算法
buddy算法是用来做内存管理的经典算法,目的是为了解决内存的外碎片.避免外碎片的方法有两种: 1,利用分页单元把一组非连续的空闲页框映射到非连续的线性地址区间. 2,开发适当的技术来记录现存的空闲连 ...
- Debug阶段成员贡献分
组名:天天向上 组长:王森 组员:张政.张金生.林莉.胡丽娜 Debug阶段各组员的贡献分分配如下: 姓名 个人工作量 组长评价 个人评价 团队贡献总分 张金生 11 7 6 6.00 张政 11 6 ...
- mysql替换制定的内容的 类似正则表达式的功能
content= 'asnfojassozxpdsgdspdps神龙架谁骄傲的骄傲搜ID飞机扫' SELECT content FROM test WHERE id =1 吧zx替换成ZZZZ UPD ...
- Go-Agent原理分析及FQ介绍
作为一个程序员,相信大家是极度依赖google/stackoverflow/github的,可是国内有强大的GFW存在,以至于编程少了很多乐趣. 最近闹GFW狂潮,很多Chrome插件被封,连Shad ...
- rabbitmq之amqp queue
rabbitmq作为一个消息中间件,暂存信息的能力是必不可少的. 镜像队列
- 解决pip国外安装源慢的问题
用默认的pip安装源pypi.python.org由于在国外经常会出现超时的问题,而且安装速度极其的慢,如下图中的超时问题=>
- 用Appium进行android自动化测试
appium是开源的移动端自动化测试框架,可以测试ios,android应用.appium让移动端自动化测试不必限定在某种语言和某个具体的框架:也就是说任何人都可以使用自己最熟悉最顺手的语言以及框架来 ...