BlueZ 學習之GATT Server開發

Linux下,使用C語言開發一個簡單的GATT Server,我的Ubuntu上跑的BlueZ版本是5.79,使用的GLib庫版本是2.85.2,這里我直接使用GLib里的D?Bus來實現與BlueZ通信。BlueZ 官方推薦通過 D-Bus 進行通信和控制,如果是要使用原始的hci接口來實現,可以參考BlueZ源碼目錄下的
tools/btgatt-server.c 文件。

我這里是使用的usb藍牙,支持經典藍牙和BLE(低功耗藍牙)模式,我直接通過bluez工具來開啟BLE功能和BLE廣播,常用的一些控制命令如下,做下記錄:

### 經典藍牙 ###bluetoothctl                        # 運行 bluetoothctl
power on                            # 打開藍牙適配器(開啟藍牙功能)
agent on                            # 啟用配對助手(允許輸入或確認配對碼等操作)
scan on                             # 開始掃描附近的藍牙設備
scan off                            # 停止掃描
pair 94:87:E0:8C:D5:94              # 與指定MAC地址的設備進行配對
connect 94:87:E0:8C:D5:94           # 連接到已配對的指定設備
trust 94:87:E0:8C:D5:94             # 設置信任該設備(以后自動連接無需確認)sudo systemctl restart bluetooth    # 重啟藍牙服務
remove 94:87:E0:8C:D5:94            # 清除舊配對信息### BLE ###通過 bluetoothctl 控制bluetoothctl power on          # 開啟藍牙適配器
bluetoothctl discoverable on   # 讓本設備對其他設備可見
bluetoothctl pairable on       # 允許其他設備與本設備配對
bluetoothctl advertise on      # 開啟BLE廣播(讓本設備可以被BLE客戶端發現)通過 btmgmt 控制sudo btmgmt power on         # 開啟藍牙
sudo btmgmt le on            # 開啟BLE(低功耗藍牙)功能
sudo btmgmt advertising on   # 開啟BLE廣播
sudo btmgmt connectable on   # 允許連接
sudo btmgmt info             # 查看適配器支持和當前狀態

源碼如下 gatt-server.c:

#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <pthread.h>
#include <locale.h>#include "gio/gio.h"// ================== 藍牙 GATT 應用的一些常量定義 ==================
#define BLUEZ_BUS_NAME "org.bluez"            // BlueZ 的 DBus 服務名
#define ADAPTER_PATH "/org/bluez/hci0"        // 默認藍牙適配器路徑(hci0)
#define APP_PATH "/org/example"               // 應用根路徑
#define SERVICE_PATH APP_PATH "/service0"     // 自定義 GATT 服務路徑
#define CHAR_READ_PATH SERVICE_PATH "/char0"  // 自定義 "只讀" 特征路徑
#define CHAR_WRITE_PATH SERVICE_PATH "/char1" // 自定義 "可寫" 特征路徑// UUID 定義(隨便寫的示例 UUID)
static const gchar *SERVICE_UUID = "12345678-1234-5678-1234-56789abcdef0";
static const gchar *CHAR_READ_UUID = "12345678-1234-5678-1234-56789abcdef1";
static const gchar *CHAR_WRITE_UUID = "12345678-1234-5678-1234-56789abcdef2";// 特征的訪問權限標志
static const gchar *flags_read[] = {"read", NULL};
static const gchar *flags_write[] = {"write", NULL};// ================== Introspection XML 定義 ==================
// 這些 XML 定義了 DBus 接口(ObjectManager、Service、Characteristic)// ObjectManager 接口(用于告訴 BlueZ 我們有哪些對象)
static const gchar *object_manager_xml ="<node>""  <interface name='org.freedesktop.DBus.ObjectManager'>""    <method name='GetManagedObjects'>""      <arg name='objects' type='a{oa{sa{sv}}}' direction='out'/>""    </method>""  </interface>""</node>";// GATT Service 接口
static const gchar *service_xml ="<node>""  <interface name='org.bluez.GattService1'>""    <property name='UUID' type='s' access='read'/>""    <property name='Primary' type='b' access='read'/>""  </interface>""</node>";// GATT Characteristic 接口(包含讀寫方法)
static const gchar *char_xml ="<node>""  <interface name='org.bluez.GattCharacteristic1'>""    <property name='UUID' type='s' access='read'/>""    <property name='Service' type='o' access='read'/>""    <property name='Flags' type='as' access='read'/>""    <method name='ReadValue'>""      <arg name='options' type='a{sv}' direction='in'/>""      <arg name='value' type='ay' direction='out'/>""    </method>""    <method name='WriteValue'>""      <arg name='value' type='ay' direction='in'/>""      <arg name='options' type='a{sv}' direction='in'/>""    </method>""  </interface>""</node>";// ================== 全局只讀特征的內容 ==================
#define READ_VALUE_MAX_LEN 256
static guint8 read_value_buffer[READ_VALUE_MAX_LEN] = "Hello"; // 初始內容
static gsize read_value_len = 5;                               // 當前長度
static pthread_mutex_t read_value_mutex = PTHREAD_MUTEX_INITIALIZER;// 一個線程,用來從終端輸入更新 "只讀特征" 的內容
void *stdin_input_thread(void *arg)
{char input[READ_VALUE_MAX_LEN];while (1){printf("請輸入新的只讀特征內容(按回車確認):\n");if (fgets(input, sizeof(input), stdin)){size_t len = strlen(input);if (len > 0 && input[len - 1] == '\n')input[len - 1] = '\0'; // 去掉換行pthread_mutex_lock(&read_value_mutex);read_value_len = strlen(input);if (read_value_len > READ_VALUE_MAX_LEN)read_value_len = READ_VALUE_MAX_LEN;memcpy(read_value_buffer, input, read_value_len);pthread_mutex_unlock(&read_value_mutex);printf("已更新只讀特征內容:%s\n", input);}}return NULL;
}// ================== DBus 方法調用處理 ==================
static void on_method_call(GDBusConnection *conn,const gchar *sender,const gchar *object_path,const gchar *interface_name,const gchar *method_name,GVariant *parameters,GDBusMethodInvocation *invocation,gpointer user_data)
{// 打印調試信息g_print("\n*** METHOD CALL RECEIVED ***\n");g_print("Sender: %s\n", sender);g_print("Object Path: %s\n", object_path);g_print("Interface: %s\n", interface_name);g_print("Method: %s\n", method_name);g_print("*****************************\n");// ============= 處理 GetManagedObjects(告訴 BlueZ 我們有哪些對象)=============if (g_strcmp0(interface_name, "org.freedesktop.DBus.ObjectManager") == 0 &&g_strcmp0(method_name, "GetManagedObjects") == 0){g_print("*** HANDLING GetManagedObjects ***\n");GVariantBuilder objects_builder;g_variant_builder_init(&objects_builder, G_VARIANT_TYPE("a{oa{sa{sv}}}"));// 添加 GATT Service 對象{GVariantBuilder service_props;g_variant_builder_init(&service_props, G_VARIANT_TYPE("a{sv}"));g_variant_builder_add(&service_props, "{sv}", "UUID", g_variant_new_string(SERVICE_UUID));g_variant_builder_add(&service_props, "{sv}", "Primary", g_variant_new_boolean(TRUE));GVariantBuilder service_interfaces;g_variant_builder_init(&service_interfaces, G_VARIANT_TYPE("a{sa{sv}}"));g_variant_builder_add(&service_interfaces, "{s@a{sv}}", "org.bluez.GattService1",g_variant_builder_end(&service_props));g_variant_builder_add(&objects_builder, "{o@a{sa{sv}}}", SERVICE_PATH,g_variant_builder_end(&service_interfaces));}// 添加 "只讀特征" 對象{GVariantBuilder char_read_props;g_variant_builder_init(&char_read_props, G_VARIANT_TYPE("a{sv}"));g_variant_builder_add(&char_read_props, "{sv}", "UUID", g_variant_new_string(CHAR_READ_UUID));g_variant_builder_add(&char_read_props, "{sv}", "Service", g_variant_new_object_path(SERVICE_PATH));g_variant_builder_add(&char_read_props, "{sv}", "Flags", g_variant_new_strv(flags_read, -1));GVariantBuilder char_read_interfaces;g_variant_builder_init(&char_read_interfaces, G_VARIANT_TYPE("a{sa{sv}}"));g_variant_builder_add(&char_read_interfaces, "{s@a{sv}}", "org.bluez.GattCharacteristic1",g_variant_builder_end(&char_read_props));g_variant_builder_add(&objects_builder, "{o@a{sa{sv}}}", CHAR_READ_PATH,g_variant_builder_end(&char_read_interfaces));}// 添加 "可寫特征" 對象{GVariantBuilder char_write_props;g_variant_builder_init(&char_write_props, G_VARIANT_TYPE("a{sv}"));g_variant_builder_add(&char_write_props, "{sv}", "UUID", g_variant_new_string(CHAR_WRITE_UUID));g_variant_builder_add(&char_write_props, "{sv}", "Service", g_variant_new_object_path(SERVICE_PATH));g_variant_builder_add(&char_write_props, "{sv}", "Flags", g_variant_new_strv(flags_write, -1));GVariantBuilder char_write_interfaces;g_variant_builder_init(&char_write_interfaces, G_VARIANT_TYPE("a{sa{sv}}"));g_variant_builder_add(&char_write_interfaces, "{s@a{sv}}", "org.bluez.GattCharacteristic1",g_variant_builder_end(&char_write_props));g_variant_builder_add(&objects_builder, "{o@a{sa{sv}}}", CHAR_WRITE_PATH,g_variant_builder_end(&char_write_interfaces));}// 返回對象列表GVariant *result = g_variant_new("(@a{oa{sa{sv}}})", g_variant_builder_end(&objects_builder));g_dbus_method_invocation_return_value(invocation, result);g_print("*** GetManagedObjects completed ***\n");return;}// ============= 處理 ReadValue 請求(讀取只讀特征的值)=============if (g_strcmp0(interface_name, "org.bluez.GattCharacteristic1") == 0 &&g_strcmp0(method_name, "ReadValue") == 0){g_print("ReadValue called on %s\n", object_path);GVariant *value;pthread_mutex_lock(&read_value_mutex);value = g_variant_new_fixed_array(G_VARIANT_TYPE_BYTE,read_value_buffer, read_value_len, sizeof(guint8));pthread_mutex_unlock(&read_value_mutex);g_dbus_method_invocation_return_value(invocation, g_variant_new("(@ay)", value));return;}// ============= 處理 WriteValue 請求(寫入數據到可寫特征)=============if (g_strcmp0(interface_name, "org.bluez.GattCharacteristic1") == 0 &&g_strcmp0(method_name, "WriteValue") == 0){GVariant *value_variant;GVariant *options_variant;g_variant_get(parameters, "(@ay@a{sv})", &value_variant, &options_variant);gsize len;const guint8 *data = g_variant_get_fixed_array(value_variant, &len, sizeof(guint8));g_print("WriteValue called on %s, data: ", object_path);for (gsize i = 0; i < len; i++){g_print("%02X ", data[i]);}g_print(" string: %.*s\n", (int)len, (const char *)data);g_variant_unref(value_variant);g_variant_unref(options_variant);g_dbus_method_invocation_return_value(invocation, NULL);return;}// 未知方法g_dbus_method_invocation_return_error(invocation, G_DBUS_ERROR,G_DBUS_ERROR_UNKNOWN_METHOD,"Unknown method: %s", method_name);
}// ================== 屬性讀取處理 ==================
static GVariant *on_get_property(GDBusConnection *conn,const gchar *sender,const gchar *object_path,const gchar *interface_name,const gchar *property_name,GError **error,gpointer user_data)
{g_print("Get property: %s.%s on %s\n", interface_name, property_name, object_path);// 處理 Service 屬性if (g_strcmp0(interface_name, "org.bluez.GattService1") == 0){if (g_strcmp0(property_name, "UUID") == 0)return g_variant_new_string(SERVICE_UUID);if (g_strcmp0(property_name, "Primary") == 0)return g_variant_new_boolean(TRUE);}// 處理 Characteristic 屬性if (g_strcmp0(interface_name, "org.bluez.GattCharacteristic1") == 0){if (g_strcmp0(object_path, CHAR_READ_PATH) == 0){if (g_strcmp0(property_name, "UUID") == 0)return g_variant_new_string(CHAR_READ_UUID);if (g_strcmp0(property_name, "Service") == 0)return g_variant_new_object_path(SERVICE_PATH);if (g_strcmp0(property_name, "Flags") == 0)return g_variant_new_strv(flags_read, -1);}if (g_strcmp0(object_path, CHAR_WRITE_PATH) == 0){if (g_strcmp0(property_name, "UUID") == 0)return g_variant_new_string(CHAR_WRITE_UUID);if (g_strcmp0(property_name, "Service") == 0)return g_variant_new_object_path(SERVICE_PATH);if (g_strcmp0(property_name, "Flags") == 0)return g_variant_new_strv(flags_write, -1);}}// 未知屬性g_set_error(error, G_DBUS_ERROR, G_DBUS_ERROR_UNKNOWN_PROPERTY,"Unknown property: %s", property_name);return NULL;
}// ================== 注冊回調 ==================
static GMainLoop *main_loop = NULL;
static gboolean registration_success = FALSE;static void on_register_application_done(GObject *source_object,GAsyncResult *res,gpointer user_data)
{GError *error = NULL;GVariant *result = g_dbus_proxy_call_finish(G_DBUS_PROXY(source_object), res, &error);if (result){g_print("SUCCESS: GATT Application registered!\n");g_variant_unref(result);registration_success = TRUE;}else{g_printerr("RegisterApplication failed: %s\n", error->message);g_error_free(error);registration_success = FALSE;}
}// ================== 主函數 ==================
int main(int argc, char *argv[])
{GError *error = NULL;// 設置本地化環境,不然 g_print 輸出中文時亂碼setlocale(LC_ALL, "");g_print("Connecting to system bus...\n");// 連接到系統總線GDBusConnection *connection = g_bus_get_sync(G_BUS_TYPE_SYSTEM, NULL, &error);if (!connection){g_printerr("Failed to connect to system bus: %s\n", error->message);return 1;}g_print("Parsing introspection XML...\n");GDBusNodeInfo *object_manager_info = g_dbus_node_info_new_for_xml(object_manager_xml, NULL);GDBusNodeInfo *service_info = g_dbus_node_info_new_for_xml(service_xml, NULL);GDBusNodeInfo *char_info = g_dbus_node_info_new_for_xml(char_xml, NULL);// 定義虛表(方法回調 + 屬性回調)static const GDBusInterfaceVTable interface_vtable = {.method_call = on_method_call,.get_property = on_get_property,.set_property = NULL};// 注冊 ObjectManagerg_print("Registering ObjectManager at %s\n", APP_PATH);guint reg_id1 = g_dbus_connection_register_object(connection, APP_PATH,object_manager_info->interfaces[0],&interface_vtable, NULL, NULL, &error);if (!reg_id1){g_printerr("Failed to register ObjectManager: %s\n", error->message);return 1;}// 注冊 Serviceg_print("Registering Service at %s\n", SERVICE_PATH);guint reg_id2 = g_dbus_connection_register_object(connection, SERVICE_PATH,service_info->interfaces[0],&interface_vtable, NULL, NULL, &error);if (!reg_id2){g_printerr("Failed to register Service: %s\n", error->message);return 1;}// 注冊 Read Characteristicg_print("Registering Read Characteristic at %s\n", CHAR_READ_PATH);guint reg_id3 = g_dbus_connection_register_object(connection, CHAR_READ_PATH,char_info->interfaces[0],&interface_vtable, NULL, NULL, &error);if (!reg_id3){g_printerr("Failed to register Read Characteristic: %s\n", error->message);return 1;}// 注冊 Write Characteristicg_print("Registering Write Characteristic at %s\n", CHAR_WRITE_PATH);guint reg_id4 = g_dbus_connection_register_object(connection, CHAR_WRITE_PATH,char_info->interfaces[0],&interface_vtable, NULL, NULL, &error);if (!reg_id4){g_printerr("Failed to register Write Characteristic: %s\n", error->message);return 1;}// 創建 GATT Manager 的代理GDBusProxy *manager_proxy = g_dbus_proxy_new_sync(connection,G_DBUS_PROXY_FLAGS_NONE,NULL,BLUEZ_BUS_NAME,ADAPTER_PATH,"org.bluez.GattManager1",NULL,&error);if (!manager_proxy){g_printerr("Failed to create GattManager1 proxy: %s\n", error->message);return 1;}g_print("Attempting to register GATT application...\n");// 啟動輸入線程(用于動態修改只讀特征內容)pthread_t tid;pthread_create(&tid, NULL, stdin_input_thread, NULL);// 創建主循環main_loop = g_main_loop_new(NULL, FALSE);// 異步調用 RegisterApplicationg_dbus_proxy_call(manager_proxy, "RegisterApplication",g_variant_new("(oa{sv})", APP_PATH, NULL),G_DBUS_CALL_FLAGS_NONE,-1,NULL,on_register_application_done,NULL);// 運行主循環(等待 DBus 事件)g_main_loop_run(main_loop);// 程序退出時的清理g_object_unref(manager_proxy);g_dbus_connection_unregister_object(connection, reg_id1);g_dbus_connection_unregister_object(connection, reg_id2);g_dbus_connection_unregister_object(connection, reg_id3);g_dbus_connection_unregister_object(connection, reg_id4);g_object_unref(connection);g_print("Program finished.\n");return registration_success ? 0 : 1;
}

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/bicheng/95614.shtml
繁體地址,請注明出處:http://hk.pswp.cn/bicheng/95614.shtml
英文地址,請注明出處:http://en.pswp.cn/bicheng/95614.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

【Linux基礎】Linux文件系統深度解析:EXT4與XFS技術詳解與應用

目錄 引言 1 Linux文件系統概述 1.1 文件系統的基本概念 1.2 日志文件系統的概念 2 EXT4文件系統詳解 2.1 EXT4概述 2.2 EXT4的磁盤結構 2.3 EXT4的inode結構 2.4 EXT4的新特性 2.4.1 Extents 2.4.2 延遲分配 2.4.3 快速文件系統檢查 2.5 EXT4的性能特點 3 XFS文…

埃文科技榮獲2025年“數據要素×”大賽河南分賽二等獎

2025年8月19日&#xff0c;2025年“數據要素”大賽河南分賽決賽在鄭州舉行&#xff0c;本屆河南分賽聚焦數據價值賦能。鄭州埃文科技有限公司&#xff08;以下簡稱“埃文科技”&#xff09;憑借其前沿成果“IP地址高精度地理定位及應用場景劃分數據集”&#xff0c;從500多支參…

鏈上迷局:區塊鏈技術的法律暗礁與合規導航

高鵬律師首席數據官&#xff0c;數字經濟團隊創作AI輔助區塊鏈&#xff0c;這個被譽為“信任機器”的技術&#xff0c;正以顛覆性的力量重塑數字經濟的底層邏輯。從比特幣的橫空出世到NFT的全民狂歡&#xff0c;從DeFi的金融革命到DAO的組織重構&#xff0c;技術永不眠&#xf…

線性代數基礎 | 基底 / 矩陣 / 行列式 / 秩 / 線性方程組

注&#xff1a;本文為 “線性代數基礎 ” 相關合輯。 略作重排&#xff0c;未作全校。 如有內容異常&#xff0c;請看原文。 線性代數的本質&#xff08;1&#xff09;——基底、向量、線性變換、逆陣、行列式 野指針小李于 2020-08-13 16:34:45 發布 零、基底 在展開后續內…

GORM.io詳細指南

GORM.io 詳細指南 GORM&#xff08;全稱 Go ORM&#xff09;是一個功能豐富的 ORM&#xff08;Object-Relational Mapping&#xff09;庫&#xff0c;用于 Go 語言。它簡化了數據庫操作&#xff0c;將 SQL 查詢映射到 Go 結構體&#xff0c;支持多種數據庫后端。GORM 的設計理念…

【Flask】測試平臺開發,應用管理模塊實現-第十一篇

概述通過Element UI抽屜和表單校驗&增改接口合并實現應用管理Drawer 抽屜之前產品修改和添加是使用Dialog組件實現的&#xff0c;但這個組件有時候并不滿足我們的需求, 比如表單很長, 亦或是你需要臨時展示一些文檔, Drawer 是可以從側面彈出的一個層&#xff0c;可以容納更…

Elasticsearch 深分頁限制與解決方案

最近在準備面試&#xff0c;正把平時積累的筆記、項目中遇到的問題與解決方案、對核心原理的理解&#xff0c;以及高頻業務場景的應對策略系統梳理一遍&#xff0c;既能加深記憶&#xff0c;也能讓知識體系更扎實&#xff0c;供大家參考&#xff0c;歡迎討論。在項目中遇到一個…

基于偏最小二乘法PLS多輸入單輸出的回歸預測【MATLAB】

基于偏最小二乘法&#xff08;PLS&#xff09;多輸入單輸出的回歸預測【MATLAB】 在科學研究和工程實踐中&#xff0c;我們常常需要根據多個相關變量來預測一個關鍵結果。例如&#xff0c;根據氣溫、濕度、風速等多個氣象因素預測空氣質量指數&#xff0c;或根據多種原材料成分…

SQL Server核心架構深度解析

SQL Server 的體系結構是一個復雜但設計精密的系統&#xff0c;主要可以分為四大核心組件&#xff0c;它們協同工作以管理數據庫、處理查詢、確保數據安全與一致性。以下是其體系結構的核心組成部分&#xff1a; 核心組件&#xff1a;協議層 (Protocol Layer) 作用&#xff1a;…

Django REST Framework Serializer 進階教程

1. 序列化器概述 在 Django REST Framework&#xff08;DRF&#xff09;中&#xff0c;序列化器&#xff08;Serializer&#xff09;用于將復雜的數據類型&#xff08;如模型實例&#xff09;轉換為 JSON 格式&#xff0c;以便于 API 返回給客戶端。此外&#xff0c;序列化器還…

面試問題詳解十四:Qt 多線程同步【QSemaphore】講解

在多線程開發中&#xff0c;經常需要控制多個線程對共享資源的訪問數量。例如限制同時下載文件的數量、控制數據庫連接池的連接使用等等。這時候&#xff0c;Qt 提供的 QSemaphore&#xff08;信號量&#xff09;就非常派得上用場。一、什么是 QSemaphore&#xff1f; QSemapho…

Spark mapGroups 函數詳解與多種用法示例

mapGroups 是 Spark 中一個強大的分組操作函數&#xff0c;它允許你對每個分組應用自定義邏輯并返回一個結果。以下是多個使用簡單樣例數據的具體用法示例。基礎示例數據假設我們有一個簡單的學生成績數據集&#xff1a;// 創建示例DataFrame val studentScores Seq(("Ma…

【圖論】Graphs.jl 圖數據的讀寫與生成器

文章目錄圖數據的讀寫Graphs.loadgraphGraphs.loadgraphsGraphs.savegraph保存單個圖保存圖字典Graphs.loadlg_multGraphs.savelgGraphs.savelg_mult圖的生成器1. 隨機圖模型1.1 Erd?s–Rnyi 模型1.2 巴拉巴西-阿爾伯特模型 (無標度網絡)1.3 小世界網絡模型1.4 隨機塊模型 (SB…

Go指針全解析:從基礎到實戰

基本概念與定義指針的定義指針是一種特殊的變量類型&#xff0c;它存儲的不是實際數據值&#xff0c;而是另一個變量在計算機內存中的地址。在底層實現上&#xff0c;指針本質上是保存內存位置的無符號整數&#xff0c;它直接指向內存中的特定位置&#xff0c;允許程序直接操作…

Oracle 查詢有哪些用戶 提示用戶名密碼無效

要查詢 Oracle 數據庫中的所有用戶&#xff0c;可以使用以下 SQL 查詢語句。這個查詢將返回數據庫中所有用戶的列表。 [] SELECT username FROM all_users ORDER BY username;如果你有足夠的權限&#xff08;通常是 DBA 權限&#xff09;&#xff0c;你也可以使用 dba_users 視…

小白成長之路-develops -jenkins部署lnmp平臺

文章目錄一、準備工作1.1兩臺虛擬機1.2配置文件1.3免密登錄二、實戰1.構建主item2.測試nginx,php,mysql2.1新建測試項目2.2與正式項目綁定構建后的操作2.3測試2.4導入discuz項目總結一、準備工作 1.1兩臺虛擬機 服務器&#xff1a;192.168.144.24 客戶端&#xff1a;192.168.…

【HarmonyOS 6】仿AI喚起屏幕邊緣流光特效

【HarmonyOS 6】仿AI喚起屏幕邊緣流光特效 一、前言 最近在做 HarmonyOS 6.0 的適配&#xff0c;發現 Beta1版本里多了個很實用的視效功能——自帶背景的雙邊流光。 之前做屏幕邊緣流光特效的時候&#xff0c;要么得自己寫漸變動畫拼效果&#xff0c;要么就得套好幾個組件疊層&…

跟做springboot尚品甄選項目

springbootvue3 【尚硅谷Java項目《尚品甄選》 SpringBootSpringCloud萌新學會企業級java項目】003.后臺系統-搭建前端環境&#xff08;工程創建&#xff09;_嗶哩嗶哩_bilibili E:\project\AllProJect\Shangpin Selection\項目材料素材\課件\尚品甄選項目課件 前端套用框架…

【Linux】創建線程

提示&#xff1a;文章寫完后&#xff0c;目錄可以自動生成&#xff0c;如何生成可參考右邊的幫助文檔 文章目錄 文章目錄 一、為什么需要線程&#xff1f; 創建線程 示例&#xff1a;計算斐波恩夕法 一、為什么需要線程&#xff1f; 在多核處理器的計算機上&#xff0c;線程可…

HTML應用指南:利用POST請求獲取全國九號電動車體驗店服務店位置信息

九號公司(Ninebot)作為全球領先的智能短途出行解決方案提供商,始終秉持“智慧移動,愉悅生活”的品牌理念,致力于為個人用戶打造安全、智能、時尚的城市出行體驗。依托“智能硬件 + 數字服務 + 線下觸點”三位一體的戰略布局,九號公司已建立起覆蓋全國、輻射全球的銷售與服…