iOS开发中通知(Notification)快速入门及推送通知实现教程

 

通知(Notification)是开发框架中观察者模式的一种实现方式,内部的实现机制由Cocoa框架支持,通常用于试图控制器和数据模型的交互。通过通知,可以向一个或者多个对象发送消息。

继承自NSObject,实现了NSCopying Protocol协议。

通知通过通知中心(NSNotificationCenter)广播给其他对象。通知包含一个名称(name), 对象(object属性), 和一个可选字典(dictionary 属性)三个参数。这些参数协助完成通知操作。

 
通知的原理                                                  

Notification一个 对象非常简单,就是poster要提供给observer的信息包裹。Notification 对象有两个重要的成员变量:name和object。一般name用来唯一标示一个通知对象,object都是指向发送者(poster)(为了让observer在接收到notification时可以回调到poster)所以,Notification有两个方法:- (NSString *)name, - (id)object可供调用。同时,Notification对象还包含一个参数,就是字典(可选参数),这个字典中存储一些传值过程中的信息,供接收者使用。系统要求这个参数是不可变字典。

以上信息构建好后,我们就可以创建一个通知。

NSNotification *notification = nil;

notification = [NSNotification notificationWithName: aName object: aObj userInfo: aDictionary];

其中,aName就是notification对象的名称,aObj 就是发送者,aDictionary是通知的可选字典参数。

通知创建好后就可以在必要的时候发送通知,发送通知的时候,需要一个控制中心来发送通知,这个控制中心就是通知中心(NSNotificationCenter)。我们也可以通过控制中心创建并发送通知,通常我们也是这么做得。

 

通知的实际运用                                            

1. 页面关联性不大,但需要传递信息的地方。这两个页面一个作为发送者,一个作为监听者;

2. 需要向多个地方发送信息。一个页面作为发送者,多个页面作为监听者。

 
如何使用通知                                       

1.注册通知

2.发送通知

3.移除通知

 

代码实现

建立一个ViewController,修改为MRC。在.m文件里面操作。

先看注册通知

- (void)viewDidLoad {
    [super viewDidLoad];
    UIView *mainView = [[[UIView alloc]initWithFrame:[UIScreen mainScreen].bounds]autorelease];
    self.view = mainView;
    
    UIButton *redButton = [UIButton buttonWithType:UIButtonTypeRoundedRect];
    redButton.frame = CGRectMake(100, 100, 100, 30);
    [redButton setTitle:@"Red" forState:UIControlStateNormal];
    [redButton addTarget:self action:@selector(setRedColor) forControlEvents:UIControlEventTouchUpInside];
    [self.view addSubview:redButton];
    
    UIButton *blueButton = [UIButton buttonWithType:UIButtonTypeRoundedRect];
    [blueButton setTitle:@"Blue" forState:UIControlStateNormal];
    blueButton.frame = CGRectMake(100, 150, 100, 30);
    [blueButton addTarget:self action:@selector(setBlueColor) forControlEvents:UIControlEventTouchUpInside];
    [self.view addSubview:blueButton];
    /**
     *  注册通知
     */
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(doChange:) name:@"ColorChange" object:nil];
   
}




发送通知

- (void)setBlueColor{
    /**
     *  发送通知
     */
    NSDictionary *dic = [NSDictionary dictionaryWithObject:@"blueColor" forKey:@"color"];
    [[NSNotificationCenter defaultCenter] postNotificationName:@"ColorChange" object:self userInfo:dic];
}

- (void)setRedColor{
    /**
     *  发送通知
     */
    NSDictionary *dic = [NSDictionary dictionaryWithObject:@"redColor" forKey:@"color"];
    [[NSNotificationCenter defaultCenter]postNotificationName:@"ColorChange" object:self userInfo:dic];
}

- (void)doChange:(NSNotification *)notification{
    NSDictionary *dic = [notification userInfo];
    NSString *colorStr = [dic valueForKey:@"color"];
    UIColor *color = nil;
    
    if ([colorStr isEqualToString:@"blueColor"]) {
        color = [UIColor blueColor];
    }
    if ([colorStr isEqualToString:@"redColor"]) {
        color = [UIColor redColor];
    }
    self.view.backgroundColor = color;
}


移除通知

 
- (void)dealloc{
    /**
     *  移除通知
     */
    [[NSNotificationCenter defaultCenter]removeObserver:self];
    [super dealloc];
}



实现效果如下。点击红色屏幕变成红色,点击蓝色,屏幕变成蓝色。

 

 

 

iOS上简单推送通知(Push Notification)的实现

 

1、理解Apple推送通知的机制

 

从上面的流程图中,可以看到一个能接收推送通知的App,需要3个东西:

App ID(唯一标识一个App程序)

Provisioning Profile(App程序的发布需要它,所以推送通知只能在真机上测试)

Device Token(设备标识,这个是推送通知功能中特有的)

而能推送通知的服务器端则要2个东西:

SSL Certificate

Private Key

(由于我对信息加密不清楚,所以这里不解释)

值得注意的是APNS(ApplePush Notification Service) Server,完成发送Device Token和通知内容的功能,而且这2个动作都是被动的,即第一个动作是由App发起的,第二个则是推送通知的服务器发起的。

对我而言,理解就这么多了。下面我按照参考文章进行实验。

2、创建App ID

 

点击“New App ID”按钮后,如下

 

Description的内容可以任意,Bundle Identifier (App ID Suffix)必须和创建App工程时的Bundle Identifier,如下

 

点击“Submit”后,点击左侧导航中的“App IDs”,找到刚才新创建的App ID,如下

 

点击“Configure”后,如下

 

勾选“Enable for Apple Push Notification service”,然后点击红色的“Configure”按钮,这里暂时只针对Development取得证书。弹出一个对话框,如下

 

点击“Continue”后,要我们上传一个CSR文件,如下

 

下面使用钥匙串访问(KeychainAccess)应用程序创建上面需要的CSR文件(.certSigningRequest文件)

3、创建CSR文件

Keychain Access位于/Applications/Utilities目录中,打开它如下

 

然后弹出窗口如下。

 

UserEmail Address随意写就可以,Common Name也是一样,注意勾选“Save to disks”,然后点击“Continue”。很快就生成好了所需文件,去找到它。

 

回到下面的网页中,上传刚才用KeychainAccess产生的HelloRemoteNotification.certSigningRequest文件。

 

很快需要的证书就OK了,如下

 

点击“Continue”,然后点击“Done”。

 

发现上面的Status是Enabled,而且多了“Download”按钮,点击它,下载了一个名为“aps_development.cer”的文件。双击打开它,

 

找到上图中“Keys”栏中名为“HelloRemoteNotification”的private key(注意是private key,而不是public key),右击它,选择“Export “HelloRemoteNotification”…”,这样会导出一个.p12文件(需要输入密码),如下(目前共有3个文件)

 

下面开始用刚才产生的.p12文件,创建Profile provision文件

4、创建ProvisioningProfile文件

 

在上图中,点击“New Profile”按钮后,如下

 

填写“Profile Name”;勾选“Certificate”;“App ID”选择正确的、之前我们创建的ID,即PushNotification;最后关联需要测试真机设备。点击“Submit”,如下

 

可以看到多了一个Provisioning Profile文件,点击“Download”按钮下载它,这时我们一共产生4个文件,如下

 

双击“PushNotification.mobileprovision”文件,或把它拖入到Xcode中。

 

在Xcode中,找到Code Signing项,如上图,将Debug一项配置成刚才拖入Provisioning Profile对应的iPhone Developer。

5、Xcode工程中取得Device Token

 

在application:didFinishLaunchingWithOptions:方法里,注册使用远程通知。

 

添加2个方法,application: didRegisterForRemoteNotificationsWithDeviceToken:和 application:didFailToRegisterForRemoteNotificationsWithError:,用于取得Device Token和打印错误。运行我们建的HelloRemoteNotification工程,如果以上步骤都正确,应该打印出Device Token,如下

 

也可能出错如下

 

6、创建.pem文件

将已有的.cer文件转成.pem文件

将已有的.p12文件转成.pem文件(需要输入密码)

最后将上面2个.pem文件合并成1个.pem文件(需要输入新密码)

 

aps_development.cer->HelloRemoteNotification.pem(下面改名为HelloRemoteNotificationCert.pem)

HelloRemoteNotification.p12-> HelloRemoteNotificationKey.pem

 

HelloRemoteNotification.pem +HelloRemoteNotificationKey.pem合并成ck2.pem

7、编写PHP服务器代码,发送通知

<?php
// Put your device token here (without spaces):
$deviceToken = '<Xcode控制台输出的Device Token>';// Put your private key's passphrase here:
$passphrase = '<最后输入的密码>';// Put your alert message here:
$message = 'My first push notification!';////////////////////////////////////////////////////////////////////////////////
$ctx = stream_context_create();
stream_context_set_option($ctx, 'ssl', 'local_cert', 'ck2.pem');
stream_context_set_option($ctx, 'ssl', 'passphrase', $passphrase);// Open a connection to the APNS server
$fp = stream_socket_client(
 'ssl://gateway.sandbox.push.apple.com:2195', $err,
 $errstr, 60, STREAM_CLIENT_CONNECT|STREAM_CLIENT_PERSISTENT, $ctx);if (!$fp)
 exit("Failed to connect: $err $errstr" . PHP_EOL);echo 'Connected to APNS' . PHP_EOL;
// Create the payload body
$body['aps'] = array(
 'alert' => $message,
 'sound' => 'default'
 );// Encode the payload as JSON
$payload = json_encode($body);// Build the binary notification
$msg = chr(0) . pack('n', 32) . pack('H*', $deviceToken) . pack('n', strlen($payload)) . $payload;// Send it to the server
$result = fwrite($fp, $msg, strlen($msg));if (!$result)
 echo 'Message not delivered' . PHP_EOL;
else
 echo 'Message successfully delivered' . PHP_EOL;// Close the connection to the server
fclose($fp);

注意:修改下面两行代码

// Put your private key's passphrase here:
$passphrase = '<最后输入的密码>';
......
......
stream_context_set_option($ctx, 'ssl', 'local_cert', 'ck2.pem');

运行上面的php脚本,如下

 

可以看到iPad上面收到的推送通知,如下,表示实践成功!