程序实现内容:

1.在station模式下,ESP8266作为client、server进行TCP连接
2.实现数据的发送、接收(同时回传)
实现思路:
TCP网络通信分层为:应用层、网络层、数据链路层、物理层;
1. 设置ESP8266为station模式,在数据链路层连接AP,获取IP地址;
2. 在网络层进行TCP连接:作为client连接远程server,作为server监听远程client信息;

数据链路层:

1. 设置ESP8266为station模式:wifi_set_opmode(STATION_MODE);

2. 配置连接到AP的相关参数,在该步骤中,需要知道AP的名称( ssid )、密码( password ): 官方所给的SDK中,该配置执行后,wifi模块自动与AP进行连接

 void user_set_station_config(){

   struct station_config stationconf;

   char ssid[]     = "Hello world";
char password[] = "chenyuping"; stationconf.bssid_set = ;
memset(stationconf.ssid, , sizeof(stationconf.ssid));
memset(stationconf.password, , sizeof(stationconf.password)); memcpy(stationconf.ssid, ssid, sizeof(ssid));
memcpy(stationconf.password, password, sizeof(password)); wifi_station_set_config(&stationconf); return ;
}

为确保ESP8266已同AP建立稳定的连接并且已获取到IP地址,可注册该事件的回调函数进行判定:
1.注册回调函数

 wifi_set_event_handler_cb(wifi_handle_even_cb);

 void wifi_handle_even_cb(System_Event_t *evt){
switch(evt->event_id){
case EVENT_STAMODE_GOT_IP:
flag_sta_conip = TRUE;
printf("Have got IP!!\n");
break;
defaut: printf("Have'n got IP\n");
break;
}
}

2.注册定时器,对回调函数中的flag_sta_conip进行判定以确定是否已获得IP地址

   os_timer_disarm(&ser_timer);
os_timer_setfn(&ser_timer, espconn_ser_timer_cb, NULL);
os_timer_arm(&ser_timer, , ); os_timer_disarm(&cli_timer);
os_timer_setfn(&cli_timer, espconn_cli_timer_cb, NULL);
os_timer_arm(&cli_timer, , );

网络层:

1. 检测ESP8266是否已同AP建立稳定连接,若是,则开始建立TCP client连接:

 void espconn_cli_timer_cb(void *timer_arg)
{
struct ip_info ipconfig; os_timer_disarm(&cli_timer);
wifi_get_ip_info(STATION_IF, &ipconfig); if (flag_sta_conip){
printf("connect successful !!!\n");
espconn_tcp_client_connect();
}
else {
printf("connect fail !!!\n");
os_timer_setfn(&cli_timer, espconn_cli_timer_cb, NULL);
os_timer_arm(&cli_timer, , );
}
}

建立TCP client连接(在该设置中,需要知道远程server的IP地址及端口号),同时注册连接、收发数据的回调函数:

 void espconn_tcp_client_connect(){

   user_tcp_conn.proto.tcp = &user_tcp;
user_tcp_conn.type = ESPCONN_TCP;
user_tcp_conn.state = ESPCONN_NONE; const char esp_tcp_server_ip[] = {, , , }; // remote IP of TCP server memcpy(user_tcp_conn.proto.tcp->remote_ip, esp_tcp_server_ip, );
user_tcp_conn.proto.tcp->remote_port = ; espconn_regist_connectcb(&user_tcp_conn, espconn_connect_cb);
espconn_regist_reconcb(&user_tcp_conn, espconn_reconnect_cb); espconn_regist_sentcb(&user_tcp_conn, espconn_sent_cb);
espconn_regist_recvcb(&user_tcp_conn, espconn_recv_data_cb); espconn_connect(&user_tcp_conn); }

2. ESP8266开启server服务(在该步骤中,通过对espconn_accept()函数返回值的判定,TRUE则关闭设置server开启的定时器,FALSE则继续保持开通,直到开启成功)

 void espconn_ser_timer_cb(void *timer_arg)
{
user_tcp_conn.proto.tcp = &user_tcp;
user_tcp_conn.type = ESPCONN_TCP;
user_tcp_conn.state = ESPCONN_NONE; os_timer_disarm(&ser_timer); user_tcp_conn.proto.tcp->local_port = espconn_port();
printf("The local port is %d\n",user_tcp_conn.proto.tcp->local_port); sint8 ret = espconn_accept(&user_tcp_conn);
printf("%d\n", ret);
if(!ret){
printf("Begin to listen!!!\n");
}
else{
printf("Fail to listen!!!\n");
os_timer_setfn(&ser_timer, espconn_ser_timer_cb, NULL);
os_timer_arm(&ser_timer, , );
}
}

接收数据的回调函数:

 espconn_regist_recvcb(&user_tcp_conn, espconn_recv_data_cb);

 void espconn_recv_data_cb(void *arg, char *pdata, unsigned short len)
{
uint8 *pDat;
const char str[] = "Light ON"; pDat = (uint8 *)malloc(len + );
memcpy(pDat, pdata, len);
*(pDat+len) = ;
// pDat[len] = 0; printf("The receiver data is %s",pDat);
printf("\n\n"); if(memcmp(pDat, str, sizeof(str)) == ) {
printf("Now Light is Run!\n");
espconn_send(&user_tcp_conn, "Now Light is Run!\n", );
} free(pDat);
}

发送数据的回调函数:

 espconn_regist_sentcb(&user_tcp_conn, espconn_sent_cb);

 void espconn_ser_timer_cb(void *timer_arg)
{
user_tcp_conn.proto.tcp = &user_tcp;
user_tcp_conn.type = ESPCONN_TCP;
user_tcp_conn.state = ESPCONN_NONE; os_timer_disarm(&ser_timer);
user_tcp_conn.proto.tcp->local_port = espconn_port();
printf("The local port is %d\n",user_tcp_conn.proto.tcp->local_port); sint8 ret = espconn_accept(&user_tcp_conn);
printf("%d\n", ret);
if(!ret){
printf("Begin to listen!!!\n");
}
else{
printf("Fail to listen!!!\n");
os_timer_setfn(&ser_timer, espconn_ser_timer_cb, NULL);
os_timer_arm(&ser_timer, , );
}
}

实现效果:

ESP8266 station模式下建立client、server TCP连接的更多相关文章

  1. .NET默认一个客户端对同一个服务器地址同时只能建立2个TCP连接

    做一个客户端的测试小程序测试web service的并发处理.开始用async task做,不管创建多少个task,用netstat看同时只有两个tcp连接.以为是async task的问题,改用Ba ...

  2. Windows下建立ArcGIS Server集群

    原创文章,转载须标明出处自: http://www.cnblogs.com/gisspace/p/8269525.html -------------------------------------- ...

  3. NAT模式下VMware中CentOS7无法连接外网的解决方法

    故障现象 ----------------------------------------------------------------------------------------------- ...

  4. TCP连接的状态与关闭方式及其对Server与Client的影响

    TCP连接的状态与关闭方式及其对Server与Client的影响 1. TCP连接的状态 首先介绍一下TCP连接建立与关闭过程中的状态.TCP连接过程是状态的转换,促使状态发生转换的因素包括用户调用. ...

  5. TCP连接的状态与关闭方式,及其对Server与Client的影响

    1. TCP连接的状态 首先介绍一下TCP连接建立与关闭过程中的状态.TCP连接过程是状态的转换,促使状态发生转换的因素包括用户调用.特定数据包以及超时等,具体状态如下所示: CLOSED:初始状态, ...

  6. ESP8266开发之旅 网络篇② ESP8266 工作模式与ESP8266WiFi库

        在网络篇①中,博主主要讲解了Arduino上开发ESP8266的插件库 Arduino Core For ESP8266.但是,并没有讲到关于这个模块的工作模式,所以本篇讲着重讲解ESP826 ...

  7. 嵌入式Linux下BOA网页server的移植

    **************************************************************************************************** ...

  8. 简述TCP连接的建立与释放(三次握手、四次挥手)

    在介绍TCP连接的建立与释放之前,先回顾一下相关知识. TCP是面向连接的运输层协议,它提供可靠交付的.全双工的.面向字节流的点对点服务.HTTP协议便是基于TCP协议实现的.(虽然作为应用层协议,H ...

  9. TCP连接的建立与释放(三次握手与四次挥手)

    TCP连接的建立与释放(三次握手与四次挥手) TCP是面向连接的运输层协议,它提供可靠交付的.全双工的.面向字节流的点对点服务.HTTP协议便是基于TCP协议实现的.(虽然作为应用层协议,HTTP协议 ...

随机推荐

  1. Git删除commit提交的log记录

    基于 GitFlow 工作流,可能某个提交(commit)导致了 bug,或者有多个提交需要返工,此时你就会用到删除提交. 接下来的内容都基于下面这张 git log 提交记录图来写.   git l ...

  2. PHP过滤

    1.过滤是不是URL(后台过滤) $url = "https://www.baidu.com"; if(!filter_var($url,FILTER_VALIDATE_URL)) ...

  3. select和epoll

    C/S编程模型,对每一个客户端都要开辟一个新的线程,效率必定低下.普通select模型是开辟两个线程,一个用来监听客户端的连接,另一个用于处理客户端请求. fd_set set; FD_ZERO(&a ...

  4. stm32 晶振不起振

    1. STM32f103有内部晶振.刚刚上电时,所有Clock都是源于内部晶振,所以当片内没有程序或内部程序没有使能外部晶振时,外部晶振是不会起振的.2. STM32f103有内部复位电路,只有当检测 ...

  5. C# 缓存工厂类

    描 述:缓存工厂类 /// <summary> /// 描 述:缓存工厂类 /// </summary> public class CacheFactory { /// < ...

  6. 实际SQL案例解决方法整理_LEAD函数相关

    表结构及数据如下: 需求: 将记录按照时间顺序排列,每三条记录为一组,若第二条记录与第一条记录相差5分钟,则删除该记录,若第三条与第二条记录相差5分钟,则删除该记录, 第二组同理,遍历全表,按要求删除 ...

  7. 面试官问你JS基本类型时他想知道什么?

    面试的时候我们经常会被问答js的数据类型.大部分情况我们会这样回答包括:1.基本类型(值类型或者原始类型): Number.Boolean.String.NULL.Undefined以及ES6的Sym ...

  8. 1005. Spell It Right(20)—PAT 甲级

    Given a non-negative integer N, your task is to compute the sum of all the digits of N, and output e ...

  9. 小白的Unity5之路(一)

    Player移动: public float speed = 6f; Vector3 movement; Rigidbody playerRididbody; void FixedUpdate () ...

  10. linux查看文件命令tail的使用

    一.介绍 linux tail命令用途是依照要求将指定的文件的最后部分输出到终端中,通俗讲来,就是把某个档案文件的最后几行显示到终端上,假设该档案有更新,tail会自己主动刷新,确保你看到最新的档案内 ...