標籤:
xib檔案的實質是xml,描述介面對象,每個對象都有一個很重要的屬性,identity inspector面板中class屬性,載入xib檔案的時候實際上是執行個體化介面對象相對應的這些class。
xib檔案的載入過程:
1.將xib檔案從磁碟載入記憶體,有兩種技術可以載入xib檔案:NSBundle和UINib。
2.執行unarchive和initialize操作,該過程主要由NSCoding Protocol中的initWithCoder:(NSCoder *)decoder完成。
3.建立connections:Outlets和Actions。Outlets使用IBOutlet關鍵字標示,使用setValue:forKey:方法建立每個Outlet,所以每個Outlet的建立都會發送KVO通知。Actions使用IBAction關鍵字標示,替換void傳回值,通過調用addTarget:action:forControlEvents:方法建立每個Action串連。注意,這裡構建Outlets和Actions是有先後順序的,先建立Outlets串連,隨後建立Actions串連。因為,Actions的建立依賴之前建立的Outlets。
4.調用awakeFromNib方法,首先要調用super的awakeFromNib方法,之後可以設定一些個人化的操作,以及一些無法在設計時設定的操作。注意,awakeFromNib訊息只發往在Interface Builder中指定的Custom Class,不會發送給Files‘s Owner,First Responder等佔位對象。
之後,該對象的載入完成,可以進行各式各樣的操作了。
使用NSBundle載入xib檔案:
[[NSBundle mainBundle]loadNibNamed:<(NSString *)> owner:<(id)> options:<(NSDictionary *)>];
這是最常見的一種,loadNibNamed:owner:options:方法返回的是一個NSArray*,裡麵包含了所載入xib檔案包含的介面對象(class)。
NSBundle每次都從磁碟上傳入xib檔案,而UINib則只是第一次從磁碟上傳入xib檔案,之後將xib檔案快取在記憶體中,每次新產生一個對象時,直接存取記憶體中的xib檔案執行上面描述的2-4步,所以效能上會有很大的提升,並且開發人員文檔也建議對於那些重複使用的xib檔案使用UINib 載入技術,當收到低記憶體警告時,會從內從中卸載xib檔案,當再次訪問的時候會從磁碟中載入。下面看一下UINib的定義:
NS_CLASS_AVAILABLE_IOS(4_0) @interface UINib : NSObject { @private id storage;}// If the bundle parameter is nil, the main bundle is used.// Releases resources in response to memory pressure (e.g. memory warning), reloading from the bundle when necessary.+ (UINib *)nibWithNibName:(NSString *)name bundle:(NSBundle *)bundleOrNil;// If the bundle parameter is nil, the main bundle is used.+ (UINib *)nibWithData:(NSData *)data bundle:(NSBundle *)bundleOrNil;// Returns an array containing the top-level objects from the NIB.// The owner and options parameters may both be nil.// If the owner parameter is nil, connections to File‘s Owner are not permitted.// Options are identical to the options specified with -[NSBundle loadNibNamed:owner:options:]- (NSArray *)instantiateWithOwner:(id)ownerOrNil options:(NSDictionary *)optionsOrNil;@end
前面兩個方法很清楚,分別以不同的方式載入,第三個方法則是執行個體化(將對象建立出來)
表視圖執行個體:
具體的細節就不說了
建立一個名為Empty的xib檔案
注意看Table View Cell的class屬性是TableViewCell,不是預設的UITableViewCell,TableViewCell.h如下:
@interface TableViewCell : UITableViewCell@property (weak, nonatomic) IBOutlet UILabel *lb1;@property (weak, nonatomic) IBOutlet UILabel *lb2;- (IBAction)bt:(id)sender;@end
三個屬性都和xib檔案進行了連結,應該能看出來。
然後在UITableViewDataSource代理中分別進行如下操作:
//標頭檔內聲明UINib *nib;//執行個體化self->nib = [UINib nibWithNibName:@"Empty" bundle:nil];
然後在來看tableView:cellForRowAtIndexPath:方法,這個方法標準的實現方法如下:
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath { static NSString *CellIdentifier = [NSString stringWithFormat:@"Cell"]; UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier]; if (cell == nil) { cell = [[[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:CellIdentifier] autorelease]; } //config the cell return cell;}
這是使用代碼的方式建立cell ,下面看使用UINib:
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{ static NSString *identifier = @"Cell"; TableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:identifier]; if (cell == nil) { cell = [[nib instantiateWithOwner:nil options:nil] objectAtIndex:0]; } switch (indexPath.section) { case redSection: cell.lb1.text = @"lb1"; cell.lb2.text = @"lb2"; break; case blueSection: break; default: [[cell textLabel] setText:@"Unknow"]; } return cell;}
如下:
以上只是個人理解,有錯誤之處歡迎指正。
ios – 使用UINib載入xib檔案實現UITableViewCell