發(fā)送 GET 和 POST 請求(中)
GET 請求的響應(yīng)處理是確保設(shè)備獲取有效數(shù)據(jù)的關(guān)鍵,嵌入式 HTTP Client 需通過 “分階段解析” 應(yīng)對響應(yīng)數(shù)據(jù)流,同時(shí)規(guī)避內(nèi)存不足風(fēng)險(xiǎn)。當(dāng) ENC28J60 接收到服務(wù)器返回的 TCP 數(shù)據(jù)后,會觸發(fā) INT 引腳中斷,MCU 通過 SPI 讀取模塊接收緩沖區(qū)(默認(rèn) 6KB)中的響應(yīng)數(shù)據(jù),首先解析響應(yīng)行中的狀態(tài)碼:若為 200 OK,說明請求成功,需繼續(xù)解析響應(yīng)頭與響應(yīng)體;若為 304 Not Modified,表明資源未更新,直接關(guān)閉連接以節(jié)省功耗;若為 400 Bad Request 或 404 Not Found,需記錄錯(cuò)誤日志并觸發(fā)重試(通常重試 2 次,間隔 3 秒);若為 5xx 服務(wù)器錯(cuò)誤,則需延長重試間隔(如 10 秒),避免頻繁請求加重服務(wù)器負(fù)擔(dān)。解析響應(yīng)頭時(shí),重點(diǎn)提取Content-Length(如Content-Length: 128)以確定響應(yīng)體大小,Content-Type(如Content-Type: application/json)以選擇對應(yīng)解析方式,例如 JSON 格式的 OTA 版本響應(yīng)體{"has_update":true,"new_version":"v1.1","firmware_url":"http://ota-server.com/fw/v1.1.bin"},MCU 可通過輕量化 JSON 解析庫(如 cJSON 的微型版本)提取has_update與firmware_url字段,無需加載完整庫;若響應(yīng)體體積較大(如 1KB 的配置文件),則采用 “分塊讀取 - 即時(shí)解析” 模式,每讀取 256 字節(jié)就解析一次,避免將完整響應(yīng)體存入 RAM 導(dǎo)致溢出。
POST 請求作為 HTTP Client 中核心的資源提交方法,其核心定位是 “向服務(wù)器推送數(shù)據(jù)”,在物聯(lián)網(wǎng)場景中常用于傳感器數(shù)據(jù)上報(bào)、設(shè)備狀態(tài)反饋、OTA 升級結(jié)果提交等,其關(guān)鍵特征是請求參數(shù)通過請求體傳遞(而非 URL),支持大體積數(shù)據(jù)傳輸,且對資源的操作不具備冪等性(多次請求可能導(dǎo)致服務(wù)器資源重復(fù)創(chuàng)建),這一特性使其在需提交動(dòng)態(tài)數(shù)據(jù)的場景中更具靈活性。與 GET 請求相比,POST 請求的構(gòu)建更復(fù)雜,需同時(shí)處理 “請求頭” 與 “請求體”,且需通過Content-Type字段明確請求體格式,以確保服務(wù)器能正確解析 —— 在嵌入式設(shè)備中,POST 請求的設(shè)計(jì)需重點(diǎn)解決 “內(nèi)存占用” 與 “數(shù)據(jù)完整性” 的平衡,例如 STM32F103(RAM 僅 20KB)在上報(bào) 10KB 傳感器日志時(shí),需通過固定大小緩沖區(qū)(如 1024 字節(jié))分塊構(gòu)建請求體,避免動(dòng)態(tài)內(nèi)存分配導(dǎo)致的碎片問題。
POST 請求的頭部構(gòu)建需額外關(guān)注與請求體相關(guān)的字段,以確保數(shù)據(jù)傳輸?shù)恼_性與效率。除Host、User-Agent、Connection等基礎(chǔ)字段外,Content-Type是 POST 請求的核心必選字段,需根據(jù)請求體格式選擇對應(yīng)值:若請求體為 JSON 格式(如傳感器結(jié)構(gòu)化數(shù)據(jù){"temp":25.3,"humidity":60.2,"timestamp":1731800000}),則Content-Type設(shè)為application/json;若為表單數(shù)據(jù)(如設(shè)備注冊信息device_id=sn_123&key=abc123),則設(shè)為application/x-www-form-urlencoded;若需傳輸二進(jìn)制數(shù)據(jù)(如小體積設(shè)備日志文件),則設(shè)為multipart/form-data并指定邊界符(如boundary=----WebKitFormBoundary7MA4YWxkTrZu0gW)。Content-Length字段同樣關(guān)鍵,需準(zhǔn)確填寫請求體的字節(jié)數(shù)(如 JSON 數(shù)據(jù)長度為 58 字節(jié),則Content-Length: 58),服務(wù)器通過該字段判斷請求體是否接收完整,避免數(shù)據(jù)截?cái)?;若請求體體積動(dòng)態(tài)變化(如實(shí)時(shí)采集的傳感器數(shù)據(jù)流),則可采用 “分塊傳輸編碼”(設(shè)置Transfer-Encoding: chunked),無需預(yù)先計(jì)算總長度,每傳輸一塊數(shù)據(jù)就附加該塊長度(十六進(jìn)制),最后以0\r\n\r\n標(biāo)識結(jié)束,這一方式尤其適合 NB-IoT 等低帶寬場景,可減少因數(shù)據(jù)長度計(jì)算錯(cuò)誤導(dǎo)致的傳輸失敗。
POST 請求體的構(gòu)建是嵌入式 HTTP Client 的實(shí)現(xiàn)難點(diǎn),需根據(jù)數(shù)據(jù)類型與設(shè)備資源設(shè)計(jì) “輕量化方案”。對于 JSON 格式的結(jié)構(gòu)化數(shù)據(jù),嵌入式設(shè)備通常采用 “靜態(tài)字符串拼接 + 參數(shù)替換” 的方式構(gòu)建請求體,例如預(yù)先定義 JSON 模板{"temp":%f,"humidity":%f,"timestamp":%d},再通過sprintf函數(shù)將采集到的溫度(25.3)、濕度(60.2)、時(shí)間戳(1731800000)替換占位符,生成完整請求體 —— 需注意的是,sprintf可能導(dǎo)致緩沖區(qū)溢出,因此需使用安全版本(如snprintf)并嚴(yán)格限制緩沖區(qū)大?。ㄈ?span> 256 字節(jié)),確保適配多數(shù)短數(shù)據(jù)上報(bào)場景。對于大體積數(shù)據(jù)(如 5KB 的設(shè)備運(yùn)行日志),則需采用 “分塊寫入” 策略:MCU 將日志數(shù)據(jù)從 Flash 中逐塊讀?。繅K 1024 字節(jié)),通過 HTTP Client 的esp_ota_write類接口寫入請求體緩沖區(qū),每寫滿一塊就觸發(fā)一次傳輸,同時(shí)記錄已傳輸字節(jié)數(shù),避免內(nèi)存堆積;若設(shè)備支持硬件 SPI DMA(如 STM32L4),可將 Flash 讀取與請求體寫入過程并行,進(jìn)一步提升效率。對于二進(jìn)制數(shù)據(jù)(如小圖片、固件片段),需通過 “Base64 編碼” 轉(zhuǎn)換為文本格式嵌入請求體(如{"log_data":"SGVsbG8gV29ybGQh..."}),或使用multipart/form-data格式直接傳輸二進(jìn)制流,前者兼容性更強(qiáng)(多數(shù)服務(wù)器支持 Base64 解碼),后者更節(jié)省帶寬(無需編碼開銷),需根據(jù)服務(wù)器支持能力選擇。





