首先我们可以看看苹果官方对这两个方法的解释:
// return 'best' size to fit given size.
does not actually resize view. Default is return existing view size
// 意思大概是 返回“最佳”大小适合给定的大小 默认返回已经存在的视图 size
- (CGSize)sizeThatFits:(CGSize)size;
// calls sizeThatFits: with current view bounds and changes bounds size.
// 调用这个方法会改变当前 view 的 bounds.size
- (void)sizeToFit;
sizeToFit: 会计算出最优的 size 而且会改变自己的 size
sizeThatFits: 会计算出最优的 size 但是不会改变 自己的 size
UILabel *label = [[UILabel alloc] initWithFrame:CGRectMake(10, 10, 0, 0)];
[label setBackgroundColor:[UIColor grayColor]];
[label setFont:[UIFont systemFontOfSize:20]];
label.text = @"北京欢饮您!!!";
//sizeToFit:直接改变了这个label的宽和高,使它根据上面字符串的大小做合适的改变
[label sizeToFit];
NSLog(@"width=%.1f height=%.1f ", label.frame.size.width, label.frame.size.height);
[self.view addSubview:label];
UILabel *label = [[UILabel alloc] initWithFrame:CGRectMake(10, 10, 0, 0)];
[label setBackgroundColor:[UIColor grayColor]];
[label setFont:[UIFont systemFontOfSize:20]];
label.text = @"北京欢饮您!!!";
//sizeThatFits并没有改变原始label的大小
CGSize sizeThatFits = [label sizeThatFits:CGSizeZero];
NSLog(@"sizeThatFits: width=%.1f height=%.1f", sizeThatFits.width, sizeThatFits.height);
NSLog(@"width=%.1f height=%.1f", label.frame.size.width, label.frame.size.height);
[self.view addSubview:label];
sizeToFit
UIView 有个 sizeToFit 方法来计算 UIView 合适的 bounds.size, 注意 autolayout [约束]过的 view 该方法失效.
当一个 view 例如 label 设置完 text 属性后 调用[label sizeToFit]; 会根据 label 内容计算出合适的 size 来完全显示 label 内容
UILabel *label1 = [[UILabel alloc] initWithFrame:CGRectMake(0.0f, 100.0f, 0.0f, 0.0f)];
label1.backgroundColor = [UIColor yellowColor];
label1.text = @"当一个 view 例如 label 设置完 text 属性后 调用[label sizeToFit]; 会根据 label 内容计算出合适的 size 来完全显示 label 内容";
label1.numberOfLines = 0;
[label1 sizeToFit];
[self.view addSubview:label1];
上边的方法 如果不设定 label 的 frame.size.with 那么 label 会自适应宽度 而不会改变它的高度,所以 label 虽然设置了 numberOfLines = 0 ,依然无法换行 .大部分应用当文字一行显示不完全时需要给 label 换行,那么就要限定 label 的 maxWidth
还是同样的代码 由于我指定了 label 的 width = 100.0f ,在 [label sizeToFit] 会发现 label 的最大 with =100 显示不完全部分换行 高度自适应.
所以 sizeToFit 并不能够完全按我们设想的那样去自适应 label 的 bounds.size
那么 sizeToFit 内部到底做了哪些操作呢?
其实当调用 UIView 的 sizeToFit 后 会调用 sizeThatFits 方法来计算 UIView 的 bounds.size 然后改变 frame.size
UIView 的 - (CGSize)sizeThatFits:(CGSize)size {
CGSize newSize = [super sizeThatFits:size];
NSLog(@"size =%@",NSStringFromCGSize(newSize));
return newSize;
}
不妨将上边代码部分发 UILabel 改成子类化的 MyLabel 重写sizeThatFits 打印出 newSize ={99, 223.5}
其实我们也可以不使用 [ label sizeToFit] 来计算 label 内容的 size ,首先调用 sizeThatFits 方法或者一个 CGSize 然后改变 label.frame.size 就可以得到 [label sizeToFit]一样的效果
// 或者给一个限定的宽度和高度 让 label 在这个范围内进行自适应 size
CGSize labelSize = [label1 sizeThatFits:CGSizeMake([[UIScreen mainScreen] bounds].size.width, MAXFLOAT)];
CGRect rect = CGRectMake(label1.frame.origin.x, label1.frame.origin.y, labelSize.width, labelSize.height);
[label1 setFrame:rect];
sizeThatFits
由上可见 sizeThatFits 比 sizeToFit 更加灵活的设置一个自适应size 的 view , 可以定制 label 计算内部内容的规则
下边是重写子类 sizeThatFits 自定义的一个 计算label siztToFit 后改变 frame.size 的方法
- (void)layoutSubviews
{
[super layoutSubviews];
_label1.frame = CGRectMake(_margin, _margin, _cacheSize1.width, _cacheSize1.height);
_label2.frame = CGRectMake(_margin, CGRectGetMaxY(_label1.frame) + _margin, _cacheSize2.width, _cacheSize2.height);
}
- (CGSize)sizeThatFits:(CGSize)size
{
CGFloat w = size.width;
w -= 2 * _margin;
_cacheSize1 = [_label1 sizeThatFits:CGSizeMake(w, MAXFLOAT)];
_cacheSize2 = [_label2 sizeThatFits:CGSizeMake(w, MAXFLOAT)];
CGFloat h = 3 * _margin + _cacheSize1.height + _cacheSize2.height;
return CGSizeMake(size.width, h);
}
当你调用 自定义 View 的 sizeToFit 方法后 会根据自定义的 sizeThatFits:(CGSize)size 规则来计算 label 来改变 label 完全内容的 size
TestSizeFit *customerView = [[TestSizeFit alloc] init];
[customerView setLabel1Text:@"大兵布莱恩特" setLabel2Text:@"巴爷科技 (上海) 有限公司"];
[customerView sizeToFit];
[self.view addSubview:customerView];
通过上边的代码示例 可以知道 sizeToFit 方法背后怎么实现的 , sizeThatFits 可以子类化重写自己的计算规则