基于XG24-EK2703A的BLE HID蓝牙键盘+鼠标复合设备功能开发(BLE+HID+FreeRTOS+Gecko SDK)

目录

  • 项目介绍
  • 硬件介绍
  • 项目设计
    • 开发环境及工程参考
    • 总体流程图
    • 硬件基本配置
    • 应用初始化
    • 按键中断回调
    • 定时器回调
    • 按键响应任务
    • 蓝牙事件回调
    • BLE HID
      • Report Map及报文
        • 键盘设备
        • 鼠标设备
        • 复合设备
      • 发送字符串
      • 上/下滚动
  • 功能展示
  • 项目总结

👉 【Funpack3-1】基于XG24-EK2703A的BLE HID蓝牙键盘+鼠标复合设备
👉 Github: EmbeddedCamerata/XG24_ble_hid_keymouse

项目介绍

本项目基于Silicon Labs XG24-EK2703A开发板,通过HID协议实现了一个蓝牙键盘+鼠标复合设备,可通过按键实现上下翻页、发送字符功能。使用板载两个按键,当BTN0按下,向上翻页;当BTN1按下,向下翻页;当两按键同时按下2s后,向主机依次发送字符“EETREE.CN”。

👉 Simplicity Studio 5

硬件介绍

XG24-EK2703A是一款基于EFR32MG24片上系统的开发套件,具备超低成本、低功耗和小巧的特点。该套件支持2.4GHz无线通信,兼容蓝牙LE、蓝牙mesh、Zigbee、Thread和Matter协议,为无线物联网产品的开发和原型制作提供了极大的便利。包含:

  1. 一个USB接口
  2. 一个板载SEGGER J-Link 调试器,支持SWD
  3. 两个LED和两个按钮
  4. 虚拟COM端口
  5. 数据包跟踪接口(PTI)
  6. 一个支持外部硬件连接的mikroBus插座和一个Qwiic连接器
  7. 32 位 ARM Cortex-M33,78 MHz最高工作频率
  8. 1536 kB 闪存和 256 kB RAM

XG24板卡资源图

项目设计

开发环境及工程参考

本项目使用Silicon Labs官方的IDE Simplicity Studio 5开发,使用Gecko SDK v4.4.0,GNU ARM Toolchain 12.2。工程目录上,按照Bluetooth - SoC Empty 空白示例的代码组织形式即可。主要的业务代码写在 app.capp.h 内,外设、驱动及蓝牙部分通过 .slcp 文件配置。

👉 本工程参考SiliconLabs蓝牙应用示例:bluetooth_hid_keyboard

总体流程图

所使用的系统外设:两个按键、两个LED及蓝牙栈。

  • 在按键中断回调中,根据不同按键按下,置位或清除各按键按下的事件
  • 使用FreeRTOS操作系统,创建按键响应任务,用以实现两个按键按下的响应服务:循环读取按键按下事件,当按键单独按下时,则用一枚举变量 km_status 记录:
    • 当BTN0按下,置 KM_SCROLL_UP
    • 当BTN1按下,置 KM_SCROLL_DOWN
    • 当同时按下,且无定时器在运行,则开启2s定时器,该定时器绑定一回调函数,在该回调内:置 km_statusKM_SEND_STRING,同时反转两LED状态(便于观察现象)
    • 最后,都向蓝牙栈发送外部事件信号
  • 在蓝牙事件回调中,当接收到外部事件信号后,根据 km_status 值进行相应操作。从而实现上/下翻页、发送字符的功能。

系统工作流程图

硬件基本配置

在基于 “Bluetooth - Soc Empty” 空白示例的基础上,打开 .slcp 文件,在 SOFTWARE COMPONENTS 选项卡下安装如下组件:

  • [Platform] → [Driver] → [Button] → [Simple Button],例化 btn0 与 btn1,对应开发板上两个按键,均设置为中断模式
  • [Platform] → [Driver] → [LED] → [Simple LED],例化 led0 与 led1,对应开发板上两个 LED
  • [Services] → [IO Stream] → [IO Stream: USART],保持默认配置即可
  • [Application] → [Utility] → [Timer for FreeRTOS]
  • [Application] → [Utility] → [Log]

并且,参考SiliconLabs蓝牙应用示例:bluetooth_hid_keyboard,使用该示例提供的 GATT 配置,导入到自己的工程中:

  1. 打开项目中 .slcp 文件
  2. 在 CONFIGURATION TOOLS 选项卡下找到 Bluetooth GATT Configurator
  3. 导入 config/btconf/gatt_configuration.btconf 文件
  4. 保存 GATT 配置

后续还会进行一定程度的修改。

应用初始化

app.h 内,定义四种按键按下的枚举类型,分别表示:未按下、发送字符(两按键同时按下)、上翻页(BTN0按下)及下翻页(BTN1按下):

typedef enum
{KM_IDLE = 0U,KM_SEND_STRING = 1U,KM_SCROLL_UP = 2U,KM_SCROLL_DOWN = 3U,
} km_status_t;

在初始化阶段,先创建按键按下事件组、按键响应任务。

#define KM_BTN_TASK_NAME        "keymouse_btn"
#define KM_BTN_TASK_STACK_SIZE  1024
#define KM_BTN_TASK_STATIC      0TaskHandle_t km_btn_task_handle = NULL;
static EventGroupHandle_t xbtn_events = NULL;
static km_status_t km_status = KM_IDLE;SL_WEAK void app_init(void)
{xbtn_events = xEventGroupCreate();if (xbtn_events == NULL) {app_log_error("BTN events create failed\r\n");}xTaskCreate(km_btn_task,KM_BTN_TASK_NAME,configMINIMAL_STACK_SIZE,NULL,tskIDLE_PRIORITY,&km_btn_task_handle);
}

按键中断回调

按键中断回调定义在 void sl_button_on_change(const sl_button_t *handle) 内,可参考示例修改。在此,根据触发中断的句柄判断是哪个按键按下或释放,相应地置位或清除事件位 xbtn_events

#include "sl_simple_button_instances.h"
#define BTN0_PRESSED            (1 << 0)
#define BTN1_PRESSED            (1 << 1)
#define BTN_NONE_PRESSED        0
#define BTN_BOTH_PRESSED        (BTN0_PRESSED | BTN1_PRESSED)
void sl_button_on_change(const sl_button_t *handle)
{BaseType_t xHigherPriorityTaskWoken;if (&sl_button_btn0 == handle) {if (sl_button_get_state(handle) == SL_SIMPLE_BUTTON_PRESSED) {xEventGroupSetBitsFromISR(xbtn_events, BTN0_PRESSED, &xHigherPriorityTaskWoken);}else {xEventGroupClearBitsFromISR(xbtn_events, BTN0_PRESSED);}}if (&sl_button_btn1 == handle) {if (sl_button_get_state(handle) == SL_SIMPLE_BUTTON_PRESSED) {xEventGroupSetBitsFromISR(xbtn_events, BTN1_PRESSED, &xHigherPriorityTaskWoken);}else {xEventGroupClearBitsFromISR(xbtn_events, BTN1_PRESSED);}}
}

定时器回调

该回调函数被捆绑在2s不自动重载定时器上,由于定时器是在两按键同时按下并持续2s后才结束,因此在回调内,需清除两个按键按下事件,最后发送给蓝牙栈外部事件信号。

static void btn_press_timer_cb(app_timer_t *timer, void *data)
{(void)data;(void)timer;BaseType_t xResult;xResult = xEventGroupClearBitsFromISR(xbtn_events, BTN_BOTH_PRESSED);if (xResult == pdFAIL) {app_log_error("Clear BTN_BOTH_PRESSED event failed\r\n");}km_status = KM_SEND_STRING;sl_led_toggle(&sl_led_led0);sl_led_toggle(&sl_led_led1);sl_bt_external_signal(1);
}

按键响应任务

主体为一循环。在循环内,通过 xEventGroupGetBits 读取按键事件,并做出不同响应。该事件在应用初始化时创建。用一bool型变量 is_running 记录定时器是否在运行,从而避免在两按键一直按下时反复重启定时器。由于可能出现先两按键按下,再释放一个或两个按键的情况,因此在其他情况下,都关闭定时器。

static void km_btn_task(void *p_arg)
{app_timer_t btn_press_timer;bool is_running = false;EventBits_t btn_events;while (1) {btn_events = xEventGroupGetBits(xbtn_events);switch (btn_events) {case (BTN_BOTH_PRESSED):if (!is_running) {app_timer_start(&btn_press_timer, 2000, btn_press_timer_cb, NULL, false);is_running = true;}case (BTN0_PRESSED):app_timer_stop(&btn_press_timer);km_status = KM_SCROLL_UP; // scroll upsl_bt_external_signal(1);break;case (BTN1_PRESSED):app_timer_stop(&btn_press_timer);km_status = KM_SCROLL_DOWN; // scroll downsl_bt_external_signal(1);break;default:app_timer_stop(&btn_press_timer);is_running = false;break;}vTaskDelay(pdMS_TO_TICKS(50));}vTaskDelete(NULL);
}

蓝牙事件回调

参考SiliconLabs蓝牙应用示例: bluetooth_hid_keyboard,修改蓝牙事件回调中当 MSG_ID 为 sl_bt_evt_system_external_signal_id 时的部分代码:根据 km_status 状态分别实现上/下翻页、发送字符,且这几个功能分别用函数封装。最后,置 km_status = KM_IDLE

...
case  sl_bt_evt_system_external_signal_id:if (notification_enabled == 1 && km_status != KM_IDLE) {if (km_status == KM_SEND_STRING) {send_eetree_string();}else if (km_status == KM_SCROLL_UP) {scroll_with_distance(0x01);}else { // KM_SCROLL_DOWNscroll_with_distance(0xFF);}app_log_info("Key report %d was sent\r\n", km_status);km_status = KM_IDLE;}break;
...

BLE HID

HID(Human Interface Device)人体学接口设备,是生活中常见的输入设备,比如键盘、鼠标等。早期的HID是设备大部分都是通过USB接口来实现,蓝牙技术出现后,通过蓝牙作为传输层,实现了无线HID设备。通过低功耗蓝牙实现的HID功能一般简称为HOGP(HID over Gatt Profile)。BLE HID 规范以 USB HID 规范为基础,因此具体含义仍需参照USB HID文档。

👉 参考:【BLE】HID设备的实现(蓝牙自拍杆、蓝牙键盘、蓝牙鼠标、HID复合设备)

Report Map及报文

键盘设备

👉 参考:DIY蓝牙键盘(1) - 理解键盘报文

Report Map用十六进制数据,描述HID设备的基本信息,例如,按键数量,数据的最大最小值,功能等。为了实现鼠标+键盘复合设备,参考SiliconLabs蓝牙应用示例: bluetooth_hid_keyboard所给出的一个键盘设备的报告映射,并加入Report ID条目:

项目
0x05, 0x01Usage Page (Generic Desktop)
0x09, 0x06Usage (Keyboard)
0xa1, 0x01Collection (Application)
0x85, 0x01Report Id (1)
0x75, 0x01Report Size (1)
0x95, 0x08Report Count (8)
0x05, 0x07Usage Page (Keyboard)
0x19, 0xe0Usage Minimum (Keyboard LeftControl)
0x29, 0xe7Usage Maximum (Keyboard Right GUI)
0x15, 0x00Logical Minimum (0)
0x25, 0x01Logical Maximum (1)
0x75, 0x01Report Size (1)
0x95, 0x08Report Count (8)
0x81, 0x02Input (Data, Variable, Absolute) Modifier byte
0x95, 0x01Report Count (1)
0x75, 0x08Report Size (8)
0x81, 0x01Input (Constant) Reserved byte
0x95, 0x06Report Count (6)
0x75, 0x08Report Size (8)
0x15, 0x00Logical Minimum (0)
0x25, 0x65Logical Maximum (101)
0x05, 0x07Usage Page (Key Codes)
0x05, 0x01Usage Minimum (Reserved (no event indicated))
0x05, 0x01Usage Maximum (Keyboard Application)
0x05, 0x01Input (Data,Array) Key arrays (6 bytes)
0xc0End Collection

这样描述的键盘设备具有通用键盘的基本功能,将按键与释放键信息通过输入报告发送到主机。同时,使用常见的键盘报文结构,其中包含保留字节、修饰符字节与6个键码字节(可以描述最多6个按键同时按下)。键盘报文格式如下所列:

Byte 0Byte 1Byte 2Byte 3Byte 4Byte 5Byte 6Byte 7
Modifier byteReserved byteKey code 1Key code 2Key code 3Key code 4Key code 5Key code 6

其中,第一个字节从LSB开始依次表示:

Bit 0Bit 1Bit 2Bit 3Bit 4Bit 5Bit 6Bit 7
L CtrlL ShiftL AltL GUIR CtrlR ShiftR AltR GUI

第二个字节保留(默认为0)。后面6个字节的每个字节都可以表示一个按键的状态,可以同时有多个按键按下。在手册《HID Usage Tables For Universal Serial Bus (USB)》中,规定了键码与按键的对应关系,例如:

Usage IDUsage Name
0x04Keyboard a & A
0x05Keyboard b & B
0x1DKeyboard z & Z
0x37Keyboard . & >

例如,下述两个报文分别表示a与A(同时按下左Shift + a):

Byte 0Byte 1Byte 2Byte 3Byte 4Byte 5Byte 6Byte 7含义
0x000x000x040x000x000x000x000x00a
Byte 0Byte 1Byte 2Byte 3Byte 4Byte 5Byte 6Byte 7含义
0x040x000x040x000x000x000x000x00A

此外,在发送按下按键的信息后,还需发送释放按键的报文,否则键盘将一直按住。

Byte 1Byte 2Byte 3Byte 4Byte 5Byte 6Byte 7Byte 8含义
0x000x000x000x000x000x000x000x00释放
鼠标设备

鼠标设备的报告映射如下所列:

项目
0x05, 0x01Usage Page (Generic Desktop)
0x09, 0x02Usage (Mouse)
0xa1, 0x01Collection (Application)
0x85, 0x02Report Id (2)
0x75, 0x01Report Size (1)
0x95, 0x08Report Count (8)
0x09, 0x01Usage (Pointer)
0xa1, 0x00Collection (Physical)
0x05, 0x09Usage Page (Buttons)
0x19, 0x01Logical Minimum (1)
0x29, 0x03Logical Maximum (3)
0x15, 0x00Logical Minimum (0)
0x25, 0x01Logical Maximum (1)
0x95, 0x03Report Count (3)
0x75, 0x01Report Count (1)
0x81, 0x02Input(Data, Variable, Absolute); 3 button bits
0x95, 0x01Report Count (1)
0x75, 0x05Report Size (5)
0x81, 0x03Input(Constant); 5 bits padding
0x05, 0x01Usage Page (Generic Desktop)
0x09, 0x30Usage (X)
0x09, 0x31Usage (Y)
0x09, 0x38Usage (Wheel)
0x15, 0x81Logical Minimum (-127)
0x25, 0x7FLogical Maximum (127)
0x75, 0x08Report Size (8)
0x95, 0x03Report Count (3)
0x81, 0x06Input(Data, Variable, Relative); 3 position bytes (X,Y,Wheel)
0xc0End Collection
0xc0End Collection

对于鼠标,上报的数据我们定义了4个字节。其中Byte 0 的bit 0~2分别表示鼠标左键、右键与中键,后4位由设备定义(默认为0)。Byte 1 表示鼠标指针X轴移动,Byte 2 表示鼠标指针Y轴移动(有符号数,具体数值与移动距离的关系可实际测试),Byte 3 表示滚轮移动。鼠标的报文格式如下所列:

Byte 0Byte 1Byte 2Byte 3
bit 0~2 左、右、中键指针X方向移动指针Y方向移动滚轮移动
复合设备

对于两个或以上的HID复合设备来说,需要额外用Report ID描述,在此键盘Report ID为1,鼠标为2。同时,设备的报文前需额外一个字节表示Report ID。将上述两个设备的报告映射写在一起,即可描述键盘+鼠标的复合设备。如图所示,将这个长字节配置至 Bluetooth GATT Configurator 内。

配置Report Map

发送字符串

通过键盘设备发送字符串“EETREE.CN”,各字符码表为:0x08(e)、0x17(t)、0x15®、0x37(.)、0x06©、0x11(n)。要实现大写,还需要修饰键按下左Shift。例如,发送“E”,并结合键盘的Report ID(0x01):

Byte 0Byte 1Byte 2Byte 3Byte 4Byte 5Byte 6Byte 7Byte 8含义
0x010x040x000x080x000x000x000x000x00E
Byte 0Byte 1Byte 2Byte 3Byte 4Byte 5Byte 6Byte 7Byte 8含义
0x010x000x000x000x000x000x000x000x00释放

发送一个字符后,再发送一次全0报文,表示按键释放。两各按键前后最好间隔几十毫秒。由于所发送字符串“EETREE.CN”有连续的字符,因此不方便在一次报文中发送(如此两个“EE”将仅表达一次“E”按下),且“.”无需修饰符,因此索性每个字符都单次发送。

#define REPORT_ID_INDEX         0
#define KB_REPORT_ID            0x01
#define MODIFIER_INDEX          1
#define DATA_INDEX              3
#define LSHIFT_KEY_OFF        	0x00
#define LSHIFT_KEY_ON         	0x02
static uint8_t kb_report_data[] = { 0, 0, 0, 0, 0, 0, 0, 0, 0 };void send_keyboard(uint8_t caps_key, uint8_t c)
{sl_status_t sc;memset(kb_report_data, 0, sizeof(kb_report_data));kb_report_data[REPORT_ID_INDEX] = KB_REPORT_ID;kb_report_data[MODIFIER_INDEX] = caps_key;kb_report_data[DATA_INDEX] = c;sc = sl_bt_gatt_server_notify_all(gattdb_report,sizeof(kb_report_data),kb_report_data);app_assert_status(sc);memset(kb_report_data, 0, sizeof(kb_report_data));kb_report_data[REPORT_ID_INDEX] = KB_REPORT_ID;sc = sl_bt_gatt_server_notify_all(gattdb_report,sizeof(kb_report_data),kb_report_data);app_assert_status(sc);sl_sleeptimer_delay_millisecond(20);
}void send_eetree_string()
{send_keyboard(LSHIFT_KEY_ON, 0x08); // Esend_keyboard(LSHIFT_KEY_ON, 0x08); // Esend_keyboard(LSHIFT_KEY_ON, 0x17); // Tsend_keyboard(LSHIFT_KEY_ON, 0x15); // Rsend_keyboard(LSHIFT_KEY_ON, 0x08); // Esend_keyboard(LSHIFT_KEY_ON, 0x08); // Esend_keyboard(LSHIFT_KEY_OFF,0x37); // .send_keyboard(LSHIFT_KEY_ON, 0x06); // Csend_keyboard(LSHIFT_KEY_ON, 0x11); // N
}

上/下滚动

通过鼠标设备实现上下滚动,并结合鼠标的Report ID(0x02):

Byte 0Byte 1Byte 2Byte 3Byte 4含义
0x020x000x000x000x01上滚
Byte 0Byte 1Byte 2Byte 3Byte 4含义
0x020x000x000x000xFF下滚
#define MOUSE_REPORT_ID         0x02
#define WHEEL_INDEX             4
static uint8_t mouse_report_data[] = { 0, 0, 0, 0, 0 };void scroll_with_distance(uint8_t distance)
{sl_status_t sc;memset(mouse_report_data, 0, sizeof(mouse_report_data));mouse_report_data[REPORT_ID_INDEX] = MOUSE_REPORT_ID;mouse_report_data[WHEEL_INDEX] = distance;sc = sl_bt_gatt_server_notify_all(gattdb_report,sizeof(mouse_report_data),mouse_report_data);app_assert_status(sc);
}

功能展示

开发板连接PC并配对蓝牙后,可以看到XG24 KeyMouse设备已连接,且电量为100%。功能演示参见视频。
板卡实物图
蓝牙配对效果
👉 详细展示参见:B站:基于XG24-EK2703A的BLE HID蓝牙键盘+鼠标复合设备功能开发

项目总结

本次项目通过BLE HID协议,实现了键盘+鼠标复合设备,使用两个按键实现上/下翻页、发送字符串的功能。Silicon Labs的IDE总体感觉还不错,直接在IDE内把GSDK、编译工具链都给安装好。对于配置开发板的外设、IO口、驱动、蓝牙GATT配置等有图形化界面,上手较为容易。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/639297.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Java streamFile

1.Stream流 1.1体验Stream流【理解】 案例需求 按照下面的要求完成集合的创建和遍历 创建一个集合&#xff0c;存储多个字符串元素 把集合中所有以"张"开头的元素存储到一个新的集合 把"张"开头的集合中的长度为3的元素存储到一个新的集合 遍历上一步得…

使用Scrapy 爬取“http://tuijian.hao123.com/”网页中左上角“娱乐”、“体育”、“财经”、“科技”、历史等名称和URL

一、网页信息 二、检查网页&#xff0c;找出目标内容 三、根据网页格式写正常爬虫代码 from bs4 import BeautifulSoup import requestsheaders {User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/107.0.0.0 Safari/53…

每天10个前端小知识 <Day 12>

前端面试基础知识题 1. Promise中&#xff0c;resolve后面的语句是否还会执行&#xff1f; 会被执行。如果不需要执行&#xff0c;需要在 resolve 语句前加上 return。 2. 什么是内存泄漏&#xff1f;什么原因会导致呢&#xff1f; 内存泄露的解释&#xff1a;程序中己动态…

酒店网络SIP融合对讲系统联动110方案

酒店网络SIP融合对讲系统联动110方案 酒店对讲系统设计采用了基于网络传输的对讲系统&#xff0c;利用网络平台&#xff0c;把管理中心和前端各个求助点连接起来&#xff0c;所有的通讯信号和控制协议通过网络进行传输&#xff0c;采用基于网络数字SIP网络对讲系统&#xff0c…

【数据结构】 双链表的基本操作 (C语言版)

目录 一、双链表 1、双链表的定义&#xff1a; 2、双链表表的优缺点&#xff1a; 二、双链表的基本操作算法&#xff08;C语言&#xff09; 1、宏定义 2、创建结构体 3、双链表的初始化 4、双链表表插入 5、双链表的查找 6、双链表的取值 7、求双链表长度 8、双链表…

Linux shell编程学习笔记41:lsblk命令

边缘计算的挑战和机遇 边缘计算面临着数据安全与隐私保护、网络稳定性等挑战&#xff0c;但同时也带来了更强的实时性和本地处理能力&#xff0c;为企业降低了成本和压力&#xff0c;提高了数据处理效率。因此&#xff0c;边缘计算既带来了挑战也带来了机遇&#xff0c;需要我…

阿里云国外服务器价格购买与使用策略

阿里云国外服务器优惠活动「全球云服务器精选特惠」&#xff0c;国外服务器租用价格24元一个月起&#xff0c;免备案适合搭建网站&#xff0c;部署独立站等业务场景&#xff0c;阿里云服务器网aliyunfuwuqi.com分享阿里云国外服务器优惠活动&#xff1a; 全球云服务器精选特惠…

学习笔记-李沐动手学深度学习(二)(08-09、线性回归、优化算法、Softmax回归、损失函数、图片分类)

总结 以_结尾的方法&#xff0c;好像是原位替换&#xff08;即 原地修改&#xff0c;就地修改变量&#xff09;如 fill_() 感恩的心&#xff1a;&#xff08;沐神的直播环境&#xff09; 08-线性回归基础优化算法 引言&#xff08;如何在美国买房&#xff09; 根据现在行…

用ChatGPT教学、科研!亚利桑那州立大学与OpenAI合作

亚利桑那州立大学&#xff08;简称“ASU”&#xff09;在官网宣布与OpenAI达成技术合作。从2024年2月份开始&#xff0c;为所有学生提供ChatGPT企业版访问权限&#xff0c;主要用于学习、课程作业和学术研究等。 为了帮助学生更好地学习ChatGPT和大语言模型产品&#xff0c;AS…

unicloud 云对象 schema

目录 云对象 创建云对象 客户端调用 二、Schema&#xff08;表结构&#xff09; 什么是Schema&#xff1f; 如何编写DB Schema Schema的一级节点 客户端直连数据库 字段属性 字段类型bsonType 默认值defaultValue/forceDefaultValue 云对象 创建云对象 云对象&#…

HarmonyOS鸿蒙应用开发 (一、环境搭建及第一个Hello World)

万事开头难。难在迈出第一步。心无旁骛&#xff0c;万事可破。没有人一开始就能想清楚&#xff0c;只有做起来&#xff0c;目标才会越来越清晰。--马克.扎克伯格 前言 2024年1月16日&#xff0c;华为目前开启已HarmonyOS NEXT开发者预览版Beta招募&#xff0c;报名周期为1月15…

Conmi的正确答案——eclipse C/C++显示“未解析的包含:<xxx.h>”/“Unresolved inclusion: <xxx.h>”

eclipse IDE 版本&#xff1a;2023-12 部分采自&#xff1a;解决方法&#xff1a;关于问题 “C - Unresolved inclusion: <iostream>” 解释事项&#xff1a;方法一可能版本不同&#xff0c;部分界面修改了。这里使用的是方法二的解决方法。&#xff08;或者各位大神的描…

JavaEE中什么是Web容器?

Web容器&#xff08;也称为Servlet引擎&#xff09;是一个用于执行Java Servlet和JSP的服务器端环境。它负责管理和执行在其上运行的Web应用程序。 Tomcat是Web容器 Apache Tomcat 是一个流行的开源的Web容器&#xff0c;它实现了Java Servlet和JavaServer Pages&#xff08;…

【深度学习目标检测】十七、基于深度学习的洋葱检测系统-含GUI和源码(python,yolov8)

使用AI实现洋葱检测对农业具有以下意义&#xff1a; 提高效率&#xff1a;AI技术可以快速、准确地检测出洋葱中的缺陷和问题&#xff0c;从而提高了检测效率&#xff0c;减少了人工检测的时间和人力成本。提高准确性&#xff1a;AI技术通过大量的数据学习和分析&#xff0c;能够…

第五课:MindSpore自动并行

文章目录 第五课&#xff1a;MindSpore自动并行1、学习总结&#xff1a;数据并行模型并行MindSpore算子级并行算子级并行示例 流水线并行GPipe和Micro batch1F1B流水线并行示例 内存优化重计算优化器并行 MindSpore分布式并行模式课程ppt及代码地址 2、学习心得&#xff1a;3、…

如何使用pytorch的Dataset, 来定义自己的Dataset

Dataset与DataLoader的关系 Dataset: 构建一个数据集&#xff0c;其中含有所有的数据样本DataLoader&#xff1a;将构建好的Dataset&#xff0c;通过shuffle、划分batch、多线程num_workers运行的方式&#xff0c;加载到可训练的迭代容器。 import torch from torch.utils.dat…

Qt6入门教程 9:QWidget、QMainWindow和QDialog

目录 一.QWidget 1.窗口和控件 2.事件 二.QMainWindow 三.QDialog 1.模态对话框 1.1模态对话框 1.2.半模态对话框 2.非模态对话框 在用Qt Creator创建Qt Widgets项目时&#xff0c;会默认提供三种基类以供选择&#xff0c;它们分别是QWidget、QMainWIndow和QDialog&am…

SQL 注入总结(详细)

一、前言 这篇文章是最近学习 SQL 注入后的笔记&#xff0c;里面整理了 SQL 常见的注入方式&#xff0c;供大家学习了解 SQL 注入的原理及方法&#xff0c;也方便后续自己回顾&#xff0c;如有什么错误的地方欢迎指出&#xff01; 二、判断注入类型 按照注入点类型分类 数字型…

外贸自建站如何建立?海洋建站的操作指南?

外贸自建站的建站流程什么&#xff1f;做跨境怎么搭建外贸网站&#xff1f; 外贸自建站成为企业开拓国际市场、提升品牌形象的重要途径。然而&#xff0c;对于许多企业而言&#xff0c;如何高效地进行外贸自建站仍然是一个挑战。海洋建站将带您一步步探讨外贸自建站的关键步骤…

计算机网络——面试问题

1 从输⼊ URL 到⻚⾯展示到底发⽣了什么&#xff1f; 1. 先检查浏览器缓存⾥是否有缓存该资源&#xff0c;如果有直接返回&#xff1b;如果没有进⼊下⼀ 步⽹络请求。 2. ⽹络请求前&#xff0c;进⾏ DNS 解析 &#xff0c;以获取请求域名的 IP地址 。 3. 浏览器与服务器…