zoukankan      html  css  js  c++  java
  • iOS开发拓展篇—蓝牙之mutipeerConnectivity的使用

    蓝牙

    mutipeerConnectivity

    简介

    * iOS 7引入的一个全新框架

    * 多点连接

    * 替代GameKit框架

    * 多用于文件的传输

    * iOS设备不联网也能跟附近的人聊天

    * FireChat
    * See You Around
    * 以上近场聊天App都是基于mutipeerConnectivity框架
    

    * 搜索和传输的方式

    * 双方WIFI和蓝牙都没有打开:无法实现
    * 双方都开启蓝牙:通过蓝牙发现和传输
    * 双方都开启WIFI:通过WIFI Direct发现和传输,速度接近AirDrop
    * 双方同时开启了WIFI和蓝牙:模拟AirDrop,通过低功耗蓝牙技术扫描发现握手,然后通过WIFI Direct传输 
    

    案例界面搭建

    开发思路步骤

    1. 注册一个广播			告诉别人	我的设备是可以被发现
    2. 扫描蓝牙设备			需要实现代理方法
    3. 使用一个MCSession对象	存储当前会话	需要实现代理方法
    4. 使用MCSession对象	 发送和接收数据
    

    连接设备

    • 创建MCSession对象用于存放当前连接的会话
    // 创建MCSession对象
    // initWithPeer:设备的ID
    // 用于存放当前的连接的会话
    self.mc_Session = [ [MCSession alloc] initWithPeer:[[MCPeerID alloc] initWithDisplayName:[UIDevice currentDevice].name]];
    
    • 开启广播对象,通知正在搜索的设备他们是可用的
    #define SERVICE_TYPE @"SLZeng"
    
    /**
     *   只要开启了可被搜索的广播,那么连接蓝牙按钮就不可点
     */
    - (IBAction)foundConnect:(id)sender {
        UISwitch *switchBtn = (UISwitch *)sender;
        if (switchBtn.isOn) {
            // 广播对象,告诉其它的设备他们是可用的
            if (self.advertiserAssistant == nil) {
                // 创建广播对象
                // initWithServiceType: 广播类型的标示(因为广播可能比较多,所以最好每个广播绑定一个唯一标示)
                // session:当前会话
                // discoveryInfo: 广播信息
                self.advertiserAssistant = [[MCAdvertiserAssistant alloc] initWithServiceType:SERVICE_TYPE discoveryInfo:nil session:self.mc_Session];
            }
            // 开启广播
            [self.advertiserAssistant start];
            self.connectBT.enabled = NO;
        }else{
            self.connectBT.enabled = YES;
        }
        
    }
    
    • 开始搜索蓝牙设备
    - (IBAction)connectBlueTooth {
        // 创建搜索蓝牙设备控制器
        MCBrowserViewController *mbVC = [[MCBrowserViewController alloc] initWithServiceType:SERVICE_TYPE session:self.mc_Session];
        // 设置控制器代理
        mbVC.delegate = self;
        // 跳转到搜索控制器
        [self presentViewController:mbVC animated:YES completion:nil];
    }
    

    选择数据

    - (IBAction)selectImage {
        // 1. 创建图片选择器
        UIImagePickerController *imgPicker = [[UIImagePickerController alloc] init];
        // 2. 判断图片选择器是否可用
        if ([UIImagePickerController isSourceTypeAvailable:UIImagePickerControllerSourceTypeSavedPhotosAlbum]) {
            // 选择器打开的类型
            imgPicker.sourceType = UIImagePickerControllerSourceTypeSavedPhotosAlbum;
            // 设置代理
            imgPicker.delegate = self;
            // 打开图片选择器
            [self presentViewController:imgPicker animated:YES completion:nil];
        }
    }
    
    • 图片选择器代理中获取数据
    // 选择完毕调用
    - (void)imagePickerController:(UIImagePickerController *)picker didFinishPickingMediaWithInfo:(NSDictionary<NSString *,id> *)info
    {
    //    NSLog(@"info == %@",info);
        // 设置显示图片
        self.showImage.image = info[UIImagePickerControllerOriginalImage];
        // 隐藏图片显示器
        [picker dismissViewControllerAnimated:YES completion:nil];
    }
    
    • 在搜索控制器的连接完成的代理方法中隐藏搜索控制器
    /**
     *  连接完成
     *
     *  @param browserViewController 搜索控制器
     */
    - (void)browserViewControllerDidFinish:(MCBrowserViewController *)browserViewController
    {
        [browserViewController dismissViewControllerAnimated:YES completion:nil];
        NSLog(@"%s %d",__func__,__LINE__);
    }
    

    发送数据

    • 实现搜索控制器的代理方法,获取连接上的设备ID
    #pragma mark - MCBrowserViewControllerDelegate
    /**
     *  连接成功
     *
     *  @param browserViewController 搜索控制器
     *  @param peerID                连接上的设备ID
     *  @param info                  连接的信息
     *
     *  @return                      YES : 只发送连接上的用户
     */
    - (BOOL)browserViewController:(MCBrowserViewController *)browserViewController
          shouldPresentNearbyPeer:(MCPeerID *)peerID
                withDiscoveryInfo:(nullable NSDictionary<NSString *, NSString *> *)info
    {
        self.peerID = peerID;
        NSLog(@"info == %@   peer = %@",info, peerID);
        return YES;
    }
    
    • 将图片发送给连接上的ID
    - (IBAction)sendImage {
    
    	// 判断是否有图片
    	if (!self.showImage.image) return;
    	
        // 获取图片
        UIImage *image = self.showImage.image;
        // 将图片转换成NSData类型
        NSData *data = UIImagePNGRepresentation(image);
        /**
         *  发送数据
         *  toPeers : 发给的设备ID的数组
         *  withMode: 发送模式,是否是安全模式
         */
        if (self.peerID != nil) {
            [self.mc_Session sendData:data toPeers:@[self.peerID] withMode:MCSessionSendDataUnreliable error:nil];
        }
    }
    

    接收数据

    • 实现MCSession的代理方法接受数据
    - (void)session:(MCSession *)session didReceiveData:(NSData *)data fromPeer:(MCPeerID *)peerID
    {
        NSLog(@"%s %d",__func__, __LINE__);
        NSLog(@"data = %@, peer = %@", data, peerID);
    }
    

    显示数据

    • 在MCSeesion代理方法中设置获取过来的数据
    /**
     *  接收到数据的时候调用
     *
     *  @param session 会话
     *  @param data    数据
     *  @param peerID  设备的ID
     */
    - (void)session:(MCSession *)session didReceiveData:(NSData *)data fromPeer:(MCPeerID *)peerID
    {
        NSLog(@"%s %d",__func__,__LINE__);
    //    NSLog(@"data = %@ , peer = %@",data,peerID);
        UIImage *image = [[UIImage alloc] initWithData:data];
        if(image != nil){
            // 设置数据
            dispatch_async(dispatch_get_main_queue(), ^{
                self.showImage.image = image;
            });
        }
    }
    
    • 只要设置MCSeesion代理,必须实现的代理方法,不然报错。
    /**
     *  会话状态发生改变的时候调用
     *
     *  @param session 会话
     *  @param peerID  设备的ID
     *  @param state   状态
     */
    - (void)session:(MCSession *)session peer:(MCPeerID *)peerID didChangeState:(MCSessionState)state
    {
        NSLog(@"%s  %d",__func__,__LINE__);
    }
    
  • 相关阅读:
    56. Merge Intervals
    Reorder List
    Merge Two Sorted Lists
    彻底删除kafka topic数据
    什么时候类加载
    checkpoint的作用
    case when
    SQL:将查询结果插入到另一个表的三种情况
    IFNULL函数
    kafka主要配置
  • 原文地址:https://www.cnblogs.com/zengshuilin/p/5781085.html
Copyright © 2011-2022 走看看