一篇文章学会页面传值的10种方法(下)
这一种方法就是运用协议实现传值,这种方式也是在工程中常用的一种传值方法。首先创建好ViewController和SubViewController,在ViewController中创建好一个展示信息的Label,在SubViewController中创建好一个TextFeild,我们实现在第二个界面点击屏幕时,将TextFeild中的文字传到第一个界面,然后让背景颜色变色。
效果如下:
第一步:
首先我们要弄清楚ViewController和SubViewController谁是代理方,谁是委托方。委托方制订协议,并且拥有一个代理,代理方准守协议,并且实现协议中的方法。因为是从SubViewController中向ViewController中传递信息,SubViewController需要让代理完成传值的动作,所以这里SubViewController便是委托方
在SubViewController中我们指定一个协议BackValueProtocol:
extension ViewController:BackValueProtocol { func backValue(text: String, color: UIColor){} }
因为我们要传递文字和颜色,所以协议里的方法带有text:String 和color:UIColor 两个参数
第二步:
制定好协议之后,我们还要给SubViewController添加一个代理属性
var delegate: BackValueProtocol?
第三步:
然后就是给SubViewController指定一个代理人,当然,这里就是ViewController了
我们设置当点击屏幕时跳转到第二个界面,并且告诉第二个界面,ViewController就是他的代理方override func touchesBegan(touches: Set, withEvent event: UIEvent?) { //获取第二个界面 let svc = SubViewController() self.presentViewController(svc, animated: true, completion: nil) //MARK:告诉第二个界面ViewController就是他的代理 svc.delegate = self }
第四步:
接下来就是在ViewController中实现协议中的方法了
extension ViewController:BackValueProtocol { func backValue(text: String, color: UIColor) { //让label的文字显示为传过来的 文字 textLabel.text = text //让背景颜色改变为传过来的颜色 self.view.backgroundColor = color } } 效果如下:
第五步
最后就是执行传值操作了,我们设置在点击第二个界面时传值所以,在SubViewController的touchBegan里开始传值
override func touchesBegan(touches: Set, withEvent event: UIEvent?) { //B页面根部不需要考虑传值的目标和方法具体是谁,直接去调用自己的属性即可 //这种方式称为回调,让目标调用目标方法,参数为我传入的TextField的值 let color = UIColor.redColor() //MARK:让自己的代理去执行传值操作 self.delegate?.backValue(self.tf.text!, color: color) self.dismissViewControllerAnimated(true, completion: nil) }
到此为止,整个的反向传值就完成了,整个步骤如下:
- 想好谁是委托方谁是代理方,委托方制定协议并拥有一个代理属性
- 代理方遵守协议并实现协议中的方法
- 告诉委托方谁是代理
- 委托方告诉代理方去执行协议中的方法
7、使用TargetAction实现传值
这种传值方式和使用代理传值有相似之处,大致的思想就是在有数据的界面进行传值开始的指令,在接收数据的界面完成实质的传值操作
效果如下:
首先,同样ViewController和SubViewController,并分别设置好label和TextFeild
第一步
在SubViewController中我们设置两个属性:target和action
//设置两个属性接收传值目标和方法 //用来去接收传值的目标是任意目标 var target: AnyObject? //用来去接收传值的方法 var action: Selector?
第二步
在SubViewController中,当点击屏幕时发送传值指令
override func touchesBegan(touches: Set, withEvent event: UIEvent?) { //B页面根部不需要考虑传值的目标和方法具体是谁,直接去调用自己的属性即可 //MARK:这种方式称为回调,让目标调用目标方法,参数为我传入的TextField的值 self.target?.performSelector(self.action!, withObject: self.tf.text ) self.dismissViewController
第三步
在ViewController中我们要给SubViewController指定是指真正的target,真正的 action是什么。
同样。我们设置当点击屏幕时,将这些信息告诉给SubVIewController,显然,target是ViewController,action则是我们指定的某个实现方法
那么,我们首先要设定一个用来回传数据的方法
//定义一个用来去回传数据的方法 func backValue(string:String) { //通过传进来的参数,给label赋值 textLabel.text = string }
然后将target和action告诉SubViewController
override func touchesBegan(touches: Set, withEvent event: UIEvent?) { //获取SubViewController let svc = SubViewController() //通过正向传值,来告诉B页面回调时需要用到的参数 svc.target = self //MARK:将action告诉SubViewController //NSSelectorFromString 方法是将一个字符串方法转换成真正的方法名 svc.action = NSSelectorFromString("backValue:") self.presentViewController(svc, animated: true, completion: nil) }
到此为止,当点击ViewController屏幕时,将taget和action告诉了第二个界面,在第二个界面点击屏幕时,执行了target action命令,实现了backValue方法,完成了完整的传值操作
8、使用系统单例去传值
这个方法就厉害了我的哥!使用UIApplication这个单例类去完成数据的存储,将需要传输的数据存储到一个公共的数据源中,任何一个界面的任何地方都可以使用里面的资源
效果图如下:
在工程中,我们对AppDelegate.swift文件都不陌生,这个文件配置好了window层,并且可以设置指定的很视图控制器,在AppDelegate中我们知道它遵守了UIResponder和UIApplicationDelegate协议,而UIApplication便是一个系统提供的单例类,AppDelegate虽然不是单例类 ,但是共享的数据可以存放到这个类中
第一步
在AppDelegate声明一个用来去和共享数据的数组,然后配置一个带有的VC1,VC2和VC3的TabBarController做为根视图控制器
import UIKit@UIApplicationMainclass AppDelegate: UIResponder, UIApplicationDelegate { var window: UIWindow? //用来去共享数据的数组 var datas = [String]() func application(application: UIApplication, didFinishLaunchingWithOptions launchOptions: [NSObject: AnyObject]?) -> Bool { //给数组里加一个原始数据 datas.append("AD") //配置window window?.frame = UIScreen.mainScreen().bounds window?.backgroundColor = UIColor.whiteColor() let vc = ViewController() vc.view.backgroundColor = UIColor.redColor() let vc1 = ViewController1() vc1.view.backgroundColor = UIColor.yellowColor() let vc2 = ViewController2() vc2.view.backgroundColor = UIColor.blueColor() //设置标题 vc.title = "视图1" vc1.title = "视图2" vc2.title = "视图3" let tvc = UITabBarController() tvc.viewControllers = [vc,vc1,vc2] window?.rootViewController = tvc window?.makeKeyAndVisible() return true }
第二步
当点击屏幕时,添加数据到公共的数据源中并且读取出来
因为在VC1,VC2,VC3中的任何一个界面都可以完成相同操作,在这里只拿一个做展示- 首先我们获取UIApplication的单例类
- 通过单例类获取AppDelegate
- 通过AppDelegate操作共享的数组
代码如下:
override func touchesBegan(touches: Set, withEvent event: UIEvent?) { //获取单例 let application = UIApplication.sharedApplication() //通过这个单例获取AppDelegate这个对象 let appDelegate = application.delegate as! AppDelegate //通过这个AppDelegate来操作共享的datas数组 appDelegate.datas.append("VC-\(n + 1)") print(appDelegate.datas) }
当点击屏幕时,我们可以看到控制台中打印出来了我们设置的初始值“AD”和新加入的值“VC-2”,
同样在VC2中我们使用同样的放法获取数据
import UIKitclass ViewController1: UIViewController { var n = 1 override func viewDidLoad() { super.viewDidLoad() // Do any additional setup after loading the view. } override func didReceiveMemoryWarning() { super.didReceiveMemoryWarning() // Dispose of any resources that can be recreated. } override func touchesBegan(touches: Set, withEvent event: UIEvent?) { //UIApplication 是一个单例类 //AppDelegate并不是单例类,但是共享的数据放到了这个类里 //获取单例 let application = UIApplication.sharedApplication() //通过这个单例获取AppDelegate这个对象 let appDelegate = application.delegate as! AppDelegate //铜鼓这个APPDelegate来操作共享的datas数组 appDelegate.datas.append("VC1-\(n + 1)") print(appDelegate.datas) } }
在控制台中我们看到VC1传入的数据和VC2自己传入的数据,实现了页面之间的传值
9、使用自定义的单例实现页面之间的传值
本方法和第8中方法的实质原理是一样的,共享一个公共的数据源数组,任何一个界面都可以通过单例找到这个数据源进而进行写入和读取的操作,只不过,这里我们使用了自己的单例。
首先将根视图控制器设置为带有三个界面的标签控制器
1、自定义一个单例
在工程中我们继承于NSObject新建一个SingleInstance单例类,在单例类中我们声明一个公共的数据源,并且提供一个公开用来去获取单例的方法
//创建一个单例类class SingleInstance: NSObject { //在单例类中,有一个用来共享数据的数组 var datas = [String]() //创建一个静态或者全局变量,保存当前单例实例值 private static let singleInstance = SingleInstance() //私有化构造方法 private override init() { //给数组加一个原始数据 datas.append("SI") } //提供一个公开的用来去获取单例的方法 class func defaultSingleInstance() ->SingleInstance { //返回初始化好的静态变量值 return singleInstance } }
2、在VC1中写入并读取数据
当点击屏幕时,将数据传入公共数据源,并读出
class ViewController1: UIViewController { override func viewDidLoad() { super.viewDidLoad() } //MARK:-当点击屏幕时写入并读取数据 override func touchesBegan(touches: Set, withEvent event: UIEvent?) { //获取单例对象 let si = SingleInstance.defaultSingleInstance() //给单例中的数组加一个数据 si.datas.append("VC1") print(si.datas) } }
3、在VC2中写入并读取数据
class ViewController2: UIViewController { override func viewDidLoad() { super.viewDidLoad() // Do any additional setup after loading the view. } override func touchesBegan(touches: Set, withEvent event: UIEvent?) { //获取单例对象 let si = SingleInstance.defaultSingleInstance() //给单例中的数组加一个数据 si.datas.append("VC2") print(si.datas) } }
在VC2中同样可以通过单例找到数据源,并且实现了页面之间的传值
10、通过通知中心NSNotificationCenter传值
在这里我们实现用通知中心的方法实现从VC中向VC1和VC2中传值
通知传值类似于广播和接收信号的过程,我们在AppDelegate中共享一个公共的数据源,在传值的界面设置了一个公共的广播站,并且获取这个数据源,需要接收数据的页面就相当于一个收音机,一旦收到广播站的信号,便去执行相应的方法
1、在AppDelegate中我们共享一个数据源
class AppDelegate: UIResponder, UIApplicationDelegate { var window: UIWindow? //用来保存共享数据的数组 var datas = [Int]() func application(application: UIApplication, didFinishLaunchingWithOptions launchOptions: [NSObject: AnyObject]?) -> Bool { window = UIWindow() window?.frame = UIScreen.mainScreen().bounds window?.backgroundColor = UIColor.whiteColor() let vc = ViewController() vc.title = "VC" let vc1 = ViewController1() vc1.title = "VC1" let vc2 = ViewController2() vc2.title = "VC2" let tvc = UITabBarController() tvc.viewControllers = [vc,vc1,vc2] window?.rootViewController = tvc window?.makeKeyAndVisible() return true }
2、在VC1和VC2中扩展一个成为观察者的方法,并设置接收到数据后需要执行的动作
扩展为观察者的方法
extension ViewController1{ //成为观察者的方法 func becomeObserver(){ //先去获取通知中心的单例 let notificationCenter = NSNotificationCenter.defaultCenter() //向通知中心去加入观察者 notificationCenter.addObserver(self, selector: #selector(ViewController1.sum), name: "又有新数据啦!", object: nil) } }
接收到信号后执行的动作
//观察者的响应方法 func sum(){ let datas = ((UIApplication.sharedApplication().delegate)as!AppDelegate).datas var sum = 0 for i in datas{ sum += i } print("VC1计算求和得:\(sum)") }
完整代码如下:
class ViewController1: UIViewController { override func viewDidLoad() { super.viewDidLoad() } } extension ViewController1{ //成为观察者的方法 func becomeObserver(){ //先去获取通知中心的单例 let notificationCenter = NSNotificationCenter.defaultCenter() //向通知中心去加入观察者 notificationCenter.addObserver(self, selector: #selector(ViewController1.sum), name: "又有新数据啦!", object: nil) } //观察者的响应方法 func sum(){ let datas = ((UIApplication.sharedApplication().delegate)as!AppDelegate).datas var sum = 0 for i in datas{ sum += i } print("VC1计算求和得:\(sum)") } }
3、回到AppDelegate中将VC1和VC2注册成为观察者
//让vc1,注册成为观察者 vc1.becomeObserver() //让vc2,注册成为观察者 vc2.becomeObserver()
4、当点击VC屏幕时写入数据并发送通知
override func touchesBegan(touches: Set, withEvent event: UIEvent?) { //在这里,向数据源中去增加数据,并发送通知 //随机生成一个数字 let num = arc4random_uniform(100) //通过UIApplication获取单例类并获取数据 ((UIApplication.sharedApplication().delegate) as! AppDelegate).datas.append(Int(num)) //MARK-获取通知中心 let nc = NSNotificationCenter.defaultCenter() //发送通知 nc.postNotificationName("又有新数据啦!", object: nil) }
在控制台中我们就可以看见当点击VC的时候,VC1和VC2都执行了响应的方法
效果如下:
到此,页面传值的10种方法全部介绍完了,文中有描述不当之处希望大家包涵和指正,有需要源代码的伙伴们可以下载!