写在前面

在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解耦合的优势。

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

详解iOS中按钮点击事件处理方式的更多相关文章

  1. HTML5 input新增type属性color颜色拾取器的实例代码

    type 属性规定 input 元素的类型。本文较详细的给大家介绍了HTML5 input新增type属性color颜色拾取器的实例代码,感兴趣的朋友跟随脚本之家小编一起看看吧

  2. amazeui模态框弹出后立马消失并刷新页面

    这篇文章主要介绍了amazeui模态框弹出后立马消失并刷新页面,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

  3. 移动HTML5前端框架—MUI的使用

    这篇文章主要介绍了移动HTML5前端框架—MUI的使用的相关资料,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧

  4. HTML5在微信内置浏览器下右上角菜单的调整字体导致页面显示错乱的问题

    HTML5在微信内置浏览器下,在右上角菜单的调整字体导致页面显示错乱的问题,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧

  5. AmazeUI 模态窗口的实现代码

    这篇文章主要介绍了AmazeUI 模态窗口的实现代码,代码简单易懂,非常不错,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

  6. iOS实现拖拽View跟随手指浮动效果

    这篇文章主要为大家详细介绍了iOS实现拖拽View跟随手指浮动,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  7. ios – containerURLForSecurityApplicationGroupIdentifier:在iPhone和Watch模拟器上给出不同的结果

    我使用默认的XCode模板创建了一个WatchKit应用程序.我向iOSTarget,WatchkitAppTarget和WatchkitAppExtensionTarget添加了应用程序组权利.(这是应用程序组名称:group.com.lombax.fiveminutes)然后,我尝试使用iOSApp和WatchKitExtension访问共享文件夹URL:延期:iOS应用:但是,测试NSURL

  8. ios – Testflight无法安装应用程序

    我有几个测试人员注册了testflight并连接了他们的设备……他们有不同的ios型号……但是所有这些都有同样的问题.当他们从“safari”或“testflight”应用程序本身单击应用程序的安装按钮时……达到约90%并出现错误消息…

  9. ibm-mobilefirst – 在iOS 7.1上获取“无法安装应用程序,因为证书无效”错误

    当我的客户端将他们的设备更新到iOS7.1,然后尝试从AppCenter更新我们的应用程序时,我收到了上述错误.经过一番搜索,我找到了一个类似问题的帖子here.但是后来因为我在客户端使用AppCenter更新应用程序的环境中,我无法使用USB插件并为他们安装应用程序.在发布支持之前,是否有通过AppCenter进行下载的解决方法?

  10. ios – 视图的简单拖放?

    我正在学习iOS,但我找不到如何向UIView添加拖放行为.我试过了:它说“UIView没有可见的接口声明选择器addTarget”此外,我尝试添加平移手势识别器,但不确定这是否是我需要的它被称为,但不知道如何获得事件的坐标.在iOS中注册移动事件回调/拖放操作的标准简单方法是什么?

随机推荐

  1. iOS实现拖拽View跟随手指浮动效果

    这篇文章主要为大家详细介绍了iOS实现拖拽View跟随手指浮动,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  2. iOS – genstrings:无法连接到输出目录en.lproj

    使用我桌面上的项目文件夹,我启动终端输入:cd然后将我的项目文件夹拖到终端,它给了我路径.然后我将这行代码粘贴到终端中找.-name*.m|xargsgenstrings-oen.lproj我在终端中收到此错误消息:genstrings:无法连接到输出目录en.lproj它多次打印这行,然后说我的项目是一个目录的路径?没有.strings文件.对我做错了什么的想法?

  3. iOS 7 UIButtonBarItem图像没有色调

    如何确保按钮图标采用全局色调?解决方法只是想将其转换为根注释,以便为“回答”复选标记提供更好的上下文,并提供更好的格式.我能想出这个!

  4. ios – 在自定义相机层的AVFoundation中自动对焦和自动曝光

    为AVFoundation定制图层相机创建精确的自动对焦和曝光的最佳方法是什么?

  5. ios – Xcode找不到Alamofire,错误:没有这样的模块’Alamofire’

    我正在尝试按照github(https://github.com/Alamofire/Alamofire#cocoapods)指令将Alamofire包含在我的Swift项目中.我创建了一个新项目,导航到项目目录并运行此命令sudogeminstallcocoapods.然后我面临以下错误:搜索后我设法通过运行此命令安装cocoapodssudogeminstall-n/usr/local/bin

  6. ios – 在没有iPhone6s或更新的情况下测试ARKit

    我在决定下载Xcode9之前.我想玩新的框架–ARKit.我知道要用ARKit运行app我需要一个带有A9芯片或更新版本的设备.不幸的是我有一个较旧的.我的问题是已经下载了新Xcode的人.在我的情况下有可能运行ARKit应用程序吗?那个或其他任何模拟器?任何想法或我将不得不购买新设备?解决方法任何iOS11设备都可以使用ARKit,但是具有高质量AR体验的全球跟踪功能需要使用A9或更高版本处理器的设备.使用iOS11测试版更新您的设备是必要的.

  7. 将iOS应用移植到Android

    我们制作了一个具有2000个目标c类的退出大型iOS应用程序.我想知道有一个最佳实践指南将其移植到Android?此外,由于我们的应用程序大量使用UINavigation和UIView控制器,我想知道在Android上有类似的模型和实现.谢谢到目前为止,guenter解决方法老实说,我认为你正在计划的只是制作难以维护的糟糕代码.我意识到这听起来像很多工作,但从长远来看它会更容易,我只是将应用程序的概念“移植”到android并从头开始编写.

  8. ios – 在Swift中覆盖Objective C类方法

    我是Swift的初学者,我正在尝试在Swift项目中使用JSONModel.我想从JSONModel覆盖方法keyMapper,但我没有找到如何覆盖模型类中的Objective-C类方法.该方法的签名是:我怎样才能做到这一点?解决方法您可以像覆盖实例方法一样执行此操作,但使用class关键字除外:

  9. ios – 在WKWebView中获取链接URL

    我想在WKWebView中获取tapped链接的url.链接采用自定义格式,可触发应用中的某些操作.例如HTTP://我的网站/帮助#深层链接对讲.我这样使用KVO:这在第一次点击链接时效果很好.但是,如果我连续两次点击相同的链接,它将不报告链接点击.是否有解决方法来解决这个问题,以便我可以检测每个点击并获取链接?任何关于这个的指针都会很棒!解决方法像这样更改addobserver在observeValue函数中,您可以获得两个值

  10. ios – 在Swift的UIView中找到UILabel

    我正在尝试在我的UIViewControllers的超级视图中找到我的UILabels.这是我的代码:这是在Objective-C中推荐的方式,但是在Swift中我只得到UIViews和CALayer.我肯定在提供给这个方法的视图中有UILabel.我错过了什么?我的UIViewController中的调用:解决方法使用函数式编程概念可以更轻松地实现这一目标.

返回
顶部