动态cell的高度并缓存

415次阅读  |  发布于4年以前

项目中有个类似微博那样的动态cell,文字和图片的多少都不是确定的

刚开始使用autolayout,结果很多问题,最后我发现了一个框架 FDTemplateLayoutCell

写的很好,自动布局cell,但是最后还是出现了很多问题,或许是不适用这种情况

最后只能用frame布局了,但是FDTemplateLayoutCell的缓存机制还是值得借鉴的

说说我的做法

首先利用frameModel计算出cell height

这个过程省略

我说说这么缓存

和FD*一样,我写了一个UITableView的category

在这个category里面又写了一个类,CellHeightCache

定义了三个方法

//是否已经缓存了
- (BOOL)existsHeightForKey:(id<NSCopying>)key {
    NSNumber *number = self.mutableCellHeightCaches[key];
    return number && ![number isEqualToNumber:@-1];
}
//缓存高度,传入key
- (void)cacheHeight:(CGFloat)height byKey:(id<NSCopying>)key {
    self.mutableCellHeightCaches[key] = @(height);
}
//传入key获得高度
- (CGFloat)heightForKey:(id<NSCopying>)key {
#if CGFLOAT_IS_DOUBLE
    return [self.mutableCellHeightCaches[key] doubleValue];
#else
    return [self.mutableCellHeightCaches[key] floatValue];
#endif
}

然后我在UITableView的分类里面写了两个方法

一个传入key获得高度,一个传入key和高度 缓存高度

- (CGFloat)getCellHeightCacheWithCacheKey:(NSString *)cacheKey
{
    if (!cacheKey) {
        return 0;
    }

    //如果已经存在cell height 则返回
    if ([self.cellHeightCache existsHeightForKey:cacheKey]) {
        CGFloat cachedHeight = [self.cellHeightCache heightForKey:cacheKey];
        return cachedHeight;
    } else {
        return 0;
    }
}

//缓存cell的高度
- (void)setCellHeightCacheWithCellHeight:(CGFloat)cellHeight CacheKey:(NSString *)cacheKey
{
    [self.cellHeightCache cacheHeight:cellHeight byKey:cacheKey];
}

他们都调用了这个方法

- (CellHeightCache *)cellHeightCache
{
    CellHeightCache *cache = objc_getAssociatedObject(self, _cmd);
    if (!cache) {
        cache = [CellHeightCache new];
        objc_setAssociatedObject(self, _cmd, cache, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
    }
    return cache;
}

这两个方法用的是OC中runtime方法,原理是两个文件关联方法,和上层的存储方法> 差不多,传入value和key对应,取出也是根据key取出value object传入self即可

1.设置关联方法

//传入object和key和value,policy
//policy即存储方式,和声明使用几种属性大致相同,有copy,retain,copy,retain_nonatomic,assign 五种)

void objc_setAssociatedObject(id object, const void *key, id value, objc_Ass

2.取出方法

//传入object和key返回value
id objc_getAssociatedObject(id object, const void *key)

这里是先获得缓存对象,如果为空的还就新创建一个,再进行关联。 这里用到了_cmd _cmd是隐藏的参数,代表当前方法的selector,他和self一样都是每个方法调用时都会传入的参数,动态运行时会提及如何传的这两个参数。

经常和关联方法搭配一起用

然后我在heightForRowAtIndexPath里面调用了

 CGFloat cellHeight = [tableView getCellHeightCacheWithCacheKey:statusFrame.identifier];
    NSLog(@"从缓存取出来的-----%f",cellHeight);

    if(!cellHeight){
        statusFrame.status = status;
        cellHeight = statusFrame.cellHeight;
        [tableView setCellHeightCacheWithCellHeight:cellHeight CacheKey:statusFrame.identifier];
    }

这样缓存高度就搞定了。

原文有gif效果

代码地址

https://github.com/AscenZ/YBDynamicCell

Copyright© 2013-2020

All Rights Reserved 京ICP备2023019179号-8