iOS Bluetooth(蓝牙)

简介: 1. 蓝牙发送照片#import "ViewController.h"#import @interface ViewController ()@property (weak, nonatomic) IBOutlet UIImageView ...

1. 蓝牙发送照片

#import "ViewController.h"
#import <GameKit/GameKit.h>

@interface ViewController ()<UINavigationControllerDelegate, UIImagePickerControllerDelegate, GKPeerPickerControllerDelegate>

@property (weak, nonatomic) IBOutlet UIImageView *imageView;

/** 会话类*/
@property (nonatomic, strong) GKSession *session;

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view, typically from a nib.
}

#pragma mark 连接设备
- (IBAction)connectClick:(id)sender {
    //1. 创建GKPeerPickerController连接控制器
    GKPeerPickerController * picker = [GKPeerPickerController new];
    
    //2. 设置代理 --> 获取数据
    picker.delegate = self;
    
    //3. 显示控制器 --> show 此控制器和AlertView很像, 不是全屏的, 不用push modal
    [picker show];
}

#pragma mark GKPeerPickerController 代理方法
/**
 此方法在连接到另一台设备时, 会调用
 peerID: 另一台设备的ID
 session: 会话类, 用于接收和发送数据
 */
- (void)peerPickerController:(GKPeerPickerController *)picker didConnectPeer:(NSString *)peerID toSession:(GKSession *)session
{
    //1. 保留session
    self.session = session;
    
    //2. 设置句柄 (设置代理) --> 将来一旦受到数据, 将由句柄的方法来处理数据
    // 一旦设置了句柄, 那么还需要实现另一个方法
    [self.session setDataReceiveHandler:self withContext:nil];
    
    //3. 消失控制器
    [picker dismiss];
}

// 一旦设置了句柄, 还需要实现此方法
#pragma mark 接收到数据的时候, 会调用此方法来处理
- (void)receiveData:(NSData *)data fromPeer:(NSString *)peer inSession: (GKSession *)session context:(void *)context
{
    //1. 将Data转换成image对象
    UIImage *image = [UIImage imageWithData:data];
    
    //2. 然后设置到界面上
    self.imageView.image = image;
}


#pragma mark 发送照片方法
- (IBAction)sendPhotoClick:(id)sender {
    //1. 将image转换成Data
    NSData *data = UIImageJPEGRepresentation(self.imageView.image, 0.5);
    
    //2. 使用会话类发送数据
    /**
     GKSendDataReliable,     如果发送失败, 会重新发送, 直到成功
     GKSendDataUnreliable,   发送一次就不管了
     */
    [self.session sendDataToAllPeers:data withDataMode:GKSendDataReliable error:nil];
}

#pragma mark 选择照片
- (IBAction)selectPhontoClick:(id)sender {
    //1. 判断是否可用
    if (![UIImagePickerController isSourceTypeAvailable:UIImagePickerControllerSourceTypePhotoLibrary]) {
        return;
    }
    
    //2. 创建UIImagePickerController
    UIImagePickerController *picker = [UIImagePickerController new];
    
    //3. 设置类型
    picker.sourceType = UIImagePickerControllerSourceTypePhotoLibrary;
    
    //4. 设置代理
    picker.delegate = self;
    
    //5. 模态视图弹出
    [self presentViewController:picker animated:YES completion:nil];
}

#pragma mark UIImagePickerController 代理方法
- (void)imagePickerController:(UIImagePickerController *)picker didFinishPickingMediaWithInfo:(NSDictionary<NSString *,id> *)info
{
    //1. 获取照片并显示到界面上
    self.imageView.image = info[UIImagePickerControllerOriginalImage];
    
    //2. 关闭控制器
    [picker dismissViewControllerAnimated:YES completion:nil];
}

@end

2. CoreBluetooth

#import "ViewController.h"
#import <CoreBluetooth/CoreBluetooth.h>

@interface ViewController ()<CBCentralManagerDelegate, CBPeripheralDelegate>

/** 中央管理者*/
@property (nonatomic, strong) CBCentralManager *mgr;

/** 扫描到的外围设备的数据数组*/
@property (nonatomic, strong) NSMutableArray *peripheralArray;

@end

@implementation ViewController

- (NSMutableArray *)peripheralArray
{
    if (_peripheralArray == nil) {
        _peripheralArray = [NSMutableArray array];
    }
    return _peripheralArray;
}

- (void)viewDidLoad {
    [super viewDidLoad];
    
    //1. 建立中央管理者
    //queue: 如果传空, 就代表着在主队列
    self.mgr = [[CBCentralManager alloc] initWithDelegate:self queue:nil];
    
    //2. 扫描周边设备
    //Services: 是服务的UUID, 而且是个数据. 如果不穿, 默认扫描全部服务
    [self.mgr scanForPeripheralsWithServices:nil options:nil];
}

#pragma mark - CBCentralManager 代理方法
#pragma mark 必须调用的代理方法
- (void)centralManagerDidUpdateState:(CBCentralManager *)central
{
    
    NSLog(@"state: %zd",central.state);
}

#pragma mark 当发现外围设备时, 会调用的方法
/**
 Peripheral: 外围设备
 Data : 相关的数据
 RSSI : 信号强度
 */
- (void)centralManager:(CBCentralManager *)central didDiscoverPeripheral:(CBPeripheral *)peripheral advertisementData:(NSDictionary<NSString *,id> *)advertisementData RSSI:(NSNumber *)RSSI
{
    //3. 记录扫描到的设备
    if (![self.peripheralArray containsObject:peripheral]) {
        [self.peripheralArray addObject:peripheral];
    }
    
    // 隐藏的步骤: 你应该搞一个列表给用户选择, 让用户自己选择要连接到哪一个设备
}

#pragma mark 连接扫描到的设备 --> 此方法是咱们自己写的, 用户当选中了设备时,应该调用此方法
- (void)connectPeripheral:(CBPeripheral *)peripheral
{
    //4. 连接外围设备
    [self.mgr connectPeripheral:peripheral options:nil];
}

#pragma mark 此方法是连接到外设时会调用的代理方法
- (void)centralManager:(CBCentralManager *)central didConnectPeripheral:(CBPeripheral *)peripheral
{
#warning
    //5. 设置外围设备的代理 --> 一旦连接外设, 那么将有外设来管理服务和特征的处理
    peripheral.delegate = self;
    //6. 扫描服务 --> 可以传入UUID
    // 传空, 代表扫描所有服务
    [peripheral discoverServices:nil];
}

#pragma mark 外设的代理方法 当发现到服务的时候会调用
- (void)peripheral:(CBPeripheral *)peripheral didDiscoverServices:(NSError *)error
{
    //7. 获取指定的服务, 然后根据此服务来查找特征
    //services : 外设的所有服务, 会保存在一个services中
    for (CBService *service in peripheral.services) {
        //加入我们的服务的UUID是 "123"
        if ([service.UUID.UUIDString isEqualToString:@"123"]) {
            
            //如果UUID一致, 则开始扫描特征
            [peripheral discoverCharacteristics:nil forService:service];
        }
    }
    
    
}

#pragma mark 外设的代理方法 当发现到特征的时候会调用
- (void)peripheral:(CBPeripheral *)peripheral didDiscoverCharacteristicsForService:(CBService *)service error:(NSError *)error
{
    //8. 获取指定的特征, 然后根据此特征, 才能根据自己的需求进行数据交互处理
    //char<#(nonnull CBCharacteristic *)#>acteristics : 服务的数组中, 会包含在一个characteristics的数组
    for (CBCharacteristic *characteristic in service.characteristics) {
        //假如我们的特征的UUID是 "456"
        if ([characteristic.UUID.UUIDString isEqualToString:@"456"]) {
            
            // 如果获取到了指定的特征, 则可以进行数据交互处理
           
            //[peripheral readValueForCharacteristic:characteristic];
           // peripheral writeValue:<#(nonnull NSData *)#> forCharacteristic: type:<#(CBCharacteristicWriteType)#>
            
        }
    }
    //
}

#pragma mark 断开连接
- (void)viewWillDisappear:(BOOL)animated
{
    [super viewWillDisappear:animated];
    //9. 断开连接
    [self.mgr stopScan];
}

@end
目录
相关文章
|
物联网 API iOS开发
iOS - Bluetooth 蓝牙
1、蓝牙介绍 具体讲解见 蓝牙 技术信息 蓝牙协议栈 2、iBeacon 具体讲解见 Beacon iBeacon 是苹果公司 2013 年 9 月发布的移动设备用 OS(iOS7)上配备的新功能。
1462 0
|
9天前
|
开发框架 前端开发 Android开发
安卓与iOS开发中的跨平台策略
在移动应用开发的战场上,安卓和iOS两大阵营各据一方。随着技术的演进,跨平台开发框架成为开发者的新宠,旨在实现一次编码、多平台部署的梦想。本文将探讨跨平台开发的优势与挑战,并分享实用的开发技巧,帮助开发者在安卓和iOS的世界中游刃有余。
|
2月前
|
Java Android开发 Swift
安卓与iOS开发对比:平台选择对项目成功的影响
【10月更文挑战第4天】在移动应用开发的世界中,选择合适的平台是至关重要的。本文将深入探讨安卓和iOS两大主流平台的开发环境、用户基础、市场份额和开发成本等方面的差异,并分析这些差异如何影响项目的最终成果。通过比较这两个平台的优势与挑战,开发者可以更好地决定哪个平台更适合他们的项目需求。
117 1
|
2月前
|
设计模式 安全 Swift
探索iOS开发:打造你的第一个天气应用
【9月更文挑战第36天】在这篇文章中,我们将一起踏上iOS开发的旅程,从零开始构建一个简单的天气应用。文章将通过通俗易懂的语言,引导你理解iOS开发的基本概念,掌握Swift语言的核心语法,并逐步实现一个具有实际功能的天气应用。我们将遵循“学中做,做中学”的原则,让理论知识和实践操作紧密结合,确保学习过程既高效又有趣。无论你是编程新手还是希望拓展技能的开发者,这篇文章都将为你打开一扇通往iOS开发世界的大门。
|
2月前
|
搜索推荐 IDE API
打造个性化天气应用:iOS开发之旅
【9月更文挑战第35天】在这篇文章中,我们将一起踏上iOS开发的旅程,通过创建一个个性化的天气应用来探索Swift编程语言的魅力和iOS平台的强大功能。无论你是编程新手还是希望扩展你的技能集,这个项目都将为你提供实战经验,帮助你理解从构思到实现一个应用的全过程。让我们开始吧,构建你自己的天气应用,探索更多可能!
66 1
|
2天前
|
存储 前端开发 Swift
探索iOS开发:从新手到专家的旅程
本文将带您领略iOS开发的奇妙之旅,从基础概念的理解到高级技巧的掌握,逐步深入iOS的世界。文章不仅分享技术知识,还鼓励读者在编程之路上保持好奇心和创新精神,实现个人成长与技术突破。
|
16天前
|
安全 数据处理 Swift
深入探索iOS开发中的Swift语言特性
本文旨在为开发者提供对Swift语言在iOS平台开发的深度理解,涵盖从基础语法到高级特性的全面分析。通过具体案例和代码示例,揭示Swift如何简化编程过程、提高代码效率,并促进iOS应用的创新。文章不仅适合初学者作为入门指南,也适合有经验的开发者深化对Swift语言的认识。
37 9
|
12天前
|
设计模式 Swift iOS开发
探索iOS开发:从基础到高级,打造你的第一款App
【10月更文挑战第40天】在这个数字时代,掌握移动应用开发已成为许多技术爱好者的梦想。本文将带你走进iOS开发的世界,从最基础的概念出发,逐步深入到高级功能实现,最终指导你完成自己的第一款App。无论你是编程新手还是有志于扩展技能的开发者,这篇文章都将为你提供一条清晰的学习路径。让我们一起开始这段旅程吧!
|
16天前
|
Android开发 Swift iOS开发
探索安卓与iOS开发的差异和挑战
【10月更文挑战第37天】在移动应用开发的广阔舞台上,安卓和iOS这两大操作系统扮演着主角。它们各自拥有独特的特性、优势以及面临的开发挑战。本文将深入探讨这两个平台在开发过程中的主要差异,从编程语言到用户界面设计,再到市场分布的不同影响,旨在为开发者提供一个全面的视角,帮助他们更好地理解并应对在不同平台上进行应用开发时可能遇到的难题和机遇。
|
14天前
|
iOS开发 开发者
探索iOS开发中的SwiftUI框架
【10月更文挑战第39天】在苹果的生态系统中,SwiftUI框架以其声明式语法和易用性成为开发者的新宠。本文将深入SwiftUI的核心概念,通过实际案例展示如何利用这一框架快速构建用户界面,并探讨其对iOS应用开发流程的影响。