- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在创建一个 iOS 应用程序,目前应该在两个 iOS 设备之间传输短信。我想/必须为此使用核心蓝牙。一台设备正在成为外围设备,而另一台设备则成为中央设备。
使用包含两个特性(1 个读取、1 个写入)的一项服务设置外设是没有问题的。我使用LightBlue App连接外设,也可以看到它的特点。
当我使用中央角色设备时,我找到外围设备,可以连接并查看其特性。我将外设和特征都存储在自己的变量中,并在稍后使用它们来调用存储的外设的 writeValue(_ data: Data, forcharacteristic: CBCharacteristic, type: CBCharacteristicWriteType) ,这不会引发任何错误。但外围设备似乎没有收到该消息,因为没有调用其 didWriteValueFor 或 didUpdateValueFor 方法。中央也不能订阅这些特性。
此外,当我使用外围设备更新其自身特征的值时,我与 LightBlue 应用程序的检查不成功,因为该值始终保持为空,并且中央设备当然也不会收到通知。
看起来应用程序中的所有内容都不起作用。只有传入的数据(例如来自 LightBlue 应用程序)似乎有效。
有人遇到过类似的问题吗?我使用了许多不同的教程,并且完全按照其中所描述的进行操作(因为到处都是几乎相同的过程)。
提前致谢!
<小时/>我使用 XCode 版本 10.1,使用 Swift 进行编码,并使用 iPhone XS Max (iOS 12.1) 和 iPad Air 2 (iOS 12.0) 进行测试。
<小时/>这是我的外围设备类:
import CoreBluetooth
import UIKit
class BLEPeripheralManager: NSObject {
//MARK:- Properties
static let shared = BLEPeripheralManager()
//just some delegates for other classes
var peripheralDataReceiver: PeripheralDataReceiver?
var peripheralChatDataReceiver: PeripheralChatDataReceiver?
var peripheralManager: CBPeripheralManager
var subscribedCentrals: [CBCentral] = []
var readCharacteristic: CBMutableCharacteristic = {
let uuidStringChar1 = UUIDs.characteristicUUID1
let uuidChar1 = CBUUID(string: uuidStringChar1)
let char = CBMutableCharacteristic(type: uuidChar1, properties: .read, value: nil, permissions: .readable)
return char
}()
var writeCharacteristic: CBMutableCharacteristic = {
let uuidStringChar2 = UUIDs.characteristicUUID2
let uuidChar2 = CBUUID(string: uuidStringChar2)
let char = CBMutableCharacteristic(type: uuidChar2, properties: .write, value: nil, permissions: .writeable)
return char
}()
//MARK:- Private Methods
private override init() {
self.peripheralManager = CBPeripheralManager(delegate: nil, queue: nil)
super.init()
self.peripheralManager.delegate = self
}
private func setupManager() {
let uuidStringServ = UUIDs.serviceUUID
let uuidServ = CBUUID(string: uuidStringServ)
let transferService = CBMutableService(type: uuidServ, primary: true)
transferService.characteristics = [self.readCharacteristic, self.writeCharacteristic]
self.peripheralManager.add(transferService)
}
private func teardownServices() {
self.peripheralManager.removeAllServices()
}
private func clearSubscribers() {
self.subscribedCentrals.removeAll()
}
//MARK:- Public Methods
public func sendMessage(fromPeripheral peripheral: String, text: String) {
if text.isEmpty { return }
let chatMessage = ChatMsg(messageText: text, fromDevice: peripheral)
let encoder = JSONEncoder()
do {
let data = try encoder.encode(chatMessage)
print(self.readCharacteristic.uuid)
if self.peripheralManager.updateValue(data, for: self.readCharacteristic, onSubscribedCentrals: nil) == false {
print("Update from Peripheral failed (ReadCharacteristic)")
} else {
print("Message sent (ReadCharacteristic)")
}
if self.peripheralManager.updateValue(data, for: self.writeCharacteristic, onSubscribedCentrals: nil) == false {
print("Update from Peripheral failed (WriteCharacteristic)")
} else {
print("Message sent (WriteCharacteristic)")
}
} catch {
print("Error in encoding data")
}
}
func startAdvertising() {
let services = [CBUUID(string: UUIDs.serviceUUID)]
let advertisingDict = [CBAdvertisementDataServiceUUIDsKey: services]
self.peripheralManager.startAdvertising(advertisingDict)
}
public func stopAdvertising() {
self.peripheralManager.stopAdvertising()
}
public func checkIfAdvertising() -> Bool {
return self.peripheralManager.isAdvertising
}
}
extension BLEPeripheralManager: CBPeripheralManagerDelegate {
func peripheralManagerDidUpdateState(_ peripheral: CBPeripheralManager) {
switch peripheral.state {
case .poweredOff:
print("Peripheral powered off")
self.teardownServices()
self.clearSubscribers()
case .poweredOn:
print("Peripheral powered on")
self.setupManager()
case .resetting:
print("Peripheral resetting")
case .unauthorized:
print("Unauthorized Peripheral")
case .unknown:
print("Unknown Peripheral")
case .unsupported:
print("Unsupported Peripheral")
}
}
//doesn`t get called
func peripheralManager(_ peripheral: CBPeripheralManager, didReceiveWrite requests: [CBATTRequest]) {
for request in requests {
if let value = request.value {
if let messageText = String(data: value, encoding: String.Encoding.utf8) {
//
}
}
}
}
//doesn`t get called
func peripheralManager(_ peripheral: CBPeripheralManager, central: CBCentral, didSubscribeTo characteristic: CBCharacteristic) {
var shouldAdd: Bool = true
for sub in self.subscribedCentrals {
if sub == central {
shouldAdd = false
}
}
if shouldAdd { self.subscribedCentrals.append(central) }
}
}
这是我的中心类(class):
import CoreBluetooth
import UIKit
class BLECentralManager: NSObject {
static let shared = BLECentralManager()
//just some delegates for other classes
var centralDataReceiver: CentralDataReceiver?
var centralChatDataReceiver: CentralChatDataReceiver?
var centralManager: CBCentralManager
var peripheralArray: [CBPeripheral] = []
var writeTransferPeripheral: (peripheral: CBPeripheral, characteristic: CBCharacteristic)?
var readTransferPeripheral: (peripheral: CBPeripheral, characteristic: CBCharacteristic)?
private override init() {
self.centralManager = CBCentralManager(delegate: nil, queue: nil, options: nil)
super.init()
self.centralManager.delegate = self
}
private func startScan() {
self.centralManager.scanForPeripherals(withServices: [CBUUID(string: UUIDs.serviceUUID)], options: nil)
//self.centralManager.scanForPeripherals(withServices: nil, options: nil)
}
public func connectTo(index: Int) {
self.centralManager.connect(self.peripheralArray[index], options: nil)
}
public func sendMessage(fromPeripheral peripheral: String, text: String) {
let chatMsg = ChatMsg(messageText: text, fromDevice: peripheral)
let encoder = JSONEncoder()
do {
let data = try encoder.encode(chatMsg)
self.writeTransferPeripheral?.peripheral.writeValue(data, for: (self.writeTransferPeripheral?.characteristic)!, type: .withoutResponse)
} catch {
print("Error in encoding data")
}
}
public func getActiveConnections() -> String {
var connString: String = ""
let conns = self.centralManager.retrieveConnectedPeripherals(withServices: [CBUUID(string: UUIDs.serviceUUID)])
for peri in conns {
if connString == "" {
connString = "\(peri)"
} else {
connString = "\(connString), \(peri)"
}
}
return connString
}
public func getMessages() {
self.readTransferPeripheral?.peripheral.readValue(for: (self.readTransferPeripheral?.characteristic)!)
}
public func lookForPeripherals() {
self.peripheralArray.removeAll()
self.startScan()
}
public func getPeripherals() -> [CBPeripheral] {
return self.peripheralArray
}
private func getNameOfPeripheral(peripheral: CBPeripheral) -> String {
if let name = peripheral.name {
return name
} else {
return "Device"
}
}
}
extension BLECentralManager: CBCentralManagerDelegate, CBPeripheralDelegate {
func centralManagerDidUpdateState(_ central: CBCentralManager) {
switch central.state {
case .poweredOff:
print("BLE has powered off")
centralManager.stopScan()
case .poweredOn:
print("BLE is now powered on")
self.startScan()
case .resetting:
print("BLE is resetting")
case .unauthorized:
print("Unauthorized BLE state")
case .unknown:
print("Unknown BLE state")
case .unsupported:
print("This platform does not support BLE")
}
}
func centralManager(_ central: CBCentralManager, didDiscover peripheral: CBPeripheral, advertisementData: [String : Any], rssi RSSI: NSNumber) {
self.peripheralArray.append(peripheral)
}
func centralManager(_ central: CBCentralManager, didConnect peripheral: CBPeripheral) {
self.centralDataReceiver?.connectionEstablished(peripheral: peripheral)
print("Connection Established")
peripheral.delegate = self
peripheral.discoverServices(nil)
}
func centralManager(_ central: CBCentralManager, didDisconnectPeripheral peripheral: CBPeripheral, error: Error?) {
self.centralDataReceiver?.connectionTornDown()
}
func peripheral(_ peripheral: CBPeripheral, didDiscoverServices error: Error?) {
for service in peripheral.services! {
peripheral.discoverCharacteristics(nil, for: service)
}
}
func peripheral(_ peripheral: CBPeripheral, didDiscoverCharacteristicsFor service: CBService, error: Error?) {
for characteristic in service.characteristics! {
print(characteristic.uuid)
let char = characteristic as CBCharacteristic
if char.uuid.uuidString == UUIDs.characteristicUUID2 {
self.writeTransferPeripheral?.peripheral = peripheral
self.writeTransferPeripheral?.characteristic = char
self.writeTransferPeripheral?.peripheral.setNotifyValue(true, for: (self.writeTransferPeripheral?.characteristic)!)
} else if char.uuid.uuidString == UUIDs.characteristicUUID1 {
self.readTransferPeripheral?.peripheral = peripheral
self.readTransferPeripheral?.characteristic = char
self.readTransferPeripheral?.peripheral.setNotifyValue(true, for: (self.readTransferPeripheral?.characteristic)!)
}
}
}
//doesn`t get called
func peripheral(_ peripheral: CBPeripheral, didWriteValueFor characteristic: CBCharacteristic, error: Error?) {
if let data = characteristic.value {
if data.isEmpty { return }
if let text = String(data: characteristic.value!, encoding: String.Encoding.utf8) {
self.centralChatDataReceiver?.receiveMessage(fromPeripheral: self.getNameOfPeripheral(peripheral: peripheral), text: text)
}
}
}
//doesn`t get called
func peripheral(_ peripheral: CBPeripheral, didUpdateValueFor characteristic: CBCharacteristic, error: Error?) {
if let data = characteristic.value {
if data.isEmpty { return }
if let text = String(data: characteristic.value!, encoding: String.Encoding.utf8) {
self.centralChatDataReceiver?.receiveMessage(fromPeripheral: self.getNameOfPeripheral(peripheral: peripheral), text: text)
}
}
}
//doesn`t get called
func peripheral(_ peripheral: CBPeripheral, didUpdateNotificationStateFor characteristic: CBCharacteristic, error: Error?) {
print("Notification state changed")
}
}
最佳答案
我发现中央出了什么问题:
发现后,我将外围设备及其特征存储在两个变量/元组中(writeTransferPeripheral 和 readTransferPeripheral)。我这样做是错误的:
self.writeTransferPeripheral?.peripheral = peripheral
self.writeTransferPeripheral?.characteristic = char
这样以后使用变量写入值时变量仍然是nil。似乎您应该像这样设置元组的值:
self.writeTransferPeripheral = (peripheral, char)
关于iOS Core蓝牙无法通过特性传输数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53654864/
IO 设备如何知道属于它的内存中的值在memory mapped IO 中发生了变化? ? 例如,假设内存地址 0 专用于保存 VGA 设备的背景颜色。当我们更改 memory[0] 中的值时,VGA
我目前正在开发一个使用Facebook sdk登录(通过FBLoginView)的iOS应用。 一切正常,除了那些拥有较旧版本的facebook的人。 当他们按下“使用Facebook登录”按钮时,他
假设我有: this - is an - example - with some - dashesNSRange将使用`rangeOfString:@“-”拾取“-”的第一个实例,但是如果我只想要最后
Card.io SDK提供以下详细信息: 卡号,有效期,月份,年份,CVV和邮政编码。 如何从此SDK获取国家名称。 - (void)userDidProvideCreditCardInfo:(Car
iOS 应用程序如何从网络服务下载图片并在安装过程中将它们安装到用户的 iOS 设备上?可能吗? 最佳答案 您无法控制应用在用户设备上的安装,因此无法在安装过程中下载其他数据。 只需在安装后首次启动应
我曾经开发过一款企业版 iOS 产品,我们公司曾将其出售给大型企业,供他们的员工使用。 该应用程序通过 AppStore 提供,企业用户获得了公司特定的配置文件(包含应用程序配置文件)以启用他们有权使
我正在尝试将 Card.io SDK 集成到我的 iOS 应用程序中。我想为 CardIO ui 做一个简单的本地化,如更改取消按钮标题或“在此保留信用卡”提示文本。 我在 github 上找到了这个
我正在使用 CardIOView 和 CardIOViewDelegate 类,没有可以设置为 YES 的 BOOL 来扫描 collectCardholderName。我可以看到它在 CardIOP
我有一个集成了通话工具包的 voip 应用程序。每次我从我的 voip 应用程序调用时,都会在 native 电话应用程序中创建一个新的最近通话记录。我在 voip 应用程序中也有自定义联系人(电话应
iOS 应用程序如何知道应用程序打开时屏幕上是否已经有键盘?应用程序运行后,它可以接收键盘显示/隐藏通知。但是,如果应用程序在分屏模式下作为辅助应用程序打开,而主应用程序已经显示键盘,则辅助应用程序不
我在模拟器中收到以下错误: ImageIO: CGImageReadSessionGetCachedImageBlockData *** CGImageReadSessionGetCachedIm
如 Apple 文档所示,可以通过 EAAccessory Framework 与经过认证的配件(由 Apple 认证)进行通信。但是我有点困惑,因为一些帖子告诉我它也可以通过 CoreBluetoo
尽管现在的调试器已经很不错了,但有时找出应用程序中正在发生的事情的最好方法仍然是古老的 NSLog。当您连接到计算机时,这样做很容易; Xcode 会帮助弹出日志查看器面板,然后就可以了。当您不在办公
在我的 iOS 应用程序中,我定义了一些兴趣点。其中一些有一个 Kontakt.io 信标的名称,它绑定(bind)到一个特定的 PoI(我的意思是通常贴在信标标签上的名称)。现在我想在附近发现信标,
我正在为警报提示创建一个 trigger.io 插件。尝试从警报提示返回数据。这是我的代码: // Prompt + (void)show_prompt:(ForgeTask*)task{
您好,我是 Apple iOS 的新手。我阅读并搜索了很多关于推送通知的文章,但我没有发现任何关于 APNS 从 io4 到 ios 6 的新更新的信息。任何人都可以向我提供 APNS 如何在 ios
UITabBar 的高度似乎在 iOS 7 和 8/9/10/11 之间发生了变化。我发布这个问题是为了让其他人轻松找到答案。 那么:在 iPhone 和 iPad 上的 iOS 8/9/10/11
我想我可以针对不同的 iOS 版本使用不同的 Storyboard。 由于 UI 的差异,我将创建下一个 Storyboard: Main_iPhone.storyboard Main_iPad.st
我正在写一些东西,我将使用设备的 iTunes 库中的一部分音轨来覆盖 2 个视频的组合,例如: AVMutableComposition* mixComposition = [[AVMutableC
我创建了一个简单的 iOS 程序,可以顺利编译并在 iPad 模拟器上运行良好。当我告诉 XCode 4 使用我连接的 iPad 设备时,无法编译相同的程序。问题似乎是当我尝试使用附加的 iPad 时
我是一名优秀的程序员,十分优秀!