结尾
学习html5、css、javascript这些基础知识,学习的渠道很多,就不多说了,例如,一些其他的优秀博客。但是本人觉得看书也很必要,可以节省很多时间,常见的javascript的书,例如:javascript的高级程序设计,是每位前端工程师必不可少的一本书,边看边用,了解js的一些基本知识,基本上很全面了,如果有时间可以读一些,js性能相关的书籍,以及设计者模式,在实践中都会用的到。
开源分享:【大厂前端面试题解析+核心总结学习笔记+真实项目实战+最新讲解视频】
高级程序设计,是每位前端工程师必不可少的一本书,边看边用,了解js的一些基本知识,基本上很全面了,如果有时间可以读一些,js性能相关的书籍,以及设计者模式,在实践中都会用的到。
void ec800_init()
{
u16 cat1_timeout = 0;
while(Iot\_SendCmd(AT,"OK", 200)){
HAL\_Delay(1);
cat1_timeout ++;
if(cat1_timeout >= 2000){
printf(" uart false\r\n");
return;
}
}
cat1_timeout = 0;
printf("uart ok\r\n");
while(Iot\_SendCmd(CPIN,"READY", 200)){
HAL\_Delay(1);
cat1_timeout ++;
if(cat1_timeout >= 2000){
return;
}
}
cat1_timeout = 0;
printf("simcard ok\r\n");
while(Iot\_SendCmd(RSSI,"+CSQ", 200)){
HAL\_Delay(1);
cat1_timeout ++;
if(cat1_timeout >= 2000){
return;
}
}
cat1_timeout = 0;
while(Iot\_SendCmd(CEREG,"0,1", 200)){
HAL\_Delay(1);
cat1_timeout ++;
if(cat1_timeout >= 2000){
return;
}
}
printf("网络注册 ok\r\n");
cat1_timeout = 0;
while(Iot\_SendCmd(CGATT,"+CGATT: 1", 200)){
HAL\_Delay(1);
cat1_timeout ++;
if(cat1_timeout >= 2000){
return;
}
}
printf(" 网络附着 ok\r\n");
}
### 2.3 PDP 上下文
我们在看文档的时候,会有一个 PDP 上下文的概念,这里我们需要对这个概念说明一下。
>
> PDP,Packet Data Protocol,分组数据规程,移动通信用户在发送和接收分组数据时应用的协议,应用于 GPRS以及 WCDMA 等分组网络。
> .
> PDP context 即 PDP 上下文,保存用户面进行隧道转发的所有信息,包括 RNC / GGSN 的用户面 IP 地址、隧道标识和 QoS 等。
>
>
>
实际上如果对网络不了解的看完了上面的基本概念,应该还是不理解,所以我这里用白话文简单解释一下:
就是说我们需要用 PDP(Packet Data Protocol )传输数据,那么就必须给他建立一个背景,就是所谓的上下文,你得告诉 SGSN 和 GGSN (什么是 SGSN 和 GGSN ,往下面看,就类似于中转站,网关类) 一些基本信息 :本地标识(cid),你想要的 pdp\_type(这里就是IP),你的接入点信息。
>
> 接入点不同的运行商是不同的:
> 联通:UNINET 移动:CMNET 电信:CTNET
>
>
>
我们不把 PDP 上下文用专业的术语去理解,简单的说,PDP 上下文就是在你准备连接 Internet 传输数据之前,必须要做的一些基本配置,给你发个通讯卡,后面你就可以传输数据了,激活一个 PDP 上下文意味着发起一个分组数据业务呼叫。
#### 2.3.1 什么是 SGSN 和 GGSN ?
在上面介绍 PDP 的时候提到过 SGSN 和 GGSN ,关于这一块,我也是参考前人的博文:
[移动数据通信网络工作原理(SGSN&GGSN)]( )
图片引用至上面推荐博文:
![在这里插入图片描述](https://img-blog.csdnimg.cn/89271832f0e64b95b6f5843a4e6069e1.png)
内容引用至上面推荐博文:
>
> **SGSN**
> SGSN主要用于为在其地理范围内的移动站传递数据包,相当于无线网络中的路由节点。它可以进行分组路由和转发,移动性管理(附着,去附着和位置管理),逻辑链路管理,鉴权以及计费功能。SGSN的位置寄存器保存着位置信息,比如当前的小区。
> SGSN的主要功能包括:
> 1 完成和GGSN的通信,通过GTP协议将用户数据传递给GGSN,并将GGSN返回的数据传递给用户
> 2 当用户地理位置发生变化,执行移动性管理。
> **GGSN**
> GSGN作为整个GPRS/UMTS网络的网关,位于GPRS网络和外部分组交换网络(Internet)之间。网关的作用能将一种协议格式的数据转换为另一种格式的数据。
> GGSN把来自的SGSN的GPRS数据包转化为适当的分组数据协议格式,比如IP,然后再把它们发送到相应的分组数据网络,比如广域有线网。反之亦然。
> **SGSN和GGSN的区别**
> 所以,GGSN和SGSN的主要区别就在于,GGSN作为网关,是在不同的通信网中转换协议,而SGSN作为路由,只是在使用相同协议的网络中发送、接受以及延迟它的数据包。
> 另外,GGSN能够实现地址的转换,比如把无线网络内部地址(PDP地址)转换为一个分组数据网络协议地址(IP地址),而SGSN只能实现PDP地址映射,即根据一个地址,映射到相同种类的另一个地址。可见,我们常说的3、4G网络的IP地址,其实就是对应GGSN的出口IP地址。
>
>
>
到这里,一些开始使用需要了解的基础问题都已经说明了,那么下面其实就可以直接开启 HTTP 的使用了。
## 三、 HTTP 流程
上面准备工作做完了,我们接下来往下面进行。
### 3.1 客户端
客户端实际上就是我们的板子,在上面我们已经给出了 EC800M 上电初始化的流程。
在完成上述的初始化以后,我们就可以按照官方手册进行下去了,本部分我们主要使用图示和实际测试代码给出说明。
我们来回顾一下上面的流程(这里我们用官方文档中的示例图说明):
![在这里插入图片描述](https://img-blog.csdnimg.cn/b7e76074e1d04ec2af71a9580f27d2d5.png)
上面的流程是官方给出的一个简单的样例,大体上,按照顺序来就行了。
其中需要注意的就是 ,有一个参数设置样例中并没有,就是数据类型,数据类型其实是大家需要 POST 的服务器有关的,这个需要自己了解服务器端需要怎样的数据类型:
![在这里插入图片描述](https://img-blog.csdnimg.cn/9bca2cdfcb574017afe58830e4621e7a.png)
比如本次测试,我使用了了 `"AT+QHTTPCFG=\"contenttype\",1\r\n"` 就是设置为 `"text/plain"` 类型。
上面还是用了 `AT+QIACT=1` 激活 PDP上下文,但是图中也提到了是默认激活的,我发现如果是激活状态,使用这个指令会返回 ERROR (有待确定)。
其他的倒是没有什么问题,这里直接上一下代码。
#### 3.1.1 PDP 上下文配置
void ec800_pdp_prepare(){
u16 cat1_timeout = 0;
while(Iot_SendCmd(“AT+QHTTPCFG=“contextid”,1\r\n”,“OK”, 200)){
HAL_Delay(1);
cat1_timeout ++;
if(cat1_timeout >= 2000){
return;
}
}
cat1_timeout = 0;
printf(“设置 QHTTPCFG ok\r\n”);
while(Iot\_SendCmd("AT+QIACT?\r\n","OK", 3000)){
HAL\_Delay(1);
cat1_timeout ++;
if(cat1_timeout >= 2000){
return;
}
}
cat1_timeout = 0;
printf("PDP\_CHECK one ok\r\n");
while(Iot\_SendCmd("AT+QHTTPCFG=\"contenttype\",1\r\n","OK", 3000)){
HAL\_Delay(1);
cat1_timeout ++;
if(cat1_timeout >= 2000){
return;
}
}
cat1_timeout = 0;
printf("CFG ok\r\n");
// while(Iot\_SendCmd("AT+QHTTPCFG=\"rspout/auto\",1\r\n","OK", 3000)){
// HAL\_Delay(1);
// cat1\_timeout ++;
// if(cat1\_timeout >= 2000){
// return;
// }
// }
// cat1\_timeout = 0;
// printf("auto header ok\r\n");
/\*
“AT+QICSGP=1,1,“CMNET”,”“,”“,1\r\n”
APN 联通:UNINET 移动:CMNET 电信:CTNET
*/
while(Iot_SendCmd(“AT+QICSGP=1,1,“CMNET”,”“,”“,1\r\n”,“OK”, 3000)){
HAL_Delay(1);
cat1_timeout ++;
if(cat1_timeout >= 2000){
return;
}
}
cat1_timeout = 0;
printf(“PDP_CONFIG ok\r\n”);
while(Iot_SendCmd(“AT+QIACT?\r\n”,“+QIACT”, 3000)){
HAL_Delay(1);
cat1_timeout ++;
if(cat1_timeout >= 2000){
return;
}
}
cat1_timeout = 0;
printf(“PDP_CHECK two ok\r\n”);
// while(Iot\_SendCmd("AT+QIACT=1\r\n","OK", 500)){
// HAL\_Delay(1);
// cat1\_timeout ++;
// if(cat1\_timeout >= 2000){
// return;
// }
// }
// cat1\_timeout = 0;
// printf("PDP\_激活 ok\r\n");
}
#### 3.1.2 URL 设置
接下来就是设置 URL ,URL 从哪里来,就是服务器会提供,比如 ONENET 平台对于 HTTP 的说明如下:
![在这里插入图片描述](https://img-blog.csdnimg.cn/4de9e5615d4340ab83524d0870de639d.png)
这里我们 POST 设置的 URL ,就是上面的 Address+URL,对应在下面的程序中,就是`char *url` 这个参数:
void http_set_url(char *url)
{
u16 cat1_timeout = 0;
char message[32];
snprintf(message, sizeof(message), “AT+QHTTPURL=%d,%d\r\n”, strlen(url), 5);
while(Iot_SendCmd(message,“CONNECT”, 1000)){
HAL_Delay(1);
cat1_timeout ++;
if(cat1_timeout >= 2000){
return;
}
}
cat1_timeout = 0;
printf(“ready to send url\r\n”);
while(Iot_SendCmd(url,“OK”, 5000)){
HAL_Delay(1);
cat1_timeout ++;
if(cat1_timeout >= 2000){
return;
}
}
cat1_timeout = 0;
printf(“url set OK\r\n”);
}
到目前为止,我们程序中整体调用流程如下图所示:
![在这里插入图片描述](https://img-blog.csdnimg.cn/bab8baa937f84b519f248d98ac631261.png)
#### 3.1.3 POST 请求
上面的 URL 设置完成,我们就可以随时发送 POST 请求了,比如我们是一个传感器设备,周期性的采集传感器数据,到了自己设定的时间,就直接 POST 就行了,下面是 POST 请求的实现函数:
void http_post_message(const char *message) {
int length = strlen(message);
char at_post[32];
u16 cat1_timeout = 0;
snprintf(at_post, sizeof(at_post), “AT+QHTTPPOST=%d,%d,%d\r\n”, length, 5, 10);
while(Iot_SendCmd(at_post,“CONNECT”, 500)){
HAL_Delay(1);
cat1_timeout ++;
if(cat1_timeout >= 2000){
return;
}
}
cat1_timeout = 0;
printf(“ready to send post message!\r\n %s\r\n”, message);
while(Iot_SendCmd(message,“OK”, 5000)){
HAL_Delay(1);
cat1_timeout ++;
if(cat1_timeout >= 2000){
return;
}
}
cat1_timeout = 0;
printf(“http post OK\r\n”);
//“AT+QHTTPREAD=1\r\n”
while(Iot_SendCmd(“AT+QHTTPREAD=5\r\n”,“+QHTTPREAD”, 1000)){
HAL_Delay(1);
cat1_timeout ++;
if(cat1_timeout >= 2000){
return;
}
}
cat1_timeout = 0;
printf(“HTTPREAD OK\r\n”);
}
当然上面的参数`const char *message` 是需要我们自己组包的,一般来说使用 JSON 格式的比较多,比如测试过程中,我的程序如下:
![在这里插入图片描述](https://img-blog.csdnimg.cn/8240a3d535324fe8ab623f2af9d7d92d.png)
最后上面流程中调用的`Iot_SendCmd`函数也上一下:
int Iot_SendCmd(const char* cmd, char* reply, int wait)
{
u8 i=0;
char* rss_str;
int rssi,res;
CLEAR_EC800_Buffer(EC800_RX_Data);
Uart3\_sendBuffer((u8\*)cmd,strlen(cmd));
while(EC800ReceiveState != true)
{
HAL\_Delay(1);
i++;
if(i >= wait){
printf("cat1 check out\r\n");
return 0xFF;
}
}
EC800ReceiveState = false;
if (!strcmp(reply,"+CSQ"))
{
rss_str = strstr((char\*)EC800_RX_BUF, "+CSQ:");
if (!rss_str) {
return 1;
}
sscanf(rss_str, "+CSQ:%d,%d", &rssi, &res);
if (rssi != 99) {
printf("RSSI is %d\r\n",rssi);
CLEAR\_EC800\_Buffer(EC800_RX_Data);
return 0;
}
}
else if (strstr((char\*)EC800_RX_BUF, reply)){
printf("\r\n%s\r\n", EC800_RX_BUF);
CLEAR\_EC800\_Buffer(EC800_RX_Data);
return 0;
}
return 1;
}
#### 3.1.4 注意事项
(此处待更新,后续一些细节问题的说明需要补充)
**1、关于字体**
首先要注意的就是 字体,要和服务器的字体匹配;
通过`"AT+QHTTPCFG=\"contenttype\",1\r\n"`设置字体,这点上面已经提到过;
**2、关于 HTTP 响应**
指令`AT+QHTTPCFG="responseheader",1` 是启用输出 HTTP 响应头信息:
![在这里插入图片描述](https://img-blog.csdnimg.cn/e58b861e79804336b873a8a969b69f12.png)
如果启用了以后,使用`AT+QHTTPREAD`读取的 HTTP 响应消息如下:
![在这里插入图片描述](https://img-blog.csdnimg.cn/522a5cd11dea44dc8817e9e92a395091.png)
>
> 此时返回比较多,响应都有 600多个字节,接近 700 字节,这里大家写程序时候需要考虑到串口缓存大小。
>
>
>
不启用这个,则读取的 HTTP 响应消息如下:
![在这里插入图片描述](https://img-blog.csdnimg.cn/927b97d514ad4016b457c7af36d0c185.png)
>
> 此时返回不到200字节
>
>
>
**3、关于 cat1 模块返回数据的处理**
本文使用的是 STM32F103 芯片,在对于 AT 指令串口处理的时候需要注意,一般来说,对于普通的 AT 指令,我们可以直接使用 IDLE 中断进行判断是否接收完成,程序处理如下:
![在这里插入图片描述](https://img-blog.csdnimg.cn/a4d7258822bf4e169e62deb82ce7f2a8.png)
但是对于 `AT+QHTTPPOST` 和 `AT+QHTTPREAD` 来说,他们返回的不是一帧数据,而是分段的数据,如果使用 上面的处理就会出问题,所以我们需要对于这两个指令进行单独的处理:
/\*省略\*/
Uart3\_sendBuffer((u8\*)cmd,strlen(cmd));
/\*
此处串口回的不止是一帧数据,所以使用 IDLE 中断不合适
*/
if ((!strcmp(reply,“+QHTTPREAD:”))||(!strcmp(reply,“+QHTTPPOST:”))){
//读取和发送的处理,直接等一段时间
HAL_Delay(1000);// 500 600 800 1000 一直加大
}
/*
另外的设置指令大多都是等待一个 OK 返回,属于一帧数据
所以可以用 IDLE 中断
*/
else{
结尾
学习html5、css、javascript这些基础知识,学习的渠道很多,就不多说了,例如,一些其他的优秀博客。但是本人觉得看书也很必要,可以节省很多时间,常见的javascript的书,例如:javascript的高级程序设计,是每位前端工程师必不可少的一本书,边看边用,了解js的一些基本知识,基本上很全面了,如果有时间可以读一些,js性能相关的书籍,以及设计者模式,在实践中都会用的到。
开源分享:【大厂前端面试题解析+核心总结学习笔记+真实项目实战+最新讲解视频】
高级程序设计,是每位前端工程师必不可少的一本书,边看边用,了解js的一些基本知识,基本上很全面了,如果有时间可以读一些,js性能相关的书籍,以及设计者模式,在实践中都会用的到。