3.wifi开发,网络编程

网络协议栈LwIP

WiFi UDP Clinet编程

WiFi UDP Server编程

WiFi TCP Client编程

WiFi TCP Server编程

一。LWIP原理介绍,API介绍,文件结构

1.Lwip支持的协议

2.API

3.文件结构

1.api目录:应用程序接口文件。

2.arch目录:与硬件和OS有关的文件,包括网络驱动、移植需要修改的文件。

3.core目录:LwIP的核心代码,包括ICMP、IP、UDP、TCP等协议的实现等。

4.include目录:LwIP的包含文件。

5.netif目录:PPP协议和LwIP网络设备驱动程序的模板,提供了网络接口驱动程序的基本框架。

4.Socket 通信模型

Socket是一个网络编程接口

Socket类型:

(1)流式套接字(SOCK_STREAM)

(2)数据报套接字(SOCK_DGRAM)

(3)原始套接字(SOCK_RAW)

5.Socket API

socket:创建一个套接字

bind:将本地端口号和IP地址绑定到套接字上

listen:TCP监听

accept:TCP监听接受处理

connect:TCP客户端连接

select:特殊套接字设置

send/sendto:发送数据包到已连接/未连接套接字上

recv/recvfrom:接收数据包从已连接/未连接套接字上

getsockopt/setsockopt:获取/改变套接字选项

getpeername/getsockname:获取远端/本地地址信息

close:关闭套接字

shutdown:按设置关闭套接字

gethostbyname/gethostbyaddr:地址域名映射

read:从套接字缓存读数据

write:向套接字缓存写数据

目的:

Lwip协议栈的实现目的,无非是要上层用来实现app的socket编程。 为了兼容性,lwip的socket通过宏定义提供标准的socket接口函数

5.介绍常用的API

(1)int socket (int domain, int type, int protocol);

        1.domain 是地址族

PF_INET  // internet IPv4协议

PF_INET6 // internet IPv6协议

PF_UNSPEC      // 用户协议

        2.type  // 套接字类型

SOCK_STREAM   // 流式套接字

SOCK_DGRAM    // 数据报套接字 S

OCK_RAW         //  原始套接字

        3.protocol //参数通常置为0     

IPPROTO_IP      0     

IPPROTO_TCP     6     

IPPROTO_UDP     17

(2)int bind (int sockfd, struct sockaddr* addr, int addrLen);

        1.sockfd 由socket() 调用返回

        2.addr 是指向 sockaddr_in 结构的指针,包含本机IP 地址和端口号

struct sockaddr_in{

        u_short sin_family // protocol family

        u_short sin_port     // port number

}

struct in_addr  sin_addr  //IP address (32-bits)
        3.addrLen : sizeof (struct sockaddr_in)

(3)地址结构

地址的数据结构

通用地址结构struct sockaddr{    u_short  sa_family;    // 地址族, AF_xxxchar  sa_data[14];     // 14字节协议地址};Internet协议地址结构struct sockaddr_in{           u_short sin_family;      // 地址族, AF_INET,2 bytesu_short sin_port;      // 端口,2 bytesstruct in_addr sin_addr;  // IPV4地址,4 bytes 	char sin_zero[8];        // 8 bytes unused,作为填充}; 
IPv4地址结构
// internet address  
struct in_addr
{in_addr_t  s_addr;            // u32 network address 
};

使用方法

1.定义一个struct sockaddr_in类型的变量并清空

struct sockaddr_in myaddr;
memset(&myaddr, 0, sizeof(myaddr));

2.填充地址信息

myaddr.sin_family = PF_INET;
myaddr.sin_port = htons(8888); 
myaddr.sin_addr.s_addr = inet_addr(“192.168.1.100”);

3.将该变量强制转换为struct sockaddr类型在函数中使用

bind(listenfd, (struct sockaddr*)(&myaddr), sizeof(myaddr));

(4)地址转换函数

1.unsigned long inet_addr(char *address);

        address是以’\0’结尾的点分IPv4字符串。该函数返回32位的地址。如果字符串包含的不是合法的IP地址,则函数返回-1。例如:

struct in_addr addr;
addr.s_addr = inet_addr(" 192.168.1.100 ");

2.char* inet_ntoa(struct in_addr address);

        address是IPv4地址结构,函数返回一指向包含点分IP地址的静态存储区字符指针。如果错误则函数返回NULL。

(5)获取/发送套接字

获取套接字

int getsockopt(int sock, int level,int optname, void *optval, socklen_t *optlen);

发送套接字

int setsockopt(int sock, int level,int optname, const void *optval, socklen_t optlen);

参数:

sock:将要被设置或者获取选项的套接字。

level:选项所在的协议层。

optname:需要访问的选项名。

optval:对于getsockopt(),指向返回选项值的缓冲。对于setsockopt(),指向包含新选项值的缓冲。

optlen:对于getsockopt(),作为入口参数时,选项值的最大长度。作为出口参数时,选项值的实际长度。对于setsockopt(),现选项的长度。

返回说明: 成功执行时,返回0。

失败返回-1,errno被设为以下的某个值

EBADF:sock不是有效的文件描述词

EFAULT:optval指向的内存并非有效的进程空间

EINVAL:在调用setsockopt()时,optlen无效

ENOPROTOOPT:指定的协议层不能识别选项

ENOTSOCK:sock描述的不是套接字

(6)listen监听

int listen (int sockfd, int backlog);

sockfd:监听连接的套接字。

backlog 指定了正在等待连接的最大队列长度,它的作用在于处理可能同时出现的几个连接请求。 DoS(拒绝服务)攻击即利用了这个原理,非法的连接占用了全部的连接数,造成正常的连接请求被拒绝。 返回值: 0 或 -1。

完成listen()调用后,socket变成了监听socket(listening socket).

(7)accept

int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen) ;

返回值:已建立好连接的套接字或-1
sockfd : 监听套接字 
addr : 对方地址
addrlen:地址长度

(8)connect

int connect(int sockfd, struct sockaddr *serv_addr, int addrlen);

返回值:0 或 -1

sockfd : socket返回的文件描述符

serv_addr : 服务器端的地址信息  

addrlen : serv_addr的长度

(9)send/sendto

ssize_t  send(int  socket,  const  void  *buffer,  size_t  length, int flags);
ssize_t  sendto(int s, const void *data, size_t size, int flags,const struct sockaddr *to, socklen_t tolen);

返回值: 成功:实际发送的字节数 失败:-1, 并设置errno

头文件: #include <sys/socket.h>

buffer : 发送缓冲区首地址 length : 发送的字节数 flags : 发送方式(通常为0)

(10)recv/recvfrom

ssize_t  recv(int  socket,  const  void  *buffer,  size_t  length, int flags);
ssize_t  recvfrom(int s, void *mem, size_t len, int flags,struct sockaddr *from, socklen_t *fromlen);

返回值: 成功:实际接收的字节数 失败:-1,并设置errno

buffer : 发送缓冲区首地址

length : 发送的字节数

flags : 接收方式(通常为0)

(11)套接字关闭

int close(int sockfd);
int shutdown(int sockfd, int howto);

close是双向通信的关闭

TCP连接是双向的(是可读写的),当我们使用close时,会把读写通道都关闭,有时侯我们希望只关闭一个方向,这个时候我们可以使用shutdown。针对不同的howto,系统回采取不同的关闭方式。 howto = 0 关闭读通道,但是可以继续往套接字写数据。 howto = 1 和上面相反,关闭写通道。只能从套接字读取数据。 howto = 2 关闭读写通道,和close()一样

二。实验:WiFi UDP Clinet编程

1.功能分析

完成UDP Client功能开发

(1)PC模拟UDP Server,指定(IP,PORT),等待Client数据

(2)UDP Client向Sever发送“I am Client!”

(3)Sever收到数据后,向Client发送“I am Server!”

2.功能实现

(1)在SDK目录下新建udpclient目录

(2)拷贝Station目录下所有文件到udpclient目录下

(3)在user目录下新建udpclient.c

(4)在include目录下新建udpclient.h

(1)Sourceinsight配置

在之前的工程中,移除AP文件夹 添加udpclient文件夹

添加lwip文件夹

同步:

发现的问题:

QQ的截屏比SourceInsight复制优先级低,即先复制代码,在截屏之后粘贴还是复制的代码

3.代码实现

1.在udpclient.c下,添加《整体复制粘贴

(1)udpclient初始化

(2)udpclient任务

#include "esp_common.h"#include "freertos/FreeRTOS.h"
#include "freertos/task.h"#include "lwip/sockets.h"
#include "lwip/dns.h"
#include "lwip/netdb.h"
#include "udpclient.h"#define SERVERADDR "192.168.3.12"//注意:这是wifi分配的ip地址,会变化
#define SERVERPORT 8000
/******************************************************************************* FunctionName : ATaskUdpClient* Description  : ATaskUdpClient 任务* Parameters   : none* Returns      : none
*******************************************************************************/
void ATaskUdpClient( void *pvParameters )
{int iVariableExample = 0;int fd = -1;int NetTimeOnt = 5000;int ret;struct sockaddr_in ServerAddr;char udpmsg[48];STATION_STATUS StaStatus;do {StaStatus = wifi_station_get_connect_status();vTaskDelay(100);}while(StaStatus != STATION_GOT_IP);fd = socket(PF_INET,SOCK_DGRAM,0);if(fd == -1){printf("get socket fail!\n");vTaskDelete(NULL);}setsockopt(fd,SOL_SOCKET,SO_RCVTIMEO,&NetTimeOnt,sizeof(int));memset(&ServerAddr,0,sizeof(ServerAddr));ServerAddr.sin_family = AF_INET;ServerAddr.sin_addr.s_addr = inet_addr(SERVERADDR);ServerAddr.sin_port = htons(SERVERPORT);ServerAddr.sin_len = sizeof(ServerAddr);for(;;){sendto(fd,"I am UdpClient!",sizeof("I am UdpClient!"),0,(struct sockaddr *)&ServerAddr,(socklen_t)ServerAddr.sin_len);do{ret = recvfrom(fd,udpmsg,48,0,(struct sockaddr *)&ServerAddr,(socklen_t*)(&ServerAddr.sin_len));if(ret > 0){printf("UdpServer:%s\n",udpmsg);}else{printf("UdpServer data is no!\n");}}while(ret == -1);}vTaskDelete( NULL );
}
/******************************************************************************* FunctionName : UdpClient_init* Description  : UdpClient_init 初始化* Parameters   : none* Returns      : none
*******************************************************************************/
void UdpClient_init(void)
{xTaskCreate(ATaskUdpClient, "UdpClient", 256, NULL, 4, NULL);
}

 注意:上述的IPv设置的是本机的IP地址

Win+R---》cmd进入命令行

输入ipconfig查询无线网络ipv4地址

对上述代码的解释

<1>判断是否获取到IP地址

<2>创建socket

<3>设置接收超时时间

<4>赋值server信息

<5>发送数据到server端

<6>从server端接收数据

2.在udpclient.h下声明《整体复制》

#ifndef __UART_H__
#define __UART_H__#ifdef __cplusplus
extern "C" {
#endifvoid ATaskUdpClient( void *pvParameters );
void UdpClient_init(void);#ifdef __cplusplus
}
#endif#endif

3.在user_main中

UdpClient_init();

5.运行方式

(1)创建一个UDP,ip为本机被分配的无线ipv4地址,端口为程序代码设置的端口

查询IP地址(通过ipconfig命令获取本机ip地址

(2)建立串口通讯,波特率为74880,关闭RTS,使用终端模式

4.结果

(1)串口中打印从UDP服务器传输过来的数据(123)

三。实验:WiFi UDP Server编程

1.功能分析

实验要求:
完成UDP Server功能开发

1.PC模拟UDP Client,指定UDP Server (IP,PORT),发送“I am Client!”

2.Sever收到数据后,向Client发送“I am Server!”

2.功能实现

新建udpserver工程目录

1.在SDK目录下新建udpserver目录

2.拷贝udpclient目录下所有文件到udpserver目录下

新建udpserver源码文件

1.在user目录下新建udpserver.c

2.在include目录下新建udpserver.h

Sourceinsight配置

1.在之前的工程中,移除udpclient文件夹

2.添加udpserver文件夹

3.代码实现

1.复制之前udpclient代码到udpserver上,进行修改

    (1)重命名UdpClient_init为UdpServer_init

    (2)重命名ATaskUdpClient为ATaskUdpServer

2.修改user_main.c

    (1)包涵udpserver.h

    (2)在user_init里修改为UdpServer_init

代码的详细解释

1.判断是否获取到IP地址

2.创建socket

3.Serveraddr 信息设置

4.设置接收超时时间

5.绑定socket

6.从Client端接收数据

7.发送数据到Client端

结果:

四。WiFi TCP Client编程

1.功能分析

实验要求:

完成TCP Client功能开发

1.PC模拟TCP Server,指定(IP,PORT),等待Client数据

2.TCP Client向Sever发送“I am Client!”

3.Sever收到数据后,向Client发送“I am Server!”

2.功能实现

新建tcpclient工程目录

1.在SDK目录下新建tcpclient目录

2.拷贝udpclient目录下所有文件到tcpclient目录下

新建tcpclient源码文件

1.在user目录下新建tcpclient.c

2.在include目录下新建tcpclient.h

Sourceinsight配置

1.在之前的工程中,移除udpserver文件夹

2.添加tcpclient文件夹

3.代码实现

1.复制之前udpclient代码到tcpclient上,进行修改

    (1)重命名UdpClient_init为TcpClient_init

    (2)重命名ATaskUdpClient为ATaskTcpClient

2.修改user_main.c

    (1)包涵tcpclient.h

    (2)在user_init里修改为TcpClient_init

代码详细解释

1.判断是否获取到IP地址

2.创建socket

3.设置接收超时时间

4.赋值server信息

5.连接到server端

6.发送数据到server端

7.从server端接收数据

五。实验:WiFi TCP Server编程

1.功能分析

实验要求:

完成TCP Server功能开发

1.PC模拟TCP Client,指定Server(IP,PORT)

    (1)进行连接,连接成功后发生“I am Client!”

    (2)Sever收到数据后,向Client发送“I am Server!”

2.功能实现

新建tcpserver工程目录

1.在SDK目录下新建tcpserver目录

2.拷贝tcpclient目录下所有文件到tcpserver目录下

新建tcpserver源码文件

1.在user目录下新建tcpserver.c

2.在include目录下新建tcpserver.h

Sourceinsight配置

1.在之前的工程中,移除tcpclient文件夹

2.添加tcpserver文件夹

3.代码实现

1.复制之前tcpclient代码到tcpserver上,进行修改

    (1)重命名TcpClient_init为TcpServer_init

    (2)重命名ATaskTcpClient为ATaskTcpServer

2.修改user_main.c

    (1)包涵tcpserver.h

    (2)在user_init里修改为TcpServer_init

代码详解

1.判断是否获取到IP地址

2.创建socket

3.设置接收超时时间

4.赋值server信息

5.绑定socket

6.监听socket

7.处理Client 连接

8.接收Client 数据

9.发送数据到Server

10.关闭socket

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/85437.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

机器学习笔记:Huber Loss

1 介绍 Huber Loss是回归问题中的一种损失函数&#xff0c;它结合了均方误差MSE和绝对误差MAE的特点。 Huber Loss在误差较小的时候是平方损失&#xff0c;而在误差较大的时候是线性损失。因此&#xff0c;它在处理有噪声的数据时&#xff0c;尤其是存在离群点的情况下&#x…

nginx反向代理vue项目

文章目录 前言一、创建站点1.添加站点2.添加ssl证书 二、反向代理vue项目1.添加反向代理2.更改vue项目配置3.修改反向代理配置 前言 项目描述&#xff1a;前端vue项目、后端Java项目、首页WordPress项目 客户要求&#xff1a;使用宝塔进行部署 需求描述&#xff1a;客户只有一…

一文彻底理解synchronized(通俗易懂的synchronized)

目录 一、什么是synchronized 二、synchronized的四种用法 2.1、修饰一个代码块 2.2、修饰一个方法 2.3、修饰一个静态的方法 2.4、修饰一个类 三、使用案例分析 3.1、修饰一个代码块 3.2、修饰一个方法 3.3、修饰一个静态的方法 3.4、修饰一个类 3.5 经典用法&…

蓝桥杯 题库 简单 每日十题 day8

01 扫雷 题目描述 在一个n行列的方格图上有一些位置有地雷&#xff0c;另外一些位置为空。 请为每个空位置标一个整数&#xff0c;表示周围八个相邻的方格中有多少个地雷。 输入描述 输入的第一行包含两个整数n&#xff0c;m。 第2行到第n1行每行包含m个整数&#xff0c;相邻整…

Winform直接与Wpf交互

Winform项目中&#xff0c;可以直接使用wpf中的自定义控件和窗体 测试环境&#xff1a; vistual studio 2017 window 10 一 winform直接使用wpf的自定义控件 步骤如下&#xff1a; 1 新建winfrom项目&#xff0c;名为WinFormDemo&#xff0c;默认有一个名为Form1的窗体…

关于表单快速开发低代码技术平台的内容介绍

运用什么样的表单快速开发软件平台可以实现高效率创收&#xff1f;随着科技的进步和飞速发展&#xff0c;专业的低代码技术平台已经走入了很多企业的办公职场中&#xff0c;它们灵活、轻量级、优质、高效、易维护等优势特点&#xff0c;可以高效助力广大企业提质增效&#xff0…

位移贴图的实现原理

在以前的文章中介绍过GLTF编辑器 &#xff0c; 编辑器可以对模型的各种材质纹理进行编辑修改&#xff0c;但是有一些新手用户可能对这些材质纹理不太了解&#xff0c;所以我收集了一些资料对这些材质纹理做一下详细的介绍&#xff0c;今天这篇文章主要是介绍位移贴图。 1、什么…

stm32之智能垃圾桶实战

之前用过51做过一个垃圾桶的小项目&#xff0c;这里用32重新搞了一下。视频的效果和之前一样&#xff0c;可参考这个垃圾桶效果 。 一、项目描述&#xff08;同51&#xff09; 项目主要是模拟不用手动打开垃圾桶盖&#xff0c;而进行自动操作。自动打开的条件如下&#xff1a…

【二叉树魔法:链式结构与递归的纠缠】

本章重点 二叉树的链式存储二叉树链式结构的实现二叉树的遍历二叉树的节点个数以及高度二叉树的创建和销毁二叉树的优先遍历和广度优先遍历二叉树基础oj练习 1.二叉树的链式存储 二叉树的链式存储结构是指&#xff0c;用链表来表示一棵二叉树&#xff0c;即用链来指示元素的逻辑…

点云从入门到精通技术详解100篇-单期点云的高斯曲率定位桥梁潜在损伤技术研究

目录 前言 国内外研究现状 三维激光扫描对桥梁损伤检测的研究现状 基于点云高斯曲率损伤检测的研究现状 柱体偏差检测技术研究现状 存在的问题 法向量约束高斯曲率的 TLS 桥面潜在损伤区域探测 2.1 高斯曲率探伤的基本理论 2.2 点云拓扑关系建立的方法比较 2.2.1 KD-…

社区分享|MeterSphere变身“啄木鸟”,助力云帐房落地接口自动化测试

云帐房网络科技有限公司&#xff08;以下简称为“云帐房”&#xff09;成立于2015年3月&#xff0c;以“成为最值得信赖的税务智能公司”为愿景&#xff0c;运用人工智能、大数据等互联网技术&#xff0c;结合深厚的财税行业服务经验&#xff0c;为代账公司和中大型企业提供智能…

【2023年中国研究生数学建模竞赛华为杯】E题 出血性脑卒中临床智能诊疗建模 问题分析、数学模型及代码实现

【2023年中国研究生数学建模竞赛华为杯】E题 出血性脑卒中临床智能诊疗建模 1 题目 1.1 背景介绍 出血性脑卒中指非外伤性脑实质内血管破裂引起的脑出血&#xff0c;占全部脑卒中发病率的10-15%。其病因复杂&#xff0c;通常因脑动脉瘤破裂、脑动脉异常等因素&#xff0c;导致…

黑马JVM总结(十九)

&#xff08;1&#xff09;GC调优1 通过官网查看查看JVM的参数&#xff1a; 可以使用java命令查看当前环境下的虚拟机参数&#xff1a; 学会使用一些工具如前面学的jmap &#xff0c;jconsole等等工具 &#xff08;2&#xff09;GC调优2 垃圾回收调优只是众多调优中的一个方…

读高性能MySQL(第4版)笔记14_备份与恢复(中)

1. 在线备份 2. 离线备份 2.1. 关闭MySQL做备份是最简单、最安全的 2.2. 所有获取一致性副本的方法中最好的 2.3. 损坏或不一致的风险最小 2.4. 根本不用关心InnoDB缓冲池中的脏页或其他缓存 2.5. 不需要担心数据在尝试备份的过程中被修改 2.5.1. 服务器不对应用提供访问…

【动态规划刷题 17】回文子串 最长回文子串

647. 回文子串 链接: 647. 回文子串 给你一个字符串 s &#xff0c;请你统计并返回这个字符串中 回文子串 的数目。 回文字符串 是正着读和倒过来读一样的字符串。 子字符串 是字符串中的由连续字符组成的一个序列。 具有不同开始位置或结束位置的子串&#xff0c;即使是由…

zookeeper + kafka

Zookeeper 概述 Zookeeper是一个开源的分布式服务管理框架。存储业务服务节点元数据及状态信息&#xff0c;并负责通知再 ZooKeeper 上注册的服务几点状态给客户端 Zookeeper 工作机制 Zookeeper从设计模式角度来理解: 是一个基于观察者模式设计的分布式服务管理框架&…

Learn Prompt-GPT-4:能力

GPT-4能力大赏​ 常识知识推理​ 一个猎人向南走了一英里&#xff0c;向东走了一英里&#xff0c;向北走了一英里&#xff0c;最后回到了起点。他看到了一只熊&#xff0c;于是开枪打了它。这只熊是什么颜色的&#xff1f; 答案是白色&#xff0c;因为这种情况只可能发生在北…

Gin学习记录4——Controller和中间件

一. Controller 用不同的Controller可以实现业务的分类&#xff0c;不同类型的请求可以共用同一套中间件 1.1 单文件Controller 几乎等同于函数封装&#xff0c;直接将ctrl的代码写入到一个文件里然后调用&#xff1a; package adminimport ("net/http""git…

数据结构与算法(六)--链表的遍历,查询和修改,删除操作

一、前言 上篇文章我们了解了链表的概念以及链表底层的搭建以及向链表中添加元素的操作。本次我们继续学习链表剩余的操作&#xff1a;遍历&#xff0c;查询和修改、删除操作。 二、链表查询以及遍历 ①获得链表的第index(0-based)个位置的元素&#xff08;不常用&#xff0…

MySQL详细案例 1:MySQL主从复制与读写分离

文章目录 1. MySQL主从复制1.1 使用场景1.2 MySQL的复制类型1.3 主从复制的作用1.4 主从复制的工作过程1.5 实现MySQL主从复制1.5.1 前置准备1.5.2 主服务器mysql配置1.5.3 从服务器1 mysql配置1.5.4 从服务器2 mysql配置1.5.5 测试 1.6 主从复制的3种同步模式1.6.1 异步复制&a…