Model类与ImageDownLoader

本文介绍如何使用封装的ImageDownLoader类请求网络图片,详细解释了图片数据如何从文字JSON文件中提取并展示在TableView上,同时讨论了封装方法的重要性以及如何在Model中统一管理图片下载过程。

哈喽 我又来了 咱们继续上一个话题 – 用ImageDownLoader去请求网络图片

ImageDownLoader的类封装好了 咱们离成功还有…..额…..一半 = = !

不过没关系 逻辑清晰明了!


好了 让我们进入主题:

在请求下来的文字JSON 文件中 会存在一段图片网址字符串 它告诉我们 要得到图片 就要用这段字符串去网络上请求

JSON文件里的数据通常是用Model的属性去保存的 当然也会保存这段网址字符串啦 我们假设这个属性名叫image

现今我们要将Model的数据展现在TableView上 那么需要很多个对应于cell的Model 这些Model都需要通过ImageDownLoader类去获取图片data 我们可以将ImageDownLoader初始化在cell的创建方法中 然后在controller类中实现协议方法 设置个属性去接收数据 然后给cell的imageView.image赋值 不过这样做不符合封装性 我们可以将这一切都封装在model中

现在 model类中除了之前的一些属性外 还需要设置一个UIImage类的属性 用以接收data

#import <Foundation/Foundation.h>
#import <UIKit/UIKit.h>
#import "ImageDownLoader.h"
@interface ActivityModel : NSObject

// 用于接收data(data可以直接转化为Image) 
@property (nonatomic, retain) UIImage *coverImage;

// 这个属性是用于解决复用问题的 作用是记录model的请求图片状态
@property (nonatomic, assign) BOOL isDownLoading;

// model 通过此方法用 网址字符串 去请求图片data 并赋给coverImage;
- (void)downLoadImage;

@end
// model 的方法实现
- (void)downLoadImage;
{
    // 将ImageDownLoader初始化方法封装在此 
    // 当model调用downLoadImage方法时 那么就可以把model的图片网址字符串传到此处进行请求了
    // 当然 ImageDownLoaderDelegate协议的代理就设置成model了
    self.imageDownLoader = [[ImageDownLoader alloc] initWithUrl:self.image delegate:self];

    // 开始请求
    [self.imageDownLoader start];

    // 将请求状态标记为YES
    self.isDownLoading = YES;

    // 释放
    [_imageDownLoader release];


}

// ImageDownLoaderDelegate的协议方法
- (void)imageDownLoadFailedWithError:(NSError *)error
{
    NSLog(@"%@",error);
    // 请求失败了 那么也就不在请求了
    self.isDownLoading = NO;

}

// 一旦链接已经完成加载(data已经接收完全) 就触发这个方法 -- 将data通过协议方法传过来
- (void)imageDownLoadSucceedWithData:(NSData *)data
{
    // 用这个data给coverImage 赋值
    self.coverImage = [UIImage imageWithData:data];

    // 请求成功了 当然也就不在请求了
    self.isDownLoading = NO;

}
我建立了一个文件夹TPBSkeletonManager,可以装对应的文件,请先帮我构思我要完成哪些文件才能更好帮我实现我的目标,注意模组设计可以参考下面的SDNDeviceDownloadImageManager // // SDNDeviceDownloadImageManager.h // Omada // // Created by wanghao on 2025/2/12. // Copyright © 2025 TP-Link. All rights reserved. // #import <Foundation/Foundation.h> NS_ASSUME_NONNULL_BEGIN @interface SDNDeviceDownloadImageManager : NSObject + (instancetype)sharedInstance; - (TPAbstractHandle *)checkDeviceUpgrade; - (UIImage *)deviceImage:(NSString *)deviceModel andModelVersion:(NSString *)modelVersion; @end NS_ASSUME_NONNULL_END // // SDNDeviceDownloadImageManager.m // Omada // // Created by wanghao on 2025/2/12. // Copyright © 2025 TP-Link. All rights reserved. // #import "SDNDeviceDownloadImageManager.h" #import "TPDataStorage.h" #import "SDWebImageManager.h" #import "DMSDNDeviceImageInfo.h" NSString *const kDeviceDownloadFirstVersion = @"0"; static const NSInteger kSDNDeviceDownloadDefaultCacheMaxCacheAge = 365 * 24 * 60 * 60 * 50; // 50 years static const NSInteger kSDNDeviceDownloadDefaultCacheMinUpdateInterval = 3 * 24 * 60 * 60; // 3 days @interface SDNDeviceDownloadImageManager () @property (nonatomic, readwrite, strong) TPECFetchDeviceIconModule *deviceIconModule; @property (nonatomic, readwrite, strong) TPAbstractHandle *checkDeviceUpdateHandle; @property (nonatomic, readwrite, copy) NSArray<DMSDNDeviceImageInfo *> *updateDeviceImageList; @property (nonatomic, readwrite, strong) SDWebImageDownloader *imageDownloader; @property (nonatomic, readwrite, strong) SDImageCache *imageCache; @property (nonatomic, readwrite, assign) BOOL isDownloading; @end @implementation SDNDeviceDownloadImageManager + (instancetype)sharedInstance { static id instance = nil; static dispatch_once_t onceToken; dispatch_once(&onceToken, ^{ instance = [[self alloc] init]; }); return instance; } - (instancetype)init { self = [super init]; if (self) { self.imageCache = [[SDImageCache alloc]initWithNamespace:@"OmadaDeviceStore"]; self.imageCache.maxCacheAge = kSDNDeviceDownloadDefaultCacheMaxCacheAge; self.imageDownloader = [SDWebImageDownloader sharedDownloader]; self.deviceIconModule = [ALAccountContext sharedInstance].eapCloudClient.fetchDeviceIconModule; self.updateDeviceImageList = [NSArray new]; self.isDownloading = NO; } return self; } - (TPAbstractHandle *)checkDeviceUpgrade { if (self.checkDeviceUpdateHandle) { return self.checkDeviceUpdateHandle; } NSString *currentVersion = [[TSSUserDefaults standardUserDefaults] objectForKey:TPDeviceIconCurrentVersionKey]; if (IsEmptyString(currentVersion)) { currentVersion = kDeviceDownloadFirstVersion; } TPWeakSelf TPQueueHandleFactoryBlock blck1 = ^TPAbstractHandle* (TPQueueHandleBlockContext* context){ TPStrongSelfNoReturn return [_self.deviceIconModule checkNeedUpgradeWithCurrentVersion:currentVersion]; }; TPQueueHandleFactoryBlock blck2 = ^TPAbstractHandle* (TPQueueHandleBlockContext* context){ TPStrongSelfNoReturn if(context.lastResult.success) { BOOL needCheckUp = _self.deviceIconModule.needUpdate; if (needCheckUp) { return [_self.deviceIconModule getDeviceIconWithCurrentVersion:currentVersion]; } } context.stop = YES; context.stopResult = context.lastResult; return nil; }; TPGCDQueueHandle *queueHandle = [[TPGCDQueueHandle alloc] initWithQueue:dispatch_get_main_queue() factoryBlockArray:@[blck1, blck2] timeout:-1]; [queueHandle addCompletionOnMainThread:^(TPHandleResult * _Nonnull result) { TPStrongSelf _self.checkDeviceUpdateHandle = nil; if (result.success) { _self.updateDeviceImageList = _self.deviceIconModule.updateDeviceImageList; if (_self.updateDeviceImageList.count > 0 && _self.deviceIconModule.needUpdate) { [_self downLoadUpdatedDeviceImage]; } } }]; return queueHandle; } - (void)updateCacheInfo { NSTimeInterval lastTime = 0.0; NSNumber *lastShowTimestampObject = [[TSSUserDefaults standardUserDefaults] objectForKey:TPDeviceIconLastUpdateTimeKey]; if (lastShowTimestampObject) { lastTime = [lastShowTimestampObject doubleValue]; } //上次更新时间距离现在超过最小间隔,则重新拉取设备图标 if (lastTime + kSDNDeviceDownloadDefaultCacheMinUpdateInterval < [[NSDate date] timeIntervalSince1970] && !self.isDownloading) { self.isDownloading = YES; [self resetCacheInfo]; [self checkDeviceUpgrade]; } } - (void)resetCacheInfo { [[TSSUserDefaults standardUserDefaults] setObject:kDeviceDownloadFirstVersion forKey:TPDeviceIconCurrentVersionKey]; [[TSSUserDefaults standardUserDefaults] setObject:[NSArray new] forKey:TPDeviceIconMdelListKey]; } - (void)storeDeviceModelList:(NSString *)modelAndVersionKey { if (IsEmptyString(modelAndVersionKey)) { return; } NSArray<NSString *> *imageList = [self deviceIconModelList]; for (NSString *itemImage in imageList) { if ([modelAndVersionKey isEqualToString:itemImage]) { return; } } imageList = [imageList arrayByAddingObject:modelAndVersionKey]; [[TSSUserDefaults standardUserDefaults] setObject:imageList forKey:TPDeviceIconMdelListKey]; } - (void)downLoadUpdatedDeviceImage { self.isDownloading = YES; __block int handledCount = 0; for (DMSDNDeviceImageInfo *deviceImageInfo in self.updateDeviceImageList) { if ([self.imageCache diskImageExistsWithKey:deviceImageInfo.modelAndVersionKey] || deviceImageInfo.imageDownLoadURL == nil) { [self storeDeviceModelList:deviceImageInfo.modelAndVersionKey]; handledCount++; [self updateControllerVersionWithTotalCount:handledCount]; continue; } TPWeakSelf [self.imageDownloader downloadImageWithURL:deviceImageInfo.imageDownLoadURL options:0 progress:nil completed:^(UIImage *image, NSData *data, NSError *error, BOOL finished) { TPStrongSelf handledCount++; [_self.imageCache storeImage:image recalculateFromImage:NO imageData:data forKey:deviceImageInfo.modelAndVersionKey toDisk:YES]; [self storeDeviceModelList:deviceImageInfo.modelAndVersionKey]; [[TSSUserDefaults standardUserDefaults] setObject:@([[NSDate date] timeIntervalSince1970]) forKey:TPDeviceIconLastUpdateTimeKey]; [_self updateControllerVersionWithTotalCount:handledCount]; }]; } } - (void)updateControllerVersionWithTotalCount:(int)totalCount { if (totalCount == self.updateDeviceImageList.count) { NSString *curVersion = [ALAccountContext sharedInstance].eapCloudClient.fetchDeviceIconModule.controllerVersion; [[TSSUserDefaults standardUserDefaults] setObject:curVersion forKey:TPDeviceIconCurrentVersionKey]; } } - (BOOL)hasDownLoadDevice:(NSString *)modelAndVersionKey { if (IsEmptyString(modelAndVersionKey)) { return NO; } return [self.imageCache diskImageExistsWithKey:modelAndVersionKey];; } - (UIImage *)deviceImage:(NSString *)deviceModel andModelVersion:(NSString *)modelVersion { NSString *modelAndVersionKey = [DMSDNDeviceImageInfo modelAndVersionKeyFromModel:deviceModel andVersion:modelVersion]; if (IsEmptyString(modelAndVersionKey)) { return nil; } UIImage *image = [self deviceImageFromModelAndVersionKey:modelAndVersionKey]; UIImage *finalImage = image == nil ? [self vigiDeviceImage:modelAndVersionKey] : image; if (finalImage == nil) { [self updateCacheInfo]; } return finalImage; } - (UIImage *)deviceImageFromModelAndVersionKey:(NSString *)modelAndVersionKey { UIImage *image = [self.imageCache imageFromDiskCacheForKey:modelAndVersionKey]; return image; } - (NSArray<NSString *> *)deviceIconModelList { NSArray<NSString *> *deviceModelList = [[TSSUserDefaults standardUserDefaults] objectForKey:TPDeviceIconMdelListKey]; return deviceModelList == nil ? [NSArray new] : deviceModelList; } - (UIImage *)vigiDeviceImage:(NSString *)modelAndVersionKey { NSArray<NSString *> *modelList = [self deviceIconModelList]; modelList = [modelList sortedArrayUsingComparator:^NSComparisonResult(NSString * obj1, NSString * obj2) { return obj1.length < obj2.length; }]; for (NSString *modelStr in modelList) { if (!IsEmptyString(modelAndVersionKey) && [modelAndVersionKey containsString:modelStr]) { return [self deviceImageFromModelAndVersionKey:modelStr]; } } return nil; } @end
11-01
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值