详解iOS中按钮点击事件处理方式

写在前面

在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文件中提供一个本地方法,在本地方法中调用代理的代理方法

#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提供的按钮代理方法。

事实上,在代理模式中,有三个角色存在:

  • 协议:一般是方法列表,规定了代理双方行为,在本例中 就是协议;
  • 代理:遵循一定的协议的类,需要实现协议中的必须方法,完成委托方的功能,本例中MyViewController就是代理;
  • 委托:拥有自己的代理,指定代理去完成功能,本例中的MyView就是委托。

代理模式用大白话说就是:委托方让代理方代替自己执行一定的动作。

总结

iOS中,类不能多继承,但协议是可以多继承的。协议并不提供具体实现。协议一般是一系列方法的集合,(也可以有属性,但这不是协议的主要使用场景),这有点像Java中的接口,继承接口的类负责提供接口中方法的具体实现。

代理模式在iOS开发中使用的地方有很多,代理模式能够实现view和controller之间的解耦。拿本文中的例子来说,controller虽然可以操作view中按钮点击后的操作,但由于按钮是作为view的私有属性声明在view的实现文件中的,因此controller并不知道view中有按钮这个属性的存在,因此无法从view外部去更改按钮的各属性,这就是view和controller之间解耦的体现。此外,由于按钮事件是在view中绑定的,而不是在controller中绑定的,因此使用该view的类只需要实现相应的代理方法就可以定制按钮点击后的事件了,这也更加方便了view的复用,体现了view与controller解耦合的优势。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • 详解iOS中Button按钮的状态和点击事件

    一.按钮的状态 1.UIControlStateNormal 1> 除开UIControlStateHighlighted.UIControlStateDisabled.UIControlStateSelected以外的其他情况,都是normal状态 2> 这种状态下的按钮[可以]接收点击事件 2.UIControlStateHighlighted 1> [当按住按钮不松开]或者[highlighted = YES]时就能达到这种状态 2> 这种状态下的按钮[可以]接收点击事件 3

  • IOS UITableViewCell详解及按钮点击事件处理实例

    IOS UITableViewCell详解及按钮点击事件处理 今天突然做项目的时候,又遇到处理自定义的UITableViewCell上按钮的点击事件问题.我知道有两种方式,可是突然想不起来之前是怎么做的了,好记性不如烂笔头,还是记录一下吧. 1.第一种方式给Button加上tag值 这里分为两种:一种是直接在原生的UITableViewCell上添加UIButton按钮,然后给UIButton设置tag值,然后在控制器里的方法里通过取数据,做界面跳转等.还是举个例子吧,省的回忆半天. - (UI

  • 详解iOS中按钮点击事件处理方式

    写在前面 在iOS开发中,时常会用到按钮,通过按钮的点击来完成界面的跳转等功能.按钮事件的实现方式有多种,其中较为常用的是目标-动作对模式.但这种方式使得view与controller之间的耦合程度较高,不推荐使用: 另一种方式是代理方式,按钮的事件在view中绑定,controller作为view的代理实现代理方法. 目标-动作对实现方式 具体来说,假设我们有一个包含一个Button的veiw,view将Button放在头文件中,以便外部访问.然后controller将view作为自己的vie

  • 详解OpenCV中简单的鼠标事件处理

    目录 cv2.setMouseCallback函数语法 回调函数 谈及鼠标事件,就是在触发鼠标按钮后程序所做出相应的反应,但是不影响程序的整个线程.这有些像异步处理.鼠标事件响应不会一直等着我们去按而后续程序不执行,这样会造成阻塞,而是在我们不按鼠标的时候程序也会正常进行,按的时候会调用鼠标的事件响应,这个过程就像程序一边正常运行一边等待鼠标响应. 为了将鼠标响应和操作画面进行绑定,我们要创建一个回调函数: cv2.setMouseCallback函数语法 cv2.setMouseCallbac

  • 详解IOS中文件路径判断是文件还是文件夹

    详解IOS中文件路径判断是文件还是文件夹 方法1 + (BOOL)isDirectory:(NSString *)filePath { BOOL isDirectory = NO; [[NSFileManager defaultManager] fileExistsAtPath:filePath isDirectory:&isDirectory]; return isDirectory; } 方法2 + (BOOL)isDirectory:(NSString *)filePath { NSNum

  • 详解IOS 单例的两种方式

    详解IOS 单例的两种方式 方法一: #pragma mark - #pragma mark sharedSingleton methods //单例函数 static RtDataModel *sharedSingletonManager = nil; + (RtDataModel *)sharedManager { @synchronized(self) { if (sharedSingletonManager == nil) { sharedSingletonManager = [[sel

  • 详解C++ 中的三种继承方式

    public 方式继承 基类成员对派生类的可见性对派生类来说,基类的公有成员和保护成员可见,基类的公有成员和保护成员作为派生类的成员时,它们都保持原有的状态;基类的私有成员不可见,基类的私有成员仍然是私有的,派生类不可访问基类中的私有成员. 基类成员对派生类对象的可见性对派生类对象来说,基类的公有成员是可见的,其他成员是不可见的. 所以,在公有继承时,派生类的对象可以访问基类中的公有成员,派生类的成员函数可以访问基类中的公有成员和保护成员. 简单来说,派生类能访问基类的public, prote

  • 详解Python中matplotlib模块的绘图方式

    目录 1.matplotlib之父简介 2.matplotlib图形结构 3.matplotlib两种画绘图方法 方法一:使用matplotlib.pyplot 方法二:面向对象方法 1.matplotlib之父简介 matplotlib之父John D. Hunter已经去世,他的一生辉煌而短暂,但是他开发的的该开源库还在继续着辉煌.国内介绍的资料太少了,查阅了一番整理如下: 1968 出身于美国的田纳西州代尔斯堡. 之后求学于普林斯顿大学. 2003年发布Matplotlib 0.1版,初衷

  • 详解ios中的SQL数据库文件加密 (使用sqlcipher)

    今天本想写一片 GAE+goAgent+SwitchySharp 的指南的!但是突然翻出了前段时间写的关于iOS中的SQL数据库文件加密的代码,于是乎决定今天就先讲讲这个!- 那么goAgent将放在周末,后续的文章中除了文件加密,还有传输数据加密,感兴趣的童鞋 敬请留意. 言归正传,sql的文件加密,我们首先要用到一个库,它就是大名鼎鼎的Sqlcipher,  奉上连接:http://sqlcipher.NET,在ios里 我们需要看的文档是这一篇http://sqlcipher.Net/io

  • 详解iOS中集成ijkplayer视频直播框架

    ijkplayer 是一款做视频直播的框架, 基于ffmpeg, 支持 Android 和 iOS, 网上也有很多集成说明, 但是个人觉得还是不够详细, 在这里详细的讲一下在 iOS 中如何集成ijkplayer, 即便以前从没有接触过, 按着下面做也可以集成成功! 一. 下载ijkplayer ijkplayer下载地址: http://xiazai.jb51.net/201612/yuanma/ijkplayer-master_jb51.rar 下载完成后解压, 解压后文件夹内部目录如下图:

  • 详解IOS中Tool Bar切换视图方法

    本文通过实例给大家详细讲解了IOS开发中Tool Bar切换视图方法以及原理解释,希望我们的整理对你有用,一起学习下. iOS中几种典型的多视图程序: (1)Tab Bar Application:程序的底部有一排按钮,轻触其中一个按钮,相应的视图被激活并显示出来: (2)Navigation-Based Application:其特点是使用navigation controller,而navigation controller使用navigation bar来控制多级视图: (3)Tool B

随机推荐