项目中有个类似微博那样的动态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_AssociationPolicy policy)
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];
}
这样缓存高度就搞定了。
实际效果
代码地址
https://github.com/AscenZ/YBDynamicCell
我写的一篇 解析FDTempLayoutCell的文章
在公众号