当前位置: 首页 > 编程笔记 >

ios通过按钮点击异步加载图片

何晗昱
2023-03-14
本文向大家介绍ios通过按钮点击异步加载图片,包括了ios通过按钮点击异步加载图片的使用技巧和注意事项,需要的朋友参考一下

比较原始的方法:


AsyncImageView.h:

#import <UIKit/UIKit.h>

@interface AsyncImageView : UIView

{

    NSURLConnection* connection;

    NSMutableData* data;

}

- (void)loadImageFromURL:(NSURL*)url;

@end

AsyncImageView.m:

#import "AsyncImageView.h"

@implementation AsyncImageView

- (id)initWithFrame:(CGRect)frame

{

    self = [super initWithFrame:frame];

    if(self) {

        // Initialization code

    }

    returnself;

}

- (void)loadImageFromURL:(NSURL*)url {

    if(connection!=nil) { [connection release]; }

    if(data!=nil) { [data release]; }

    NSURLRequest* request = [NSURLRequest requestWithURL:url

                                             cachePolicy:NSURLRequestUseProtocolCachePolicy

                                         timeoutInterval:60.0];

    connection = [[NSURLConnection alloc]

                  initWithRequest:request delegate:self];

}

- (void)connection:(NSURLConnection *)theConnection

    didReceiveData:(NSData *)incrementalData {

    if(data==nil) {

        data =

        [[NSMutableData alloc] initWithCapacity:2048];

    }

    [data appendData:incrementalData];

}

- (void)connectionDidFinishLoading:(NSURLConnection*)theConnection {

    [connection release];

    connection=nil;

    if([[self subviews] count] > 0) {

        [[[self subviews] objectAtIndex:0] removeFromSuperview];

    }

    UIImageView *imageView = [[[UIImageView alloc] initWithImage:[UIImage imageWithData:data]] autorelease];

    imageView.contentMode = UIViewContentModeScaleAspectFit;

    imageView.autoresizingMask = ( UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleHeight );

    [self addSubview:imageView];

    imageView.frame = self.bounds;

    [imageView setNeedsLayout];

    [self setNeedsLayout];

    [data release];

    data=nil;

}

- (UIImage*) image {

    UIImageView* iv = [[self subviews] objectAtIndex:0];

    return[iv image];

}

- (void)dealloc {

    [connection cancel];

    [connection release];

    [data release];

    [super dealloc];

}

@end

方法二:


@interface UIButton (AsyncImage)

//size by point

- (void)setImageFromURL:(NSString *)urlString adjustToSize:(CGSize)size completion:(void (^)(void))completion logo:(UIImage *)logoImage; 

@end

@implementation UIButton (AsyncImage)

- (void)setImageFromURL:(NSString *)urlString adjustToSize:(CGSize)size completion:(void (^)(void))completion logo:(UIImage *)logoImage

{

    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{

        UIImage *image = nil;

        NSURL *url = [NSURL URLWithString:urlString];

        NSData *data = [NSData dataWithContentsOfURL:url];

        image = [UIImage imageWithData:data];    

        if (image) {

            if (!CGSizeEqualToSize(size, CGSizeZero)) {

                image = [UIImage imageWithCGImage:image.CGImage scale:[self scaleImage:image adjustToSize:size] orientation:image.imageOrientation];

            }

            if (logoImage) {

                image = [self addLogoImage:logoImage toImage:image];

            }

            dispatch_async(dispatch_get_main_queue(), ^{

                [self setImage:image forState:UIControlStateNormal];

                completion();

            });

        }

        else {

            NSLog(@"async load error.");

        }

    });

}

// 缩放图片以适应按钮大小

- (CGFloat)scaleImage:(UIImage *)image adjustToSize:(CGSize)size

{

    CGFloat xScale = size.width / image.size.width;

    CGFloat yScale = size.height / image.size.height;

    return 1.0 / MIN(xScale, yScale);

}

- (UIImage *)addLogoImage:(UIImage *)logo toImage:(UIImage *)img

{

    //get image width and height

    CGFloat scale = [UIScreen mainScreen].scale;

    int w = scale * img.size.width;

    int h = scale * img.size.height;

    int logoWidth = logo.scale * logo.size.width;

    int logoHeight = logo.scale * logo.size.height;

    CGColorSpaceRef colorSpace = CGColorSpaceCreateDeviceRGB();

    //create a graphic context with CGBitmapContextCreate

    CGContextRef context = CGBitmapContextCreate(NULL, w, h, 8, 4 * w, colorSpace, kCGImageAlphaPremultipliedFirst);

    CGContextDrawImage(context, CGRectMake(0, 0, w, h), img.CGImage);

    CGContextDrawImage(context, CGRectMake(w - logoWidth, 0, logoWidth, logoHeight), [logo CGImage]);

    CGImageRef imageMasked = CGBitmapContextCreateImage(context);

    CGContextRelease(context);

    CGColorSpaceRelease(colorSpace);

    return [UIImage imageWithCGImage:imageMasked scale:scale orientation:img.imageOrientation];

}

@end

方法三:

#import <Foundation/Foundation.h>
#import "StringUtils.h"

@interface ImageManager : NSObject
{
  NSMutableDictionary *_imageDict;
  NSMutableArray *_imageArr;
}

@property(nonatomic, strong) NSString *httpUrl;
@property(nonatomic, strong) NSMutableDictionary *imageDict;

@property(nonatomic, assign) dispatch_queue_t networkQueue;

+ (ImageManager *) sharedInstance;


- (void)asyncImage:(NSString *)imageName imageView:(UIImageView *)imageView;
//插队
- (void)asyncImageInsert:(NSString *)imageName imageView:(UIImageView *)imageView insert:(BOOL)insert;
//不要在下载之前的数据
- (void)asyncImageCleanOld:(NSString *)imageName imageView:(UIImageView *)imageView cleanOld:(BOOL)cleanOld;

@end

实现文件:

//
// ImageManager.m
// myb-ios
//
// Created by warrior gao on 13-6-5.
// Copyright (c) 2013年 51myb. All rights reserved.
//

#import "ImageManager.h"

@interface ImageManager()

@end

@implementation ImageManager

//缓存图片的最大数量
static int counter = 0;

@synthesize imageDict = _imageDict;

//Singleton 
+ (ImageManager *)sharedInstance
{
  static id instance;
  static dispatch_once_t onceToken;
  dispatch_once(&onceToken, ^{
    instance = self.new;
  });
  return instance;
}

- (id)init
{
  if((self = [super init]))
  {
    self.networkQueue = dispatch_queue_create("com.warrior.network.image", nil);
    _imageDict = [[NSMutableDictionary alloc] init];
    _imageArr = [[NSMutableArray alloc] init];
  }
  return self;
}

- (NSString *) fileFullPath:(NSString *)fileName
{
  NSString *cachePath = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) objectAtIndex:0];
  
  NSString *fileFullPath = [NSString stringWithFormat:@"%@/%@",cachePath,fileName];
  
  return fileFullPath;
}

//不要在下载之前的数据
- (void)asyncImageCleanOld:(NSString *)imageName imageView:(UIImageView *)imageView cleanOld:(BOOL)cleanOld
{
  if(cleanOld)
  {
    [_imageArr removeAllObjects];
  }
  
  [self asyncImage:imageName imageView:imageView];
}

//插队,优先
- (void)asyncImageInsert:(NSString *)imageName imageView:(UIImageView *)imageView insert:(BOOL)insert
{
  if([StringUtils isEmpty:imageName]){
    return;
  }
  
  NSData *data = [NSData dataWithContentsOfFile:[self fileFullPath:[imageName stringByReplacingOccurrencesOfString:@"/" withString:@"-"]]];
  if(data == nil){
    [_imageDict setValue:imageView forKey:imageName];
    if(insert)
    {
      [_imageArr insertObject:imageName atIndex:0];
    }
    else
    {
      [_imageArr addObject:imageName];
    }
    
    [self cacheImage];
  } else {
    [imageView setImage:[UIImage imageWithData:data]];
  }
}

//正常,附加到后面
- (void)asyncImage:(NSString *)imageName imageView:(UIImageView *)imageView
{
  [self asyncImageInsert:imageName imageView:imageView insert:NO];
}

//异步缓存图片到本地,最多有两个线程
-(void)cacheImage
{
  for (; counter < 2 && _imageArr.count > 0; counter++)
  {
    NSString *imageName = nil;
    @synchronized(self){
      imageName = [[_imageArr objectAtIndex:0] copy];
      [_imageArr removeObjectAtIndex:0];
    }
    
    if(imageName == nil) continue;
    
    dispatch_async(self.networkQueue, ^{
      
      NSLog(@"Starting: %@", imageName);
      UIImage *avatarImage = nil;
      NSURL *url = [NSURL URLWithString:[NSString stringWithFormat:@"%@%@",self.httpUrl, imageName]];
      NSData *responseData = [NSData dataWithContentsOfURL:url];
      if(responseData.length > 0)
      {
        [responseData writeToFile:[self fileFullPath:[imageName stringByReplacingOccurrencesOfString:@"/" withString:@"-"]] atomically:NO];
        avatarImage = [UIImage imageWithData:responseData];
        NSLog(@"Finishing: %@", imageName);
        
        if (avatarImage) {
          dispatch_async(dispatch_get_main_queue(), ^{
            UIImageView *imageView = [_imageDict objectForKey:imageName];
            if(imageView != nil && avatarImage != nil){
              [imageView setImage:avatarImage];
            }
            
            [_imageDict removeObjectForKey:imageName];
            [imageName release];
          });
        }
      }
      counter--;
      [self cacheImage];
    });
    
  }
}

@end

以上所述就是本文的全部内容 了,希望大家能够喜欢。

 类似资料:
  • HTML: 代码试用: 也试过很多其他的方法。无法单击“取消”按钮。错误: org.openqa.selenium.nosuchelementException:没有这样的元素:找不到元素:{“method”:“XPath”,“selector”:“//button[@type='button'][@class='modal-footer-button g-capitalize btn btn-l

  • 在我的android程序中,我想点击一个按钮,在不同的交通灯图像之间循环。每当应用程序加载时,它会以一个红灯的图像开始,当我单击它时,我希望它将绿灯变为绿灯,然后再次单击变为黄灯。这就是我的Java文件中的内容 和XML文件

  • 问题内容: 我正在尝试编写一个超简单的解决方案以异步加载一堆JS文件。到目前为止,我下面有以下脚本。但是,当未实际加载脚本时,有时会调用回调,这会导致未找到变量错误。如果我刷新页面有时会起作用,因为我猜这些文件是直接从缓存中提取的,因此比调用回调要快,这很奇怪吗? 无论如何,有没有方法可以测试JS文件是否已完全加载,而无需在实际的JS文件本身中放入任何东西,因为我想使用相同的模式从我的控件(GMa

  • 以下是我试图单击的按钮的HTML代码: 我可以使用XPath/Class Name/CSS选择器点击按钮的可能方式有哪些? 我做了如下尝试:

  • 这是我第一次用飞溅刮网站。我需要告诉spash点击一个按钮,以便其他元素加载到浏览器上。这是无限的。然后我希望飞溅返回超文本标记语言代码,这样我就可以用我的蜘蛛刮掉它。加载按钮没有href,所以我不能使用分页。因此,我试图写一个闪存脚本来做到这一点。但是当我用Splash运行脚本时,似乎“btn”部分在返回的超文本标记语言中没有任何作用(每次只返回第一页的超文本标记语言)。) 这是我写的Splas

  • 问题内容: 我正在尝试让jsTree与子节点的按需加载一起使用。我的代码是这样的: 调用返回的json是 每个元素可以有很多孩子,树会很大。当前,这将立即加载整个树,这可能需要一些时间。当用户打开子节点时,我该怎么做以实现按需加载? 提前致谢。 问题答案: Irishka向我指出了正确的方向,但并不能完全解决我的问题。我在弄弄她的答案,然后想到了这个。仅为了清楚起见,使用了两个不同的服务器功能。第