程序实现内容:

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. 关于python线程池threadpool

    #coding=utf-8 import time import threadpool def wait_time(n): print('%d\n' % n) time.sleep(2) #在线程池中 ...

  2. rsync + mysql + gzip + --single-transaction

    1.rsync -avz 172.16.2.61:~/vs/program/elasticsearch-5.0.0 --exclude=elasticsearch-5.0.0/data/* ./ 从其 ...

  3. 允许跨域资源共享(CORS)携带 Cookie (转载)

    如何让CORS携带Cookie CORS 是一个 W3C 标准,全称是“跨域资源共享”(Cross-origin resource sharing).默认浏览器为了安全,遵循“同源策略”,不允许 Aj ...

  4. C#中枚举

    1.枚举是一组命名整形常量,枚举类型使用Enum关键字进行声明的.在C#中枚举是值数据类型,枚举包含自己的值,且不能继承或传递继承.

  5. nodejs运行的时候报错:Error: write EIO以及乱码解决方式

    在运行node.js的过程中报如下错误: events.js:72 throw er; // Unhandled 'error' event ^ Error: write EIO at errnoEx ...

  6. 【Dubbo源码阅读系列】之远程服务调用(上)

    今天打算来讲一讲 Dubbo 服务远程调用.笔者在开始看 Dubbo 远程服务相关源码的时候,看的有点迷糊.后来慢慢明白 Dubbo 远程服务的调用的本质就是动态代理模式的一种实现.本地消费者无须知道 ...

  7. C++编译器是如何管理类和对象的,类的成员函数和成员变量

    C++中的class从面向对象理论出发,将变量(属性)和函数(方法)集中定义在一起,用于描述现实世界中的类.从计算机的角度,程序依然由数据段(栈区内存)和代码段(代码区内存)构成. #include ...

  8. 底层文件I/O操作中read()函数的缓存问题

    最近在学习Linux过程中看到文件I/O操作这里时,文件I/O操作的系统调用涉及的5个函数:open(),read(),write(),lseek(),close().在一开始就阐明这些函数的特点是不 ...

  9. NodeJs 实现简单WebSocket 即时通讯

    至于服务器语言选择nodeJs,一是因为自己是做前端的,对javascript比较熟悉,相比于其他后台语言,自然会更喜欢nodeJs了, 二是NodeJs本身事件驱动的方式很擅长与大量客户端保持高并发 ...

  10. php composer 实现类的自动加载

    我们在开发项目中会经常用到第三方的类库插件,但是如果每次需要使用的时候都会在代码的某一处去引入,然后在实例化,这样做感觉很不方便,那么怎么实现自动加载呢,下面简单介绍使用composer实现自动加载: ...