6、运行结果
界面:

输出:

7、小结
这种将view中的属性暴露在头文件中的方式在一定程度上破坏了封装性。因为一旦将属性暴露在头文件中,外部任何包含该view的类可能在不知情的情况下修改了属性,这不符合代码高内聚、低耦合的开发要求,因此不推荐这种编写按钮事件的方式。
代理监听按钮事件
使用代理监听按钮事件的思路是:不暴露view中的按钮,而是为按钮创建一个代理,在view头文件中声明一个代理,然后让controller成为view的代理,并实现代理方法,在view中回调controller中的回调方法,从而实现按钮事件。具体代码如下:
1、MyView.h -- 不再将按钮暴露在头文件中
在头文件中声明一个协议,协议也可以写在单独的文件中,然后通过import导入。
#import <UIKit/UIKit.h>
//自定义的按钮协议,该协议实现了<NSObject>协议,协议的名称自定,不过不要和Apple的协议重名
@protocol myBtnDelegate <NSObject>
//协议中的方法,遵循该协议的类提供其具体的实现,协议有@optional和@required两个修饰符,默认情况下是@required
- (void) BtnClick:(UIButton *)btn;
@end
//MyView的接口
@interface MyView : UIView
//声明一个属性,这个属性用于指定谁来成为本类的代理,由于不能确定什么类型的对象会成为本类的代理,因此声明为id类型
@property (weak, nonatomic) id<myBtnDelegate> delegate;
@end
2、MyView.m
按钮被封装在.m文件中,同时在.m文件中提供一个本地方法,在本地方法中调用代理的代理方法
#import "MyView.h"
@interface MyView ()
//声明在.m中的按钮对外部不可见
@property (strong, nonatomic) UIButton *myBtn;
@end
@implementation MyView
//初始化
- (id)initWithFrame:(CGRect)frame
{
self = [super initWithFrame:frame];
if (self)
{
_myBtn = [[UIButton alloc] initWithFrame:CGRectMake(140, 100, 100, 50)];
_myBtn.backgroundColor = [UIColor redColor];
//为按钮设置目标-动作,其中目标是self即包含该按钮的view自身,动作是有目标(view)提供的myBtnClick:方法
[_myBtn addTarget:self
action:@selector(myBtnClick:) forControlEvents:UIControlEventTouchUpInside];
[self addSubview:_myBtn];
}
return self;
}
//view中按钮的事件
- (void)myBtnClick:(UIButton *)btn
{
NSLog(@"Method in view");
//在回调代理方法时,首先判断自身的代理是否实现了代理方法,否则会导致崩溃
//如果自身代理实现了代理方法,在该方法中回调代理实现的具体的代理方法
if ( [self.delegate respondsToSelector:@selector(BtnClick:)] )
{
[self.delegate BtnClick: btn];
}
else
{
NSLog(@"BtnClick: haven't found in delegate.");
}
}
@end










