一 概述
HTTP是一個屬于應用層的面向對象的協議,由于其簡捷、快速的方式,適用于分布式超媒體信息系統。它于1990年提出,經過幾年的使用與發展,得到不斷地完善和擴展。目前在WWW中使用的是HTTP/1.0的第六版,HTTP/1.1的規范化工作正在進行之中,而且HTTP-NG(Next Generation of HTTP)的建議已經提出。
HTTP協議的主要特點可概括如下:
支持客戶/服務器模式。
簡單快速:客戶向服務器請求服務時,只需傳送請求方法和路徑。請求方法常用的有GET、HEAD、POST。每種方法規定了客戶與服務器聯系的類型不同。由于HTTP協議簡單,使得HTTP服務器的程序規模小,因而通信速度很快。
靈活:HTTP允許傳輸任意類型的數據對象。正在傳輸的類型由Content-Type加以標記。
無連接:無連接的含義是限制每次連接只處理一個請求。服務器處理完客戶的請求,并收到客戶的應答后,即斷開連接。采用這種方式可以節省傳輸時間。
無狀態:HTTP協議是無狀態協議。無狀態是指協議對于事務處理沒有記憶能力。缺少狀態意味著如果后續處理需要前面的信息,則它必須重傳,這樣可能導致每次連接傳送的數據量增大。另一方面,在服務器不需要先前信息時它的應答就較快。
二 HTTP的URL
HTTP(超文本傳輸協議)是一個基于請求與響應模式的、無狀態的、應用層的協議,常基于TCP的連接方式,HTTP1.1版本中給出一種持續連接的機制,絕大多數的Web開發,都是構建在HTTP協議之上的Web應用。
HTTP URL (URL是一種特殊類型的URI,包含了用于查找某個資源的足夠的信息)的格式如下: http://host[":"port][abs_path] http表示要通過HTTP協議來定位網絡資源;host表示合法的Internet主機域名或者IP地址;port指定一個端口號,為空則使用缺省端口80;abs_path指定請求資源的URI;如果URL中沒有給出abs_path,那么當它作為請求URI時,必須以“/”的形式給出,通常這個工作瀏覽器自動幫我們完成。 eg: 1、輸入:www.guet.edu.cn 瀏覽器自動轉換成:http://www.guet.edu.cn/ 2、http:192.168.0.116:8080/index.jsp
二 HTTP的請求
HTTP請求由三部分組成,分別是:請求行、消息報頭、請求正文
1.請求行以一個方法符號開頭,以空格分開,后面跟著請求的URI和協議的版本,格式如下:Method Request-URI HTTP-Version CRLF
其中 Method表示請求方法;Request-URI是一個統一資源標識符;HTTP-Version表示請求的HTTP協議版本;CRLF表示回車和換行(除了作為結尾的CRLF外,不允許出現單獨的CR或LF字符)。
請求方法(所有方法全為大寫)有多種,各個方法的解釋如下:
GET 請求獲取Request-URI所標識的資源
POST 在Request-URI所標識的資源后附加新的數據
HEAD 請求獲取由Request-URI所標識的資源的響應消息報頭
PUT 請求服務器存儲一個資源,并用Request-URI作為其標識
DELETE 請求服務器刪除Request-URI所標識的資源
TRACE 請求服務器回送收到的請求信息,主要用于測試或診斷
CONNECT 保留將來使用
OPTIONS 請求查詢服務器的性能,或者查詢與資源相關的選項和需求
應用舉例:
GET方法:在瀏覽器的地址欄中輸入網址的方式訪問網頁時,瀏覽器采用GET方法向服務器獲取資源,eg:GET /form.html HTTP/1.1 (CRLF)
POST方法要求被請求服務器接受附在請求后面的數據,常用于提交表單。
eg:POST /reg.jsp HTTP/ (CRLF)
Accept:image/gif,image/x-xbit,... (CRLF)
...
HOST:www.guet.edu.cn (CRLF)
Content-Length:22 (CRLF)
Connection:Keep-Alive (CRLF)
Cache-Control:no-cache (CRLF)
(CRLF) //該CRLF表示消息報頭已經結束,在此之前為消息報頭
user=jeffrey&pwd=1234 //此行以下為提交的數據
HEAD方法與GET方法幾乎是一樣的,對于HEAD請求的回應部分來說,它的HTTP頭部中包含的信息與通過GET請求所得到的信息是相同的。利用這個方法,不必傳輸整個資源內容,就可以得到Request-URI所標識的資源的信息。該方法常用于測試超鏈接的有效性,是否可以訪問,以及最近是否更新。
2、請求報頭后述
3、請求正文(略)
三 HTTP響應
在接收和解釋請求消息后,服務器返回一個HTTP響應消息。
HTTP響應也是由三個部分組成,分別是:狀態行、消息報頭、響應正文
1、狀態行格式如下:
HTTP-Version Status-Code Reason-Phrase CRLF
其中,HTTP-Version表示服務器HTTP協議的版本;Status-Code表示服務器發回的響應狀態代碼;Reason-Phrase表示狀態代碼的文本描述。
狀態代碼有三位數字組成,第一個數字定義了響應的類別,且有五種可能取值:
1xx:指示信息--表示請求已接收,繼續處理
2xx:成功--表示請求已被成功接收、理解、接受
3xx:重定向--要完成請求必須進行更進一步的操作
4xx:客戶端錯誤--請求有語法錯誤或請求無法實現
5xx:服務器端錯誤--服務器未能實現合法的請求
常見狀態代碼、狀態描述、說明:
200 OK //客戶端請求成功
400 Bad Request //客戶端請求有語法錯誤,不能被服務器所理解
401 Unauthorized //請求未經授權,這個狀態代碼必須和WWW-Authenticate報頭域一起使用
403 Forbidden //服務器收到請求,但是拒絕提供服務
404 Not Found //請求資源不存在,eg:輸入了錯誤的URL
500 Internal Server Error //服務器發生不可預期的錯誤
503 Server Unavailable //服務器當前不能處理客戶端的請求,一段時間后可能恢復正常
eg:HTTP/1.1 200 OK (CRLF)
2、響應報頭后述
3、響應正文就是服務器返回的資源的內容
四 HTTP組件的使用
1 Gitee鏈接地址
組件位于amaziot_bloom_os_sdklibrariesamxtuam_http.c
Gitee源碼地址:https://gitee.com/ning./hongdou
Github源碼地址:https://github.com/ayumid/hongdou
2 應用層組件功能介紹
提供HTTP連接實例,可以通過調用組件內的API,來實現HTTP本地客戶端和服務器之間的通信。
3 代碼講解
1 dtu_http_response_cb
http請求回調函數
static int dtu_http_response_cb(char *buffer, int size, int nitems, void *private_data) { struct http_data_s *client_data = private_data; if ((client_data->data_sz + size) < sizeof(client_data-?>data)) { memcpy(client_data->data + client_data->data_sz, buffer, size); client_data->data_sz += size; return 0; } return -1; }
2 dtu_http_pg_data_get
get請求實現
static void dtu_http_pg_data_get(DTU_MSG_UART_DATA_PARAM_T* data, UINT8 type, UINT8 channel) { struct http_client *client = NULL; struct http_data_s *client_data = NULL; struct http_client_list * header = NULL; DTU_FILE_PARAM_T* dtu_file_ctx = NULL; char url[DTU_HTTP_S_URL_LEN] = {0}; char head_tmp[DTU_HTTP_HEAD_MAX_LEN] = {0}; UINT8* d = NULL; int response_code = 0; int i = 0; dtu_file_ctx = dtu_get_file_ctx(); response_code = 0; d = data->UArgs; client_data = malloc(sizeof(*client_data)); if (!client_data){ return ; } memset(client_data, 0, sizeof(*client_data)); client = http_client_init(); if (!client){ free(client_data); client_data = NULL; return ; } d[data->len] = ''; if(1 == channel) { if(NULL == memchr(dtu_file_ctx->http.http1.url, '?', strlen(dtu_file_ctx->http.http1.url))) { snprintf(url, DTU_HTTP_S_URL_LEN, "%s?data=%s", dtu_file_ctx->http.http1.url, (char*)data->UArgs); } else { snprintf(url, DTU_HTTP_S_URL_LEN, "%s&data=%s", dtu_file_ctx->http.http1.url, (char*)data->UArgs); } http_client_setopt(client, HTTPCLIENT_OPT_URL, url); } else if(2 == channel) { if(NULL == memchr(dtu_file_ctx->http.http1.url, '?', strlen(dtu_file_ctx->http.http1.url))) { snprintf(url, DTU_HTTP_S_URL_LEN, "%s?data=%s", dtu_file_ctx->http.http2.url, (char*)data->UArgs); } else { snprintf(url, DTU_HTTP_S_URL_LEN, "%s&data=%s", dtu_file_ctx->http.http2.url, (char*)data->UArgs); } http_client_setopt(client, HTTPCLIENT_OPT_URL, url); } uprintf("http get url: %s", url); http_client_setopt(client, HTTPCLIENT_OPT_RESPONSECB, dtu_http_response_cb); http_client_setopt(client, HTTPCLIENT_OPT_RESPONSECB_DATA, client_data); http_client_setopt(client, HTTPCLIENT_OPT_METHOD, HTTPCLIENT_REQUEST_GET); // Add private HTTP header if(1 == channel) { if(strlen(dtu_file_ctx->http.http1.head1)) { snprintf(head_tmp, DTU_HTTP_HEAD_MAX_LEN, "%srn", dtu_file_ctx->http.http1.head1); header = http_client_list_append(header, head_tmp); uprintf("head11: %s", head_tmp); } if(strlen(dtu_file_ctx->http.http1.head2)) { snprintf(head_tmp, DTU_HTTP_HEAD_MAX_LEN, "%srn", dtu_file_ctx->http.http1.head2); header = http_client_list_append(header, head_tmp); uprintf("head12: %s", head_tmp); } if(strlen(dtu_file_ctx->http.http1.head3)) { snprintf(head_tmp, DTU_HTTP_HEAD_MAX_LEN, "%srn", dtu_file_ctx->http.http1.head3); header = http_client_list_append(header, head_tmp); uprintf("head13: %s", head_tmp); } } else if(2 == channel) { if(strlen(dtu_file_ctx->http.http2.head1)) { snprintf(head_tmp, DTU_HTTP_HEAD_MAX_LEN, "%srn", dtu_file_ctx->http.http2.head1); header = http_client_list_append(header, head_tmp); uprintf("head21: %s", head_tmp); } if(strlen(dtu_file_ctx->http.http2.head2)) { snprintf(head_tmp, DTU_HTTP_HEAD_MAX_LEN, "%srn", dtu_file_ctx->http.http2.head2); header = http_client_list_append(header, head_tmp); uprintf("head22: %s", head_tmp); } if(strlen(dtu_file_ctx->http.http2.head3)) { snprintf(head_tmp, DTU_HTTP_HEAD_MAX_LEN, "%srn", dtu_file_ctx->http.http2.head3); header = http_client_list_append(header, head_tmp); uprintf("head23: %s", head_tmp); } } http_client_setopt(client, HTTPCLIENT_OPT_HTTPHEADER, header); http_client_perform(client); http_client_getinfo(client, HTTPCLIENT_GETINFO_RESPONSE_CODE, &response_code); uprintf("[http get]Get tcp state %dn", response_code); if (response_code >= 200 && response_code < 300) { if(client_data-?>data_sz) { uprintf("rn data_sz=%u, %s", client_data->data_sz,client_data->data); for (i = 0; i < 100; i++) { //uprintf("%02x ",client_data-?>data[i]); } } }else if (response_code == 404) { uprintf("response_code == %drn%s",response_code ,client_data->data); } if (client_data) { free(client_data); client_data = NULL; } if (client) { http_client_shutdown(client); client = NULL; } }
3 dtu_http_pg_data_post
post請求實現
static void dtu_http_pg_data_post(DTU_MSG_UART_DATA_PARAM_T* data, UINT8 type, UINT8 channel) { struct http_client *client = NULL; struct http_data_s *client_data = NULL; struct http_client_list * header = NULL; DTU_FILE_PARAM_T* dtu_file_ctx = NULL; char head_tmp[DTU_HTTP_HEAD_MAX_LEN] = {0}; UINT8* d = NULL; int response_code = 0; int i = 0; dtu_file_ctx = dtu_get_file_ctx(); response_code = 0; d = data->UArgs; client_data = malloc(sizeof(*client_data)); if (!client_data){ return ; } memset(client_data, 0, sizeof(*client_data)); client = http_client_init(); if (!client){ free(client_data); client_data = NULL; return ; } if(1 == channel) { http_client_setopt(client, HTTPCLIENT_OPT_URL, dtu_file_ctx->http.http1.url); } else if(2 == channel) { http_client_setopt(client, HTTPCLIENT_OPT_URL, dtu_file_ctx->http.http2.url); } http_client_setopt(client, HTTPCLIENT_OPT_RESPONSECB, dtu_http_response_cb); http_client_setopt(client, HTTPCLIENT_OPT_RESPONSECB_DATA, client_data); http_client_setopt(client, HTTPCLIENT_OPT_METHOD, HTTPCLIENT_REQUEST_POST); // // Add private HTTP header if(1 == channel) { if(strlen(dtu_file_ctx->http.http1.head1)) { snprintf(head_tmp, DTU_HTTP_HEAD_MAX_LEN, "%srn", dtu_file_ctx->http.http1.head1); header = http_client_list_append(header, head_tmp); uprintf("head11: %s", head_tmp); } if(strlen(dtu_file_ctx->http.http1.head2)) { snprintf(head_tmp, DTU_HTTP_HEAD_MAX_LEN, "%srn", dtu_file_ctx->http.http1.head2); header = http_client_list_append(header, head_tmp); uprintf("head12: %s", head_tmp); } if(strlen(dtu_file_ctx->http.http1.head3)) { snprintf(head_tmp, DTU_HTTP_HEAD_MAX_LEN, "%srn", dtu_file_ctx->http.http1.head3); header = http_client_list_append(header, head_tmp); uprintf("head13: %s", head_tmp); } } else if(2 == channel) { if(strlen(dtu_file_ctx->http.http2.head1)) { snprintf(head_tmp, DTU_HTTP_HEAD_MAX_LEN, "%srn", dtu_file_ctx->http.http2.head1); header = http_client_list_append(header, head_tmp); uprintf("head21: %s", head_tmp); } if(strlen(dtu_file_ctx->http.http2.head2)) { snprintf(head_tmp, DTU_HTTP_HEAD_MAX_LEN, "%srn", dtu_file_ctx->http.http2.head2); header = http_client_list_append(header, head_tmp); uprintf("head22: %s", head_tmp); } if(strlen(dtu_file_ctx->http.http2.head3)) { snprintf(head_tmp, DTU_HTTP_HEAD_MAX_LEN, "%srn", dtu_file_ctx->http.http2.head3); header = http_client_list_append(header, head_tmp); uprintf("head23: %s", head_tmp); } } d[data->len] = ''; http_client_setopt(client, HTTPCLIENT_OPT_HTTPHEADER, header); http_client_setopt(client, HTTPCLIENT_OPT_POSTDATA, data->UArgs); /*post data is http context*/ http_client_setopt(client, HTTPCLIENT_OPT_POSTLENGTH, strlen(data->UArgs)); /*http context length*/ http_client_perform(client); http_client_getinfo(client, HTTPCLIENT_GETINFO_RESPONSE_CODE, &response_code); uprintf("[http post]Get tcp state %dn", response_code); if (response_code >= 200 && response_code < 300) { if(client_data-?>data_sz) { uprintf("rn data_sz=%u, %s", client_data->data_sz,client_data->data); for (i = 0; i < 100; i++) { //uprintf("%02x ",client_data-?>data[i]); } } }else if (response_code == 404) { uprintf("response_code == %drn%s",response_code ,client_data->data); } if (client_data) { free(client_data); client_data = NULL; } if (client) { http_client_shutdown(client); client = NULL; } }
4 Demo實戰
參考進階實戰
本文章源自奇跡物聯開源的物聯網應用知識庫Cellular IoT Wiki,更多技術干貨歡迎關注收藏Wiki:Cellular IoT Wiki 知識庫(https://rckrv97mzx.feishu.cn/wiki/wikcnBvAC9WOkEYG5CLqGwm6PHf)
歡迎同學們走進AmazIOT知識庫的世界!
這里是為物聯網人構建的技術應用百科,以便幫助你更快更簡單的開發物聯網產品。
Cellular IoT Wiki初心:
在我們長期投身于蜂窩物聯網 ODM/OEM 解決方案的實踐過程中,一直被物聯網技術碎片化與產業資源碎片化的問題所困擾。從產品定義、芯片選型,到軟硬件研發和測試,物聯網技術的碎片化以及產業資源的碎片化,始終對團隊的產品開發交付質量和效率形成制約。為了減少因物聯網碎片化而帶來的重復開發工作,我們著手對物聯網開發中高頻應用的技術知識進行沉淀管理,并基于 Bloom OS 搭建了不同平臺的 RTOS 應用生態。后來我們發現,很多物聯網產品開發團隊都面臨著相似的困擾,于是,我們決定向全體物聯網行業開發者開放奇跡物聯內部沉淀的應用技術知識庫 Wiki,期望能為更多物聯網產品開發者減輕一些重復造輪子的負擔。
Cellular IoT Wiki沉淀的技術內容方向如下:
奇跡物聯的業務服務范圍:基于自研的NB-IoT、Cat1、Cat4等物聯網模組,為客戶物聯網ODM/OEM解決方案服務。我們的研發技術中心在石家莊,PCBA生產基地分布在深圳、石家莊、北京三個工廠,滿足不同區域&不同量產規模&不同產品開發階段的生產制造任務。跟傳統PCBA工廠最大的區別是我們只服務物聯網行業客戶。
連接我們,和10000+物聯網開發者一起 降低技術和成本門檻
讓蜂窩物聯網應用更簡單~~
哈哈你終于滑到最重要的模塊了,
千萬不!要!劃!走!忍住沖動!~
歡迎加入飛書“開源技術交流群”,隨時找到我們哦~
點擊鏈接如何加入奇跡物聯技術話題群(https://rckrv97mzx.feishu.cn/docx/Xskpd1cFQo7hu9x5EuicbsjTnTf)可以獲取加入技術話題群攻略
Hey 物聯網從業者,
你是否有了解過奇跡物聯的官方公眾號“eSIM物聯工場”呢?
這里是奇跡物聯的物聯網應用技術開源wiki主陣地,歡迎關注公眾號,不迷路~
及時獲得最新物聯網應用技術沉淀發布
(如有侵權,聯系刪除)
審核編輯 黃宇
-
物聯網
+關注
關注
2903文章
44275瀏覽量
371249 -
HTTP
+關注
關注
0文章
501瀏覽量
31063
發布評論請先 登錄
相關推薦
評論