IBOutletCollection

移动开发 iOS
在IB与相关文件做连接时,我们经常会用到两个关键字:IBOutlet和IBAction。经常用xib或storyboard的童鞋应该用这两上关键字非常熟悉了。不过UIKit还提供了另一个伪关键字IBOutletCollection,我们使用这个关键字,可以将界面上一组相同的控件连接到同一个数组中

IBOutletCollection

在IB与相关文件做连接时,我们经常会用到两个关键字:IBOutlet和IBAction。经常用xib或storyboard的童鞋应该用这两上关键字非常熟悉了。不过UIKit还提供了另一个伪关键字IBOutletCollection,我们使用这个关键字,可以将界面上一组相同的控件连接到同一个数组中

我们先来看看这个伪关键字的定义,可以从UIKit.framework的头文件UINibDeclarations.h找到如下定义:

  1. #ifndef IBOutletCollection 
  2. #define IBOutletCollection(ClassName) 
  3. #endif 

另外,在Clang源码中,有更安全的定义方式,如下所示:

  1. #define IBOutletCollection(ClassName) __attribute__((iboutletcollection(ClassName))) 

从上面的定义可以看到,与IBOutlet不同的是,IBOutletCollection带有一个参数,该参数是一个类名。

通常情况下,我们使用一个IBOutletCollection属性时,属性必须是strong的,且类型是NSArray,如下所示:

  1. @property (strong, nonatomic) IBOutletCollection(UIScrollView) NSArray *scrollViews; 

假定我们的xib文件中有三个横向的scrollView,我们便可以将这三个scrollView都连接至scrollViews属性,然后在我们的代码中便可以做一些统一处理,如下所示:

  1. - (void)setupScrollViewImages 
  2. for (UIScrollView *scrollView in self.scrollViews) { 
  3. [self.imagesData enumerateObjectsUsingBlock:^(NSString *imageName, NSUInteger idx, BOOL *stop) { 
  4. UIImageView *imageView = [[UIImageView alloc] initWithFrame:CGRectMake(CGRectGetWidth(scrollView.frame) * idx, 0, CGRectGetWidth(scrollView.frame), CGRectGetHeight(scrollView.frame))]; 
  5. imageView.contentMode = UIViewContentModeScaleAspectFill; 
  6. imageView.image = [UIImage imageNamed:imageName]; 
  7. [scrollView addSubview:imageView]; 
  8. }]; 

这段代码会影响到三个scrollView。这样做的好处是我们不需要手动通过addObject:方法将scrollView添加到scrollViews中。

不过在使用IBOutletCollection时,需要注意两点:

IBOutletCollection集合中对象的顺序是不确定的。我们通过调试方法可以看到集合中对象的顺序跟我们连接的顺序是一样的。但是这个顺序可能会因为不同版本的Xcode而有所不同。所以我们不应该试图在代码中去假定这种顺序。

不管IBOutletCollection(ClassName)中的控件是什么,属性的类型始终是NSArray。实际上,我们可以声明是任何类型,如NSSet,NSMutableArray,甚至可以是UIColor,但不管我们在此设置的是什么类,IBOutletCollection属性总是指向一个NSArray数组。

关于第二点,我们以上面的scrollViews为例,作如下修改:

  1. @property (strong, nonatomic) IBOutletCollection(UIScrollView) NSSet *scrollViews; 

实际上我们在控制台打印这个scrollViews时,结果如下所示:

  1. (lldb) po self.scrollViews 
  2. <__NSArrayI 0x1740573d0>( 
  3. <UIScrollView: 0x12d60d770; frame = (0 0320 162); clipsToBounds = YES; autoresize = W+H; gestureRecognizers = <NSArray: 0x1740574f0>; layer = <CALayer: 0x174229480>; contentOffset: {00}; contentSize: {00}>, 
  4. <UIScrollView: 0x12d60dee0; frame = (0 0320 161); clipsToBounds = YES; autoresize = W+H; gestureRecognizers = <NSArray: 0x174057790>; layer = <CALayer: 0x1742297c0>; contentOffset: {00}; contentSize: {00}>, 
  5. <UIScrollView: 0x12d60e650; frame = (0 0320 163); clipsToBounds = YES; autoresize = W+H; gestureRecognizers = <NSArray: 0x1740579a0>; layer = <CALayer: 0x1742298e0>; contentOffset: {00}; contentSize: {00}> 

可以看到,它指向的是一个NSArray数组。

另外,IBOutletCollection实际上在iOS 4版本中就有了。不过,现在的Objective-C已经支持object literals了,所以定义数组可以直接用@[],方便了许多。而且object literals方式可以添加不在xib中的用代码定义的视图,所以显得更加灵活。当然,两种方式选择哪一种,就看我们自己的实际需要和喜好了。

参考

IBAction / IBOutlet / IBOutlet​Collection

IBOutletCollection.m

 

责任编辑:chenqingxiang 来源: 南峰子的技术博客
点赞
收藏

51CTO技术栈公众号