harmony 鸿蒙连接和传输数据
连接和传输数据
简介
本指南主要提供了基于串口通信协议(Serial Port Profile,SPP)实现设备间连接和传输数据的开发指导。当两个设备间进行SPP通信交互时,依据设备功能的不同,可区分为客户端与服务端,本指南将分别介绍客户端与服务端的实现方法。
实现原理
客户端获取到服务端的设备地址后,即可向服务端特定的UUID发起连接。服务端设备地址可以通过查找设备流程获取,详见查找设备。待两端连接成功后,可向服务端发送数据或者接收服务端的数据。
服务端需要支持客户端连接的UUID服务,保持连接状态监听即可。待两端连接成功后,即可接收客户端数据或者向客户端发送数据。
客户端和服务端都可以主动断开连接,应用需要根据实际场景决定由哪一端执行断开操作。
开发步骤
申请蓝牙权限
需要申请权限ohos.permission.ACCESS_BLUETOOTH。如何配置和申请权限,请参考声明权限和向用户申请授权。
导入所需API模块
导入socket和错误码模块。
import { socket } from '@kit.ConnectivityKit';
import { BusinessError } from '@kit.BasicServicesKit';
客户端
1. 发起连接
客户端通过查找设备流程搜索到目标设备后,即可发起连接。需要连接的UUID服务,必须与服务端创建socket时构造的UUID服务一致。在连接过程中,蓝牙子系统会去查询服务端是否支持该UUID服务,若不支持,则会连接失败。因此应用需要确保目标设备是否支持需要的UUID服务,否则发起的是无效连接。
// 设备地址可以通过查找设备流程获取
let peerDevice = 'XX:XX:XX:XX:XX:XX';
// 定义客户端scoket id
let clientNumber = -1;
// 配置连接参数
let option: socket.SppOptions = {
uuid: '00009999-0000-1000-8000-00805F9B34FB', // 需要连接的服务端UUID服务,确保服务端支持
secure: false,
type: socket.SppType.SPP_RFCOMM
};
console.info('startConnect ' + peerDevice);
socket.sppConnect(peerDevice, option, (err, num: number) => {
if (err) {
console.error('startConnect errCode: ' + (err as BusinessError).code + ', errMessage: ' + (err as BusinessError).message);
} else {
console.info('startConnect clientNumber: ' + num);
clientNumber = num;
}
});
console.info('startConnect after ' + peerDevice);
2. 传输数据
2.1 发送数据
待客户端和服务端建立的连接建立成功后,即可向服务端发送数据。
let clientNumber = 1; // 注意:该值需要的是客户端发起连接时,异步callback获取到的客户端socket id,此处是伪代码id
let arrayBuffer = new ArrayBuffer(2);
let data = new Uint8Array(arrayBuffer);
data[0] = 3;
data[1] = 4;
try {
socket.sppWrite(clientNumber, arrayBuffer);
} catch (err) {
console.error('errCode: ' + (err as BusinessError).code + ', errMessage: ' + (err as BusinessError).message);
}
2.2 接收数据
待客户端和服务端建立的连接建立成功后,即可接收服务端的数据。通过订阅读取数据接口socket.on(‘sppRead’)实现。
let clientNumber = 1; // 注意:该值需要的是客户端发起连接时,异步callback获取到的客户端socket id,此处是伪代码id
// 定义接收数据的回调函数
function read(dataBuffer: ArrayBuffer) {
let data = new Uint8Array(dataBuffer);
console.info('client data: ' + JSON.stringify(data));
}
try {
// 发起订阅
socket.on('sppRead', clientNumber, read);
} catch (err) {
console.error('readData errCode: ' + (err as BusinessError).code + ', errMessage: ' + (err as BusinessError).message);
}
3. 断开连接
当应用不再需要已建立的连接时,可以通过客户端主动断开连接。需要先取消读取数据的订阅,再断开连接。
let clientNumber = 1; // 注意:该值需要的是客户端发起连接时,异步callback获取到的客户端socket id,此处是伪代码id
// 定义接收数据的回调函数
function read(dataBuffer: ArrayBuffer) {
let data = new Uint8Array(dataBuffer);
console.info('client data: ' + JSON.stringify(data));
}
try {
// 取消接收数据订阅
socket.off('sppRead', clientNumber, read);
} catch (err) {
console.error('off sppRead errCode: ' + (err as BusinessError).code + ', errMessage: ' + (err as BusinessError).message);
}
try {
// 从client端断开连接
socket.sppCloseClientSocket(clientNumber);
} catch (err) {
console.error('errCode: ' + (err as BusinessError).code + ', errMessage: ' + (err as BusinessError).message);
}
服务端
1. 创建服务端套接字
服务端需通过创建套接字的方式,在蓝牙子系统中注册指定的UUID服务。该UUID服务的名称无限制,可使用应用名称。当客户端发起连接请求时,会携带一个UUID以表示所需连接的服务。只有服务端与客户端的UUID一致时,连接才能成功建立。
// 定义服务端scoket id
let serverNumber = -1;
// 配置监听参数
let option: socket.SppOptions = {
uuid: '00009999-0000-1000-8000-00805F9B34FB',
secure: false,
type: socket.SppType.SPP_RFCOMM
};
// 创建服务端监听socket,将在蓝牙子系统中注册该UUID服务
socket.sppListen("Demo", option, (err, num: number) => {
if (err) {
console.error('sppListen errCode: ' + (err as BusinessError).code + ', errMessage: ' + (err as BusinessError).message);
} else {
console.info('sppListen serverNumber: ' + num);
serverNumber = num;
}
});
2. 监听客户端连接
创建好服务端套接字后,服务端即可监听连接。待收到客户端连接后,会获取到标识此次客户端的socket id,此时也表示服务端和客户端的连接已建立成功。
let serverNumber = 1; // 注意:该值需要的是创建服务端套接字时,异步callback获取到的服务端socket id,此处是伪代码id
// 定义客户端scoket id
let clientNumber = -1;
socket.sppAccept(serverNumber, (err, num: number) => {
if (err) {
console.error('accept errCode: ' + (err as BusinessError).code + ', errMessage: ' + (err as BusinessError).message);
} else {
console.info('accept clientNumber: ' + num);
clientNumber = num;
}
});
3. 传输数据
3.1 发送数据
待服务端和客户端的连接建立成功后,即可向客户端发送数据。
let clientNumber = 1; // 注意:该值需要的是服务端监听连接时,异步callback获取到的客户端socket id,此处是伪代码id
let arrayBuffer = new ArrayBuffer(2);
let data = new Uint8Array(arrayBuffer);
data[0] = 9;
data[1] = 8;
try {
socket.sppWrite(clientNumber, arrayBuffer);
} catch (err) {
console.error('sppWrite errCode: ' + (err as BusinessError).code + ', errMessage: ' + (err as BusinessError).message);
}
3.2 接收数据
待服务端和客户端的连接建立成功后,即可接收客户端的数据。通过订阅读取数据接口socket.on(‘sppRead’)实现。
let clientNumber = 1; // 注意:该值需要的是服务端监听连接时,异步callback获取到的客户端socket id,此处是伪代码id
// 定义接收数据的回调函数
read(dataBuffer: ArrayBuffer) {
let data = new Uint8Array(dataBuffer);
console.info('client data: ' + JSON.stringify(data));
}
try {
// 发起订阅
socket.on('sppRead', clientNumber, this.read);
} catch (err) {
console.error('readData errCode: ' + (err as BusinessError).code + ', errMessage: ' + (err as BusinessError).message);
}
4. 断开连接
当应用不再需要已建立的连接时,可以通过服务端主动断开连接。
- 需要先取消读取数据的订阅,再断开连接。 “`ts let clientNumber = 1; // 注意:该值需要的是服务端监听连接时,异步callback获取到的客户端socket id,此处是伪代码id
// 定义接收数据的回调函数 function read(dataBuffer: ArrayBuffer) { let data = new Uint8Array(dataBuffer); console.info(‘client data: ’ + JSON.stringify(data)); }
try { // 取消订阅 socket.off(‘sppRead’, clientNumber, read); } catch (err) { console.error(‘off sppRead errCode: ’ + (err as BusinessError).code + ‘, errMessage: ’ + (err as BusinessError).message); } try { // 从server断开连接 socket.sppCloseClientSocket(this.clientNumber); } catch (err) { console.error(‘errCode: ’ + (err as BusinessError).code + ‘, errMessage: ’ + (err as BusinessError).message); }
#### 5. 删除服务端套接字
当应用不再需要该服务端套接字时,需要主动关闭创建时获取到的套接字,蓝牙子系统会删除此前注册的UUID服务。如果此时客户端发起连接,就会连接失败。
- 应用也可以通过删除套接字时,实现断开连接。在此之前,需要先取消读取数据的订阅。
```ts
let serverNumber = 1; // 注意:该值需要的是创建服务端套接字时,异步callback获取到的服务端socket id,此处是伪代码id
// 定义接收数据的回调函数
function read(dataBuffer: ArrayBuffer) {
let data = new Uint8Array(dataBuffer);
console.info('client data: ' + JSON.stringify(data));
}
try {
// 取消订阅
socket.off('sppRead', clientNumber, read);
} catch (err) {
console.error('off sppRead errCode: ' + (err as BusinessError).code + ', errMessage: ' + (err as BusinessError).message);
}
try {
// 若应用不再需要此能力,则主动删除
socket.sppCloseServerSocket(serverNumber);
} catch (err) {
console.error('errCode: ' + (err as BusinessError).code + ', errMessage: ' + (err as BusinessError).message);
}
完整示例
客户端
import { socket } from '@kit.ConnectivityKit'
import { BusinessError } from '@kit.BasicServicesKit';
class SppClientManager {
// 定义客户端的socket id
clientNumber: number = -1;
// 发起连接
public startConnect(peerDevice: string): void {
// 配置连接参数
let option: socket.SppOptions = {
uuid: '00009999-0000-1000-8000-00805F9B34FB', // 需要连接的服务端UUID服务,确保服务端支持
secure: false,
type: socket.SppType.SPP_RFCOMM
};
console.info('startConnect ' + peerDevice);
socket.sppConnect(peerDevice, option, (err, num: number) => {
if (err) {
console.error('startConnect errCode: ' + (err as BusinessError).code + ', errMessage: ' + (err as BusinessError).message);
} else {
console.info('startConnect clientNumber: ' + num);
this.clientNumber = num;
}
});
console.info('startConnect after ' + peerDevice);
}
// 发送数据
public sendData() {
console.info('sendData ' + this.clientNumber);
let arrayBuffer = new ArrayBuffer(2);
let data = new Uint8Array(arrayBuffer);
data[0] = 3;
data[1] = 4;
try {
socket.sppWrite(this.clientNumber, arrayBuffer);
} catch (err) {
console.error('sppWrite errCode: ' + (err as BusinessError).code + ', errMessage: ' + (err as BusinessError).message);
}
}
// 定义接收数据的回调函数
read = (dataBuffer: ArrayBuffer) => {
let data = new Uint8Array(dataBuffer);
console.info('client data: ' + JSON.stringify(data));
};
// 接收数据
public readData() {
try {
// 发起订阅
socket.on('sppRead', this.clientNumber, this.read);
} catch (err) {
console.error('readData errCode: ' + (err as BusinessError).code + ', errMessage: ' + (err as BusinessError).message);
}
}
// 断开连接
public stopConnect() {
console.info('closeSppClient ' + this.clientNumber);
try {
// 取消接收数据订阅
socket.off('sppRead', this.clientNumber, this.read);
} catch (err) {
console.error('off sppRead errCode: ' + (err as BusinessError).code + ', errMessage: ' + (err as BusinessError).message);
}
try {
// 从client端断开连接
socket.sppCloseClientSocket(this.clientNumber);
} catch (err) {
console.error('stopConnect errCode: ' + (err as BusinessError).code + ', errMessage: ' + (err as BusinessError).message);
}
}
}
let sppClientManager = new SppClientManager();
export default sppClientManager as SppClientManager;
服务端
import { socket } from '@kit.ConnectivityKit'
import { BusinessError } from '@kit.BasicServicesKit';
class SppServerManager {
serverNumber: number = -1;
clientNumber: number = -1;
// 创建服务端监听socket
public startListen(): void {
console.info('startListen');
// 配置监听参数
let option: socket.SppOptions = {
uuid: '00009999-0000-1000-8000-00805F9B34FB',
secure: false,
type: socket.SppType.SPP_RFCOMM
};
// 创建服务端监听socket,将在蓝牙子系统中注册该UUID服务
socket.sppListen("Demo", option, (err, num: number) => {
if (err) {
console.error('sppListen errCode: ' + (err as BusinessError).code + ', errMessage: ' + (err as BusinessError).message);
} else {
console.info('sppListen serverNumber: ' + num);
this.serverNumber = num;
}
});
}
// 监听连接请求,等待连接
public accept() {
console.info('accept ' + this.serverNumber);
socket.sppAccept(this.serverNumber, (err, num: number) => {
if (err) {
console.error('accept errCode: ' + (err as BusinessError).code + ', errMessage: ' + (err as BusinessError).message);
} else {
console.info('accept clientNumber: ' + num);
this.clientNumber = num;
}
});
}
// 发送数据
public sendData() {
console.info('sendData serverNumber: ' + this.serverNumber + ' clientNumber: ' + this.clientNumber);
let arrayBuffer = new ArrayBuffer(2);
let data = new Uint8Array(arrayBuffer);
data[0] = 9;
data[1] = 8;
try {
socket.sppWrite(this.clientNumber, arrayBuffer);
} catch (err) {
console.error('sppWrite errCode: ' + (err as BusinessError).code + ', errMessage: ' + (err as BusinessError).message);
}
}
// 定义接收数据的回调函数
read = (dataBuffer: ArrayBuffer) => {
let data = new Uint8Array(dataBuffer);
console.info('client data: ' + JSON.stringify(data));
};
// 接收数据
public readData() {
try {
// 发起订阅
socket.on('sppRead', this.clientNumber, this.read);
} catch (err) {
console.error('readData errCode: ' + (err as BusinessError).code + ', errMessage: ' + (err as BusinessError).message);
}
}
// 停止连接
public stopConnect() {
console.info('stopConnect');
try {
// 取消订阅
socket.off('sppRead', this.clientNumber, this.read);
} catch (err) {
console.error('off sppRead errCode: ' + (err as BusinessError).code + ', errMessage: ' + (err as BusinessError).message);
}
try {
// 从server断开连接
socket.sppCloseClientSocket(this.clientNumber);
} catch (err) {
console.error('stopConnect errCode: ' + (err as BusinessError).code + ', errMessage: ' + (err as BusinessError).message);
}
}
// 删除能力
public closeSppServer() {
console.info('closeSppServer');
try {
// 若应用不再需要此能力,则主动删除
socket.sppCloseServerSocket(this.serverNumber);
} catch (err) {
console.error('sppCloseServerSocket errCode: ' + (err as BusinessError).code + ', errMessage: ' + (err as BusinessError).message);
}
}
}
let sppServerManager = new SppServerManager();
export default sppServerManager as SppServerManager;
你可能感兴趣的鸿蒙文章
- 所属分类: 后端技术
- 本文标签:
热门推荐
-
2、 - 优质文章
-
3、 gate.io
-
7、 golang
-
9、 openharmony
-
10、 Vue中input框自动聚焦