写在前面
在iOS开发中,时常会用到按钮,通过按钮的点击来完成界面的跳转等功能。按钮事件的实现方式有多种,其中较为常用的是目标-动作对模式。但这种方式使得view与controller之间的耦合程度较高,不推荐使用;
另一种方式是代理方式,按钮的事件在view中绑定,controller作为view的代理实现代理方法。
目标-动作对实现方式
具体来说,假设我们有一个包含一个Button的veiw,view将Button放在头文件中,以便外部访问。然后controller将view作为自己的view,在viewcontroller中实现按钮的点击事件。文字描述起来好像不够直观,直接上代码
1、MyView.h
包含一个可被外部访问的按钮的view
@interface MyView : UIView @property (strong, nonatomic) UIButton *myBtn; @end
2、MyView.m
#import "MyView.h" @implementation MyView //view的初始化方法 - (id)initWithFrame:(CGRect)frame { self = [super initWithFrame:frame]; if (self) { //初始化按钮 _myBtn = [[UIButton alloc] initWithFrame:CGRectMake(140, 100, 100, 50)]; _myBtn.backgroundColor = [UIColor redColor]; //将按钮添加到自身 [self addSubview:_myBtn]; } return self; } @end
3、MyViewController.h
#import <UIKit/UIKit.h> @interface MyViewController : UIViewController @end
4、MyViewController.m
添加MyView作为自身view
#import "MyViewController.h" #import "MyView.h" @interface MyViewController () @property (strong, nonatomic) MyView *myview; @end @implementation MyViewController - (void)loadView { MyView *myView = [[MyView alloc] initWithFrame: [[UIScreen mainScreen] bounds] ]; self.view = myView; self.myview = myView; //在controller中设置按钮的目标-动作,其中目标是self,也就是控制器自身,动作是用目标提供的BtnClick:方法, [self.myview.myBtn addTarget:self action:@selector(BtnClick:) forControlEvents:UIControlEventTouchUpInside]; } //MyView中的按钮的事件 - (void)BtnClick:(UIButton *)btn { NSLog(@"Method in controller."); NSLog(@"Button clicked."); }
5、 AppDelegate.m
#import "AppDelegate.h" #import "MyViewController.h" @interface AppDelegate () @end @implementation AppDelegate - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { self.window = [ [UIWindow alloc] initWithFrame: [[UIScreen mainScreen] bounds ] ]; MyViewController *myVC = [[MyViewController alloc] init]; self.window.rootViewController = myVC; self.window.backgroundColor = [UIColor whiteColor]; [self.window makeKeyAndVisible]; return YES; }
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文件中提供一个本地方法,在本地方法中html" target="_blank">调用代理的代理方法
#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
3、MyViewController.h
同上(目标-动作对实现方式)
4、MyViewController.m
#import "MyViewController.h" #import "MyView.h" //声明该controller遵循 <myBtnDelegate>协议,因此需要实现协议中的方法 @interface MyViewController () <myBtnDelegate> @end @implementation MyViewController - (void)loadView { //创建MyView类型的myView MyView *myView = [[MyView alloc] initWithFrame: [[UIScreen mainScreen] bounds] ]; //将myView的代理设置为self,即当前controller自身 myView.delegate = self; //将controller的view指向myView self.view = myView; } //该方法是代理中的方法,在controller中决定点击myBtn按钮后具体要做的事情,但controller并不能直接获取到myBtn - (void)BtnClick:(UIButton *)btn { NSLog(@"Method in controller."); NSLog(@"Button clicked."); }
5、AppDelegate
同上(目标-动作对实现方式)
6、运行结果
界面同上
日志:
7、小结
从日志可以看出,使controller成为view的代理,实现按钮的代理方法,与按钮相关的方法的执行顺序为:view中按钮的动作方法->controller提供的按钮代理方法。
事实上,在代理模式中,有三个角色存在:
代理模式用大白话说就是:委托方让代理方代替自己执行一定的动作。
总结
iOS中,类不能多继承,但协议是可以多继承的。协议并不提供具体实现。协议一般是一系列方法的集合,(也可以有属性,但这不是协议的主要使用场景),这有点像Java中的接口,继承接口的类负责提供接口中方法的具体实现。
代理模式在iOS开发中使用的地方有很多,代理模式能够实现view和controller之间的解耦。拿本文中的例子来说,controller虽然可以操作view中按钮点击后的操作,但由于按钮是作为view的私有属性声明在view的实现文件中的,因此controller并不知道view中有按钮这个属性的存在,因此无法从view外部去更改按钮的各属性,这就是view和controller之间解耦的体现。此外,由于按钮事件是在view中绑定的,而不是在controller中绑定的,因此使用该view的类只需要实现相应的代理方法就可以定制按钮点击后的事件了,这也更加方便了view的复用,体现了view与controller解耦合的优势。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持小牛知识库。
本文向大家介绍IOS UITableViewCell详解及按钮点击事件处理实例,包括了IOS UITableViewCell详解及按钮点击事件处理实例的使用技巧和注意事项,需要的朋友参考一下 IOS UITableViewCell详解及按钮点击事件处理 今天突然做项目的时候,又遇到处理自定义的UITableViewCell上按钮的点击事件问题。我知道有两种方式,可是突然想不起来之前是怎么做的了,好
本文向大家介绍java处理按钮点击事件的方法,包括了java处理按钮点击事件的方法的使用技巧和注意事项,需要的朋友参考一下 不同的事件源可以产生不同类别的事件。例如,按钮可以发送一个ActionEvent对象,而窗口可以发送WindowEvent对象。 AWT时间处理机制的概要: 1. 监听器对象是一个实现了特定监听器接口(listener interface)的类的实例。 2. 事件源是一个能够
本文向大家介绍详解iOS按钮暴力点击的便捷解决方案,包括了详解iOS按钮暴力点击的便捷解决方案的使用技巧和注意事项,需要的朋友参考一下 iOS点击事件分类 1.程序中大量按钮没有做连续响应的校验,测试人员连续点击出现了很多不必要的问题,所以只能利用运行时特性,进行hook一下。runtime不能大量使用,但是偶尔使用也是很方便的哈哈。 2.设置单个按钮不需要hook 3.多次点击按钮,只执行最后一
如果这个解决方案是错误的,如何处理这些点击和设置监听器? PS:请不要告诉我。
为了保持一致,我将列出它们: > 在活动中拥有类的成员,并将其分配给将在活动方法中处理逻辑的实例。 在'on Create'活动方法中创建'on ClickListener',并使用setOnClickListener将其分配给按钮 附言。我试着用谷歌搜索这个主题,但我找到的唯一的东西是描述“如何”做到这一点,而不是为什么它是好是坏。
本文向大家介绍易语言破解软件时的按钮断点事件处理方法,包括了易语言破解软件时的按钮断点事件处理方法的使用技巧和注意事项,需要的朋友参考一下 按钮事件断点时 断在时钟的处理方法,新人看看比较好,按钮事件断点断在时钟是很常见的事情。(主要针对易语言程序来讲解) 1.首先载入程序 2。按ALT+M,搜索FF 55 FC 5F 5E 89 5D F4(这个是易语言按钮事件特征码) 3.会弹出这个 4.然后