Go Web后端开发:智能家居系统的后端实现

Go Web后端开发:智能家居系统的后端实现

关键词:Go语言、Web后端开发、智能家居系统、RESTful API、MQTT、数据库

摘要:本文聚焦于使用Go语言进行智能家居系统后端的开发。首先介绍了智能家居系统后端开发的背景,包括目的、预期读者、文档结构和相关术语。接着阐述了核心概念如RESTful API、MQTT协议以及它们之间的联系,并给出了相应的架构示意图和Mermaid流程图。详细讲解了核心算法原理,通过Python代码示例进行说明,同时介绍了相关的数学模型和公式。在项目实战部分,从开发环境搭建开始,给出了源代码的详细实现和解读。还探讨了智能家居系统后端的实际应用场景,推荐了学习资源、开发工具框架以及相关论文著作。最后总结了未来发展趋势与挑战,并提供了常见问题解答和扩展阅读参考资料。

1. 背景介绍

1.1 目的和范围

智能家居系统旨在为用户提供便捷、高效、安全的家居生活体验。通过将各种家居设备连接到网络,用户可以通过手机、平板电脑等终端设备远程控制这些设备。本项目的目的是使用Go语言开发一个智能家居系统的后端,实现设备管理、用户认证、数据存储和处理等功能。

本项目的范围包括:

  • 设计和实现RESTful API,用于与前端应用进行交互。
  • 集成MQTT协议,实现与智能家居设备的通信。
  • 开发数据库管理模块,用于存储设备信息、用户信息和设备状态数据。
  • 实现用户认证和授权机制,确保系统的安全性。

1.2 预期读者

本文预期读者包括:

  • 对Go语言开发感兴趣的程序员和开发者。
  • 想要学习智能家居系统后端开发的技术爱好者。
  • 从事物联网相关领域的工程师和研究人员。

1.3 文档结构概述

本文的文档结构如下:

  1. 背景介绍:介绍项目的目的、范围、预期读者和文档结构。
  2. 核心概念与联系:阐述RESTful API、MQTT协议等核心概念及其之间的联系。
  3. 核心算法原理 & 具体操作步骤:讲解核心算法原理,并通过Python代码示例进行说明。
  4. 数学模型和公式 & 详细讲解 & 举例说明:介绍相关的数学模型和公式,并举例说明。
  5. 项目实战:代码实际案例和详细解释说明:包括开发环境搭建、源代码实现和代码解读。
  6. 实际应用场景:探讨智能家居系统后端的实际应用场景。
  7. 工具和资源推荐:推荐学习资源、开发工具框架和相关论文著作。
  8. 总结:未来发展趋势与挑战:总结智能家居系统后端开发的未来发展趋势和面临的挑战。
  9. 附录:常见问题与解答:提供常见问题的解答。
  10. 扩展阅读 & 参考资料:提供扩展阅读的资料和参考链接。

1.4 术语表

1.4.1 核心术语定义
  • RESTful API:一种基于HTTP协议的API设计风格,遵循REST原则,使用HTTP方法(如GET、POST、PUT、DELETE)来进行资源的操作。
  • MQTT:一种轻量级的消息传输协议,适用于物联网设备之间的通信,采用发布 - 订阅模式。
  • JSON:一种轻量级的数据交换格式,易于人类阅读和编写,同时也易于机器解析和生成。
  • JWT:JSON Web Token,一种用于身份验证和授权的开放标准,通过JSON对象来安全地传输信息。
1.4.2 相关概念解释
  • 微服务架构:一种将应用程序拆分为多个小型、自治的服务的架构风格,每个服务可以独立开发、部署和扩展。
  • 数据库事务:一组不可分割的数据库操作序列,要么全部执行成功,要么全部失败回滚。
1.4.3 缩略词列表
  • API:Application Programming Interface,应用程序编程接口。
  • HTTP:Hypertext Transfer Protocol,超文本传输协议。
  • TCP:Transmission Control Protocol,传输控制协议。
  • UDP:User Datagram Protocol,用户数据报协议。

2. 核心概念与联系

核心概念原理

RESTful API

RESTful API是一种基于HTTP协议的API设计风格,它将资源抽象为URL,使用HTTP方法(如GET、POST、PUT、DELETE)来对资源进行操作。例如,要获取一个设备的信息,可以使用GET请求访问该设备的URL;要创建一个新的设备,可以使用POST请求将设备信息发送到设备列表的URL。

RESTful API的优点包括:

  • 易于理解和使用,符合HTTP协议的语义。
  • 支持缓存,提高性能。
  • 可扩展性强,易于添加新的资源和操作。
MQTT协议

MQTT是一种轻量级的消息传输协议,适用于物联网设备之间的通信。它采用发布 - 订阅模式,设备可以发布消息到特定的主题,其他设备可以订阅这些主题以接收消息。MQTT协议的优点包括:

  • 轻量级,占用带宽少,适合低带宽、不稳定的网络环境。
  • 支持QoS(Quality of Service),确保消息的可靠传输。
  • 易于实现,有多种编程语言的客户端库可供使用。

架构示意图

+------------------+        +------------------+        +------------------+
|   前端应用        | <----> |   RESTful API      | <----> |   数据库          |
+------------------+        +------------------+        +------------------+
                                 |
                                 | MQTT
                                 v
+------------------+
|   智能家居设备    |
+------------------+

Mermaid流程图

HTTP请求
数据库操作
MQTT消息
MQTT消息
前端应用
RESTful API
数据库
智能家居设备

3. 核心算法原理 & 具体操作步骤

核心算法原理

用户认证和授权

在智能家居系统中,用户认证和授权是确保系统安全的重要环节。我们可以使用JWT(JSON Web Token)来实现用户认证和授权。JWT是一种用于身份验证和授权的开放标准,它通过JSON对象来安全地传输信息。

JWT的工作流程如下:

  1. 用户登录时,后端验证用户的用户名和密码。
  2. 如果验证成功,后端生成一个JWT,并将其返回给前端。
  3. 前端在后续的请求中,将JWT包含在请求头中。
  4. 后端接收到请求后,验证JWT的有效性。如果有效,则允许用户访问相应的资源。
设备状态同步

为了实现设备状态的实时同步,我们可以使用MQTT协议。当设备状态发生变化时,设备将状态信息发布到MQTT主题。后端订阅这些主题,接收到设备状态信息后,更新数据库中的设备状态。

具体操作步骤

用户认证和授权的Python代码示例
import jwt
from datetime import datetime, timedelta
from flask import Flask, request, jsonify

app = Flask(__name__)
SECRET_KEY = 'your_secret_key'

# 模拟用户数据库
users = {
    'user1': 'password1'
}

# 用户登录接口
@app.route('/login', methods=['POST'])
def login():
    data = request.get_json()
    username = data.get('username')
    password = data.get('password')

    if username in users and users[username] == password:
        # 生成JWT
        payload = {
            'username': username,
            'exp': datetime.utcnow() + timedelta(minutes=30)
        }
        token = jwt.encode(payload, SECRET_KEY, algorithm='HS256')
        return jsonify({'token': token})
    else:
        return jsonify({'message': 'Invalid username or password'}), 401

# 受保护的接口
@app.route('/protected', methods=['GET'])
def protected():
    token = request.headers.get('Authorization')
    if not token:
        return jsonify({'message': 'Missing token'}), 401

    try:
        token = token.replace('Bearer ', '')
        payload = jwt.decode(token, SECRET_KEY, algorithms=['HS256'])
        return jsonify({'message': f'Hello, {payload["username"]}'})
    except jwt.ExpiredSignatureError:
        return jsonify({'message': 'Token has expired'}), 401
    except jwt.InvalidTokenError:
        return jsonify({'message': 'Invalid token'}), 401

if __name__ == '__main__':
    app.run(debug=True)
设备状态同步的Python代码示例
import paho.mqtt.client as mqtt

# MQTT配置
MQTT_BROKER = 'localhost'
MQTT_PORT = 1883
MQTT_TOPIC = 'smart_home/devices/status'

# 连接成功回调函数
def on_connect(client, userdata, flags, rc):
    print(f"Connected with result code {rc}")
    client.subscribe(MQTT_TOPIC)

# 接收到消息回调函数
def on_message(client, userdata, msg):
    print(f"Received message on topic '{msg.topic}': {msg.payload.decode()}")
    # 这里可以更新数据库中的设备状态

client = mqtt.Client()
client.on_connect = on_connect
client.on_message = on_message

client.connect(MQTT_BROKER, MQTT_PORT, 60)
client.loop_forever()

4. 数学模型和公式 & 详细讲解 & 举例说明

哈希算法

在用户认证中,为了安全地存储用户密码,我们通常使用哈希算法对密码进行加密。常见的哈希算法有MD5、SHA-1、SHA-256等。哈希算法的特点是将任意长度的输入数据转换为固定长度的输出数据,并且无法通过输出数据反推出输入数据。

以SHA-256为例,其数学公式可以表示为:
H = S H A 256 ( m ) H = SHA256(m) H=SHA256(m)
其中, m m m 是输入的消息, H H H 是输出的哈希值。

举例说明

在Python中,我们可以使用hashlib库来计算SHA-256哈希值:

import hashlib

password = 'password1'
hash_object = hashlib.sha256(password.encode())
hex_dig = hash_object.hexdigest()
print(hex_dig)

消息队列的消息可靠性模型

在MQTT协议中,为了确保消息的可靠传输,引入了QoS(Quality of Service)等级。QoS有三个等级:0、1和2。

QoS 0

QoS 0表示最多一次传输,消息可能会丢失,但不会重复。其数学模型可以表示为:
P l o s t > 0 , P d u p l i c a t e = 0 P_{lost} > 0, P_{duplicate} = 0 Plost>0,Pduplicate=0
其中, P l o s t P_{lost} Plost 是消息丢失的概率, P d u p l i c a t e P_{duplicate} Pduplicate 是消息重复的概率。

QoS 1

QoS 1表示至少一次传输,消息不会丢失,但可能会重复。其数学模型可以表示为:
P l o s t = 0 , P d u p l i c a t e > 0 P_{lost} = 0, P_{duplicate} > 0 Plost=0,Pduplicate>0

QoS 2

QoS 2表示恰好一次传输,消息既不会丢失也不会重复。其数学模型可以表示为:
P l o s t = 0 , P d u p l i c a t e = 0 P_{lost} = 0, P_{duplicate} = 0 Plost=0,Pduplicate=0

举例说明

在Python的paho-mqtt库中,我们可以通过设置qos参数来指定QoS等级:

import paho.mqtt.client as mqtt

client = mqtt.Client()
client.connect('localhost', 1883, 60)

# 发布消息,QoS等级为1
client.publish('test/topic', 'Hello, World!', qos=1)

client.disconnect()

5. 项目实战:代码实际案例和详细解释说明

5.1 开发环境搭建

安装Go语言

首先,我们需要安装Go语言开发环境。可以从Go官方网站(https://golang.org/dl/)下载适合你操作系统的安装包,然后按照安装向导进行安装。

安装数据库

我们选择使用MySQL作为数据库。可以从MySQL官方网站(https://dev.mysql.com/downloads/installer/)下载MySQL安装包,然后按照安装向导进行安装。

安装MQTT服务器

我们可以使用Mosquitto作为MQTT服务器。可以从Mosquitto官方网站(https://mosquitto.org/download/)下载适合你操作系统的安装包,然后按照安装向导进行安装。

安装依赖库

在Go项目中,我们可以使用go get命令来安装依赖库。以下是一些常用的依赖库:

  • github.com/gorilla/mux:用于处理HTTP路由。
  • github.com/go-sql-driver/mysql:用于连接MySQL数据库。
  • github.com/eclipse/paho.mqtt.golang:用于连接MQTT服务器。

可以使用以下命令来安装这些依赖库:

go get -u github.com/gorilla/mux
go get -u github.com/go-sql-driver/mysql
go get -u github.com/eclipse/paho.mqtt.golang

5.2 源代码详细实现和代码解读

项目结构
smart_home_backend/
├── main.go
├── config/
│   └── config.go
├── models/
│   ├── device.go
│   └── user.go
├── handlers/
│   ├── device_handler.go
│   └── user_handler.go
├── mqtt/
│   └── mqtt_client.go
└── database/
    └── db.go
main.go
package main

import (
    "log"
    "net/http"

    "smart_home_backend/handlers"
    "smart_home_backend/mqtt"
    "smart_home_backend/database"
    "github.com/gorilla/mux"
)

func main() {
    // 初始化数据库
    database.InitDB()

    // 初始化MQTT客户端
    mqtt.InitMQTTClient()

    // 创建路由器
    r := mux.NewRouter()

    // 注册路由
    handlers.RegisterRoutes(r)

    // 启动服务器
    log.Println("Starting server on port 8080...")
    log.Fatal(http.ListenAndServe(":8080", r))
}

代码解读

  • 初始化数据库和MQTT客户端。
  • 创建一个mux路由器,并注册路由。
  • 启动HTTP服务器,监听端口8080。
config/config.go
package config

import (
    "os"
    "fmt"
)

// GetDBConfig 获取数据库配置
func GetDBConfig() string {
    username := os.Getenv("DB_USERNAME")
    password := os.Getenv("DB_PASSWORD")
    host := os.Getenv("DB_HOST")
    port := os.Getenv("DB_PORT")
    dbname := os.Getenv("DB_NAME")

    return fmt.Sprintf("%s:%s@tcp(%s:%s)/%s", username, password, host, port, dbname)
}

// GetMQTTConfig 获取MQTT配置
func GetMQTTConfig() (string, string) {
    broker := os.Getenv("MQTT_BROKER")
    clientID := os.Getenv("MQTT_CLIENT_ID")
    return broker, clientID
}

代码解读

  • 从环境变量中获取数据库和MQTT的配置信息。
models/device.go
package models

import (
    "database/sql"
)

// Device 设备模型
type Device struct {
    ID       int    `json:"id"`
    Name     string `json:"name"`
    Status   string `json:"status"`
}

// GetAllDevices 获取所有设备
func GetAllDevices(db *sql.DB) ([]Device, error) {
    rows, err := db.Query("SELECT id, name, status FROM devices")
    if err != nil {
        return nil, err
    }
    defer rows.Close()

    var devices []Device
    for rows.Next() {
        var device Device
        err := rows.Scan(&device.ID, &device.Name, &device.Status)
        if err != nil {
            return nil, err
        }
        devices = append(devices, device)
    }

    return devices, nil
}

代码解读

  • 定义了Device结构体,表示设备模型。
  • 实现了GetAllDevices函数,用于从数据库中获取所有设备信息。
handlers/device_handler.go
package handlers

import (
    "encoding/json"
    "net/http"

    "smart_home_backend/models"
    "smart_home_backend/database"
)

// GetAllDevicesHandler 获取所有设备的处理函数
func GetAllDevicesHandler(w http.ResponseWriter, r *http.Request) {
    db := database.GetDB()
    devices, err := models.GetAllDevices(db)
    if err != nil {
        http.Error(w, err.Error(), http.StatusInternalServerError)
        return
    }

    w.Header().Set("Content-Type", "application/json")
    json.NewEncoder(w).Encode(devices)
}

代码解读

  • 实现了GetAllDevicesHandler函数,用于处理获取所有设备的HTTP请求。
  • 从数据库中获取所有设备信息,并将其以JSON格式返回给客户端。
mqtt/mqtt_client.go
package mqtt

import (
    "log"

    mqtt "github.com/eclipse/paho.mqtt.golang"
    "smart_home_backend/config"
)

var mqttClient mqtt.Client

// InitMQTTClient 初始化MQTT客户端
func InitMQTTClient() {
    broker, clientID := config.GetMQTTConfig()
    opts := mqtt.NewClientOptions()
    opts.AddBroker(broker)
    opts.SetClientID(clientID)

    mqttClient = mqtt.NewClient(opts)
    if token := mqttClient.Connect(); token.Wait() && token.Error() != nil {
        log.Fatalf("Failed to connect to MQTT broker: %v", token.Error())
    }

    log.Println("Connected to MQTT broker")

    // 订阅设备状态主题
    if token := mqttClient.Subscribe("smart_home/devices/status", 1, messageHandler); token.Wait() && token.Error() != nil {
        log.Fatalf("Failed to subscribe to MQTT topic: %v", token.Error())
    }
}

// messageHandler 处理接收到的MQTT消息
var messageHandler mqtt.MessageHandler = func(client mqtt.Client, msg mqtt.Message) {
    log.Printf("Received message on topic '%s': %s", msg.Topic(), string(msg.Payload()))
    // 这里可以更新数据库中的设备状态
}

代码解读

  • 初始化MQTT客户端,并连接到MQTT服务器。
  • 订阅设备状态主题,并定义消息处理函数。
database/db.go
package database

import (
    "database/sql"
    "log"

    _ "github.com/go-sql-driver/mysql"
    "smart_home_backend/config"
)

var db *sql.DB

// InitDB 初始化数据库连接
func InitDB() {
    dsn := config.GetDBConfig()
    var err error
    db, err = sql.Open("mysql", dsn)
    if err != nil {
        log.Fatalf("Failed to connect to database: %v", err)
    }

    if err = db.Ping(); err != nil {
        log.Fatalf("Failed to ping database: %v", err)
    }

    log.Println("Connected to database")
}

// GetDB 获取数据库连接
func GetDB() *sql.DB {
    return db
}

代码解读

  • 初始化数据库连接,并测试连接是否成功。
  • 提供GetDB函数,用于获取数据库连接。

5.3 代码解读与分析

整体架构分析

本项目采用了分层架构,将不同的功能模块分离,提高了代码的可维护性和可扩展性。主要的分层包括:

  • 配置层:负责读取和管理配置信息。
  • 模型层:定义数据模型和数据库操作方法。
  • 处理层:处理HTTP请求,调用模型层的方法,并返回响应。
  • MQTT层:负责与MQTT服务器进行通信,处理设备状态消息。
  • 数据库层:负责与数据库进行连接和操作。
代码复用和可扩展性

通过将不同的功能模块分离,代码的复用性得到了提高。例如,database包中的GetDB函数可以在多个处理函数中复用。同时,当需要添加新的功能时,只需要在相应的层中添加新的代码即可,不会影响到其他层的代码。

错误处理

在代码中,我们对可能出现的错误进行了处理。例如,在数据库查询和MQTT连接过程中,如果出现错误,会返回相应的错误信息给客户端或记录日志。这样可以提高系统的健壮性。

6. 实际应用场景

家庭自动化控制

用户可以通过手机APP或语音助手远程控制家中的智能设备,如灯光、空调、窗帘等。例如,用户在下班回家的路上,可以提前打开家中的空调,调节到合适的温度。

能源管理

智能家居系统可以实时监测家中的能源消耗情况,如电量、水量等。通过分析能源消耗数据,系统可以提供节能建议,帮助用户降低能源成本。例如,系统可以根据用户的使用习惯,自动调整设备的运行模式,实现节能目的。

安全监控

智能家居系统可以集成摄像头、门磁传感器、烟雾报警器等安全设备,实现家庭安全监控。当检测到异常情况时,系统会及时向用户发送警报信息。例如,当门磁传感器检测到门被打开时,系统会向用户的手机发送通知。

健康监测

智能家居系统可以集成健康监测设备,如智能手环、体脂秤等,实时监测用户的健康数据。系统可以对健康数据进行分析,并提供健康建议。例如,当用户的心率异常时,系统会提醒用户及时就医。

7. 工具和资源推荐

7.1 学习资源推荐

7.1.1 书籍推荐
  • 《Go语言实战》:介绍了Go语言的基础知识和高级特性,通过实际案例帮助读者快速掌握Go语言编程。
  • 《物联网实战:从硬件到云端》:涵盖了物联网的各个方面,包括传感器、通信协议、云计算等,对智能家居系统的开发有很大的帮助。
  • 《RESTful API 设计模式》:详细介绍了RESTful API的设计原则和最佳实践,是开发RESTful API的必备书籍。
7.1.2 在线课程
  • Coursera上的“Go Programming Specialization”:由知名大学教授授课,系统地介绍了Go语言的编程知识。
  • Udemy上的“Building a Smart Home with Raspberry Pi and Home Assistant”:教你如何使用树莓派和Home Assistant搭建智能家居系统。
  • edX上的“Internet of Things (IoT) Foundations”:介绍了物联网的基本概念和技术,为智能家居系统的开发打下基础。
7.1.3 技术博客和网站
  • Go官方博客(https://blog.golang.org/):发布Go语言的最新消息和技术文章。
  • MQTT官方网站(https://mqtt.org/):提供MQTT协议的详细文档和教程。
  • Hackster.io(https://www.hackster.io/):分享物联网和智能家居项目的创意和经验。

7.2 开发工具框架推荐

7.2.1 IDE和编辑器
  • GoLand:JetBrains开发的专门用于Go语言开发的IDE,提供了丰富的代码提示和调试功能。
  • Visual Studio Code:一款轻量级的代码编辑器,支持Go语言开发,通过安装相关插件可以提高开发效率。
7.2.2 调试和性能分析工具
  • Delve:Go语言的调试器,可以帮助开发者快速定位和解决代码中的问题。
  • pprof:Go语言的性能分析工具,可以分析程序的CPU和内存使用情况,找出性能瓶颈。
7.2.3 相关框架和库
  • Gin:一个轻量级的Go语言Web框架,具有高性能和简单易用的特点。
  • Beego:一个功能强大的Go语言Web框架,提供了丰富的插件和工具,适合开发大型项目。
  • Echo:一个高性能的Go语言Web框架,专注于速度和易用性。

7.3 相关论文著作推荐

7.3.1 经典论文
  • “RESTful Web Services”:介绍了RESTful API的设计原则和方法,是RESTful API领域的经典论文。
  • “MQTT - A publish/subscribe protocol for constrained devices and low-bandwidth, high-latency or unreliable networks”:详细介绍了MQTT协议的设计和实现。
7.3.2 最新研究成果
  • 在IEEE Xplore、ACM Digital Library等学术数据库中,可以搜索到关于智能家居系统和物联网的最新研究成果。
7.3.3 应用案例分析
  • 《智能家居系统应用案例集》:收集了各种智能家居系统的应用案例,包括系统架构、功能实现和用户体验等方面的分析。

8. 总结:未来发展趋势与挑战

未来发展趋势

人工智能与机器学习的融合

未来的智能家居系统将更多地融合人工智能和机器学习技术,实现更加智能化的控制和决策。例如,系统可以通过学习用户的行为习惯,自动调整设备的运行模式,提供个性化的服务。

设备互联互通

随着物联网技术的发展,智能家居设备之间的互联互通将更加便捷。不同品牌、不同类型的设备可以实现无缝连接和协同工作,为用户提供更加统一的智能家居体验。

安全性和隐私保护

随着智能家居系统的普及,安全性和隐私保护将成为重要的关注点。未来的智能家居系统将采用更加先进的安全技术,如加密通信、身份认证等,保障用户的信息安全和隐私。

挑战

标准不统一

目前,智能家居行业缺乏统一的标准,不同品牌的设备之间难以实现互联互通。这给用户的使用和系统的集成带来了很大的困难。

网络可靠性

智能家居系统依赖于网络连接,网络可靠性是影响系统性能的重要因素。在一些网络不稳定的地区,设备的控制和数据传输可能会受到影响。

数据安全和隐私问题

智能家居系统会收集大量的用户数据,如设备使用记录、健康数据等。如何保障这些数据的安全和隐私,防止数据泄露和滥用,是智能家居系统面临的重要挑战。

9. 附录:常见问题与解答

1. 如何解决设备之间的兼容性问题?

可以选择支持通用协议(如MQTT、ZigBee等)的设备,或者使用智能家居平台来实现设备的集成和管理。智能家居平台可以提供统一的接口,将不同品牌、不同类型的设备连接在一起。

2. 如何保障智能家居系统的安全性?

可以采取以下措施来保障智能家居系统的安全性:

  • 使用强密码和定期更换密码。
  • 对设备进行定期更新和维护,及时修复安全漏洞。
  • 采用加密通信技术,保障数据传输的安全。
  • 对用户进行身份认证和授权,防止非法访问。

3. 如何处理大量的设备数据?

可以使用数据库管理系统(如MySQL、MongoDB等)来存储和管理设备数据。同时,可以采用数据缓存和压缩技术,减少数据存储和传输的成本。另外,可以使用数据分析工具(如Python的Pandas、NumPy等)对设备数据进行分析和挖掘,提取有价值的信息。

4. 如何实现设备的远程控制?

可以通过开发RESTful API来实现设备的远程控制。前端应用可以通过HTTP请求调用这些API,向后端发送控制指令。后端接收到指令后,通过MQTT协议将指令发送给相应的设备。

10. 扩展阅读 & 参考资料

扩展阅读

  • 《智能家居系统设计与实现》
  • 《物联网安全技术》
  • 《Go语言高级编程》

参考资料

  • Go官方文档(https://golang.org/doc/)
  • MQTT官方文档(https://mqtt.org/documentation/)
  • MySQL官方文档(https://dev.mysql.com/doc/)
  • Mosquitto官方文档(https://mosquitto.org/documentation/)
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值