Linux网络编程之socket通信

news2025/1/31 2:49:47

Linux网络编程之socket通信

一、socket相关函数使用

1.1 IP地址转换函数:

小端法:(pc本地存储) 高位存高地址,低位存低地址。

大端法:(网络存储) 高位存低地址,低位存高地址。

htonl --> 本地 -->> 网络 (IP)

htons --> 本地 -->> 网络 (port)

ntohl --> 网络 -->> 本地(IP)

ntohs --> 网络 -->> 本地(Port)
1.int inet_pton(int af, const char *src, void *dst);
本地字节序(string IP) ---> 网络字节序
af:AF_INET、AF_INET6
src:传入,IP地址(点分十进制)
dst:传出,转换后的 网络字节序的 IP地址。 
返回值:
成功: 1
异常: 0 说明src指向的不是一个有效的ip地址。
失败:-1
2.const char *inet_ntop(int af, const void *src, char *dst, socklen_t size);
网络字节序 ---> 本地字节序(string IP)
af:AF_INET、AF_INET6
src:网络字节序IP地址
dst:本地字节序(string IP)
size:dst 的大小
返回值: 成功 dst	
失败:NULL
sockaddr地址结构: IP + port	--> 在网络环境中唯一标识一个进程。
struct sockaddr_in addr;
addr.sin_family = AF_INET/AF_INET6
addr.sin_port = htons(9527);
int dst;
inet_pton(AF_INET, "192.157.22.45", (void *)&dst);
addr.sin_addr.s_addr = dst;*】addr.sin_addr.s_addr = htonl(INADDR_ANY); 取出系统中有效的任意IP地址,二进制类型。
bind(fd, (struct sockaddr *)&addr, size);

1.2 socket函数:

#include <sys/socket.h>
1.int socket(int domain, int type, int protocol);
创建一个套接字
domain:AF_INET、AF_INET6、AF_UNIX
type:SOCK_STREAM、SOCK_DGRAM
protocol:0 使用type对应的典型协议
返回值:
成功: 新套接字所对应文件描述符
失败: -1 errno
#include <arpa/inet.h>
2.int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
给socket绑定一个地址结构 (IP+port)
sockfd:socket 函数返回值

struct sockaddr_in addr;
addr.sin_family = AF_INET;
addr.sin_port = htons(8888);
addr.sin_addr.s_addr = htonl(INADDR_ANY);
addr: 传入参数(struct sockaddr *)&addr

addrlen: sizeof(addr) 地址结构的大小。
返回值:
成功:0
失败:-1 errno
3.int listen(int sockfd, int backlog);
设置同时与服务器建立连接的上限数(同时进行3次握手的客户端数量)
sockfd:socket 函数返回值
backlog:上限数值,最大值 128
返回值:
成功:0
失败:-1 errno	
4.int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen);
阻塞等待客户端建立连接,成功的话,返回一个与客户端成功连接的socket文件描述符。
sockfd:socket 函数返回值
addr:传出参数。成功与服务器建立连接的那个客户端的地址结构(IP+port)

socklen_t clit_addr_len = sizeof(addr);
addrlen:传入传出参数,&clit_addr_len
入:addr的大小。 出:客户端addr实际大小。

返回值:
成功:能与客户端进行数据通信的 socket 对应的文件描述。
失败: -1 , errno
5.int connect(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
使用现有的 socket 与服务器建立连接
sockfd: socket 函数返回值

struct sockaddr_in srv_addr; // 服务器地址结构
srv_addr.sin_family = AF_INET;
srv_addr.sin_port = 9527 	// 跟服务器bind时设定的 port 完全一致。
inet_pton(AF_INET, "服务器的IP地址"&srv_adrr.sin_addr.s_addr);
addr:传入参数,服务器的地址结构
addrlen:服务器的地址结构的大小

返回值:
成功:0
失败:-1 errno
如果不使用bind绑定客户端地址结构, 采用"隐式绑定"

二、客户端服务端socket通信

服务端:

#include <stdio.h>
#include <ctype.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <errno.h>
#include <pthread.h>

// 端口号
#define SERV_PORT 9527


void sys_err(const char *str)
{
    perror(str);
    exit(1);
}

int main(int argc, char *argv[])
{
	// lfd为监听socket句柄 cfd为与客户端建立服务socket句柄 在accept函数中生成
    int lfd = 0, cfd = 0;
    int ret, i;
    // BUFSIZ为系统宏
    char buf[BUFSIZ], client_IP[1024];

	// 定义服务器地址结构 和 客户端地址结构
    struct sockaddr_in serv_addr, clit_addr;
    // 客户端地址结构大小
    socklen_t clit_addr_len;

	// IPv4
    serv_addr.sin_family = AF_INET;
    // 转为网络字节序的 端口号	
    serv_addr.sin_port = htons(SERV_PORT);
    // 获取本机任意有效IP
    serv_addr.sin_addr.s_addr = htonl(INADDR_ANY);

    lfd = socket(AF_INET, SOCK_STREAM, 0);		//创建一个 socket
    if (lfd == -1) {
        sys_err("socket error");
    }

	// 给服务器socket绑定地址结构(IP+port)
    bind(lfd, (struct sockaddr *)&serv_addr, sizeof(serv_addr));
	// 设置监听上限
    listen(lfd, 128);
	// 获取客户端地址结构大小
    clit_addr_len = sizeof(clit_addr);
	// 阻塞等待客户端连接请求
    cfd = accept(lfd, (struct sockaddr *)&clit_addr, &clit_addr_len);
    if (cfd == -1)
        sys_err("accept error");

	// 根据accept传出参数,获取客户端 ip 和 port
    printf("client ip:%s port:%d\n", 
            inet_ntop(AF_INET, &clit_addr.sin_addr.s_addr, client_IP, sizeof(client_IP)), 
            ntohs(clit_addr.sin_port));

    while (1) {
    	// 读客户端数据
        ret = read(cfd, buf, sizeof(buf));
        // 写到屏幕查看
        write(STDOUT_FILENO, buf, ret);
		// 小写 -- 大写
        for (i = 0; i < ret; i++)
            buf[i] = toupper(buf[i]);
		// 将大写 写回给客户端
        write(cfd, buf, ret);
    }

    close(lfd);
    close(cfd);

    return 0;
}

客户端:

#include <stdio.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <errno.h>
#include <pthread.h>

#define SERV_PORT 9527

void sys_err(const char *str)
{
	perror(str);
	exit(1);
}

int main(int argc, char *argv[])
{
    int cfd;
    int conter = 10;
    char buf[BUFSIZ];
    //服务器地址结构
    struct sockaddr_in serv_addr;

    serv_addr.sin_family = AF_INET;
    serv_addr.sin_port = htons(SERV_PORT);
    //inet_pton(AF_INET, "127.0.0.1", &serv_addr.sin_addr.s_addr);
    inet_pton(AF_INET, "127.0.0.1", &serv_addr.sin_addr);

    cfd = socket(AF_INET, SOCK_STREAM, 0);
    if (cfd == -1)
        sys_err("socket error");

    int ret = connect(cfd, (struct sockaddr *)&serv_addr, sizeof(serv_addr));
    if (ret != 0)
        sys_err("connect err");

    while (--conter) {
        write(cfd, "hello\n", 6);
        ret = read(cfd, buf, sizeof(buf));
        write(STDOUT_FILENO, buf, ret);
        sleep(1);
    }

    close(cfd);

	return 0;
}

三、通信时序与代码对应关系

在这里插入图片描述
通信时序总结:
三次握手:

主动发起连接请求端,发送 SYN 标志位,请求建立连接。 携带序号号、数据字节数(0)、滑动窗口大小。

被动接受连接请求端,发送 ACK 标志位,同时携带 SYN 请求标志位。携带序号、确认序号、数据字节数(0)、滑动窗口大小。

主动发起连接请求端,发送 ACK 标志位,应答服务器连接请求。携带确认序号。

四次挥手:

主动关闭连接请求端, 发送 FIN 标志位。 

被动关闭连接请求端, 应答 ACK 标志位。 		 ----- 半关闭完成。


被动关闭连接请求端, 发送 FIN 标志位。

主动关闭连接请求端, 应答 ACK 标志位。		 ----- 连接全部关闭

滑动窗口:

发送给连接对端,本端的缓冲区大小(实时),保证数据不会丢失。

四、socket通信封装接口

// 读取固定的长度
ssize_t Readn(int fd, void *vptr, size_t n)
{
	size_t  nleft;              //usigned int 剩余未读取的字节数
	ssize_t nread;              //int 实际读到的字节数
	char   *ptr;

	ptr = vptr;
	nleft = n;                  //n 未读取字节数

	while (nleft > 0) {
		if ((nread = read(fd, ptr, nleft)) < 0) {
			if (errno == EINTR)
				nread = 0;
			else
				return -1;
		} else if (nread == 0)
			break;

		nleft -= nread;   //nleft = nleft - nread 
		ptr += nread;
	}
	return n - nleft;
}
// 写固定长度
ssize_t Writen(int fd, const void *vptr, size_t n)
{
	size_t nleft;
	ssize_t nwritten;
	const char *ptr;

	ptr = vptr;
	nleft = n;
	while (nleft > 0) {
		if ( (nwritten = write(fd, ptr, nleft)) <= 0) {
			if (nwritten < 0 && errno == EINTR)
				nwritten = 0;
			else
				return -1;
		}
		nleft -= nwritten;
		ptr += nwritten;
	}
	return n;
}
static ssize_t my_read(int fd, char *ptr)
{
	static int read_cnt;
	static char *read_ptr;
	static char read_buf[100];

	if (read_cnt <= 0) {
again:
		if ( (read_cnt = read(fd, read_buf, sizeof(read_buf))) < 0) {   //"hello\n"
			if (errno == EINTR)
				goto again;
			return -1;
		} else if (read_cnt == 0)
			return 0;

		read_ptr = read_buf;
	}
	read_cnt--;
	*ptr = *read_ptr++;

	return 1;
}

/*readline --- fgets*/    
//读一行 传出参数 vptr
ssize_t Readline(int fd, void *vptr, size_t maxlen)
{
	ssize_t n, rc;
	char    c, *ptr;
	ptr = vptr;

	for (n = 1; n < maxlen; n++) {
		if ((rc = my_read(fd, &c)) == 1) {   //ptr[] = hello\n
			*ptr++ = c;
			if (c == '\n')
				break;
		} else if (rc == 0) {
			*ptr = 0;
			return n-1;
		} else
			return -1;
	}
	*ptr = 0;

	return n;
}

五、TCP通信流程分析

server:
	1. socket()	           创建socket
	2. bind()	           绑定服务器地址结构
	3. listen()	           设置监听上限
	4. accept()	           阻塞监听客户端连接
	5. read(fd)	           读socket获取客户端数据
	6. toupper()--大写
	7. write(fd)
	8. close()

client:
	1. socket()	            创建socket
	2. connect()	        与服务器建立连接
	3. write()	            写数据到 socket
	4. read()	            读转换后的数据。
	5. write(STDOUT_FILENO) 显示读取结果
	6. close()

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

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

相关文章

13基于多目标粒子群算法的微电网优化调度(matlab程序)

参考文献 基于多目标粒子群算法的微电网优化调度——王金全&#xff08;2014电网与清洁能源&#xff09; 主要内容 针对光伏电池、风机、微型燃气轮机、柴油发电机以及蓄电池组成的微电网系统的优化问题进行研究&#xff0c;在满足系统约束条件下&#xff0c;建立了包含运行…

day25【代码随想录】左叶子之和、找树左下角的值、从中序与后序遍历序列构造二叉树、从中序与前序遍历序列构造二叉树、最大二叉树

文章目录前言一、左叶子之和&#xff08;力扣404&#xff09;1、递归遍历2、非递归遍历二、找树左下角的值&#xff08;力扣513&#xff09;1、迭代法&#xff08;层序遍历&#xff09;2、递归法三、从中序与后序遍历序列构造二叉树&#xff08;力扣106&#xff09;四、从中序与…

微服务框架 SpringCloud微服务架构 微服务面试篇 54 微服务篇 54.1 SpringCloud常见组件有哪些?

微服务框架 【SpringCloudRabbitMQDockerRedis搜索分布式&#xff0c;系统详解springcloud微服务技术栈课程|黑马程序员Java微服务】 微服务面试篇 文章目录微服务框架微服务面试篇54 微服务篇54.1 SpringCloud常见组件有哪些&#xff1f;54 微服务篇 54.1 SpringCloud常见组…

【验证码逆向专栏】某片滑块、点选验证码逆向分析

声明 本文章中所有内容仅供学习交流使用&#xff0c;不用于其他任何目的&#xff0c;不提供完整代码&#xff0c;抓包内容、敏感网址、数据接口等均已做脱敏处理&#xff0c;严禁用于商业用途和非法用途&#xff0c;否则由此产生的一切后果均与作者无关&#xff01; 本文章未…

30岁了想转行学Python,来得及吗?

是否来得及要看决心有多大&#xff0c;行动力有多强。一般来说&#xff0c;只要目标明确&#xff0c;足够自律&#xff0c;心理强大&#xff0c;做任何事情都是来得及的&#xff0c;当下就是最好的开始。30岁真的不算啥&#xff0c;有人四五十岁才开始奋斗&#xff0c;依然能过…

C语言之内存管理(十七)(转世灵童现世)

上一篇: C语言入门篇之轮回法器&#xff08;十六&#xff09;&#xff08;指针第五卷&#xff09; 逐梦编程&#xff0c;让中华屹立世界之巅。 简单的事情重复做,重复的事情用心做,用心的事情坚持做&#xff1b; 文章目录前言一、内存管理具体介绍1.作用域2.生命周期的定义3.局…

为什么说学人工智能一定要学Python?

有很多人在问博主&#xff0c;为什么人工智能学习要用Python&#xff1f;运行速度慢不好之类的&#xff0c;今天就让博主谈谈自己的感受。 先来说说前景 随着“大数据”“云计算”“人工智能”等等科技的兴起&#xff0c;IT行业在今后三到五年将会迎来一个高速发展期。这也就意…

QT调用python传递图像和二维数组,并接受python返回值(图像)

任务目的&#xff1a; 用QT调用python代码&#xff0c;将QT读取的图像(Mat矩阵)作为参数传入python中&#xff0c;将QT的二维数组作为参数传递给python&#xff0c;python接收QT传入的图像进行计算&#xff0c;将结果返回给QT。 实现过程 1.新建QT项目 说明&#xff1a;QT的…

[Cortex-M3]-5-cache uncache

目录 1 cache的引入 2 cache的工作原理 3 cache使用限制 1 cache的引入 程序运行的流程&#xff08;很简单&#xff09;&#xff1a; 程序编译&#xff1a;存放在flash&#xff1b;程序加载&#xff1a;程序加载到内存&#xff1b;程序运行&#xff1a;指令从内存复制到CP…

【产品人卫朋】自媒体运营的5个阶段,以及增长策略

本篇内容以微信公众号为例讲解自媒体的运营策略。 建立一个快速发展的微信公众号&#xff0c;需要多长时间呢&#xff1f; 有些人在一年内就可以建立一个蓬勃发展的公众号&#xff0c;而其他人则可能需要两年、三年甚至是五年的时间。 在发展的过程中&#xff0c;你的公众号将经…

阿里工程师告诉你,0基础如何自学python进大厂

大概一年前这个朋友就想学习Python了&#xff0c;但因为工作比较忙&#xff0c;而且觉得Python肯定不太好学&#xff0c;所以一直搁置在那里。 宅家学Python 到了今年1月28日也就是大年初三的时候&#xff0c;眼看新冠肺炎疫情不会短时间结束了&#xff0c;全国各地都在严控&…

原型模式

开始原型模式前&#xff0c;我们要知道深拷贝的定义&#xff0c;因为原型模式中的克隆操作核心就是深拷贝。 深拷贝和浅拷贝 下图为浅拷贝(即是编译器的默认版本), 只拷贝了指针, 两个指针同时指向一个内存, 会有危险(a改变时b也改变, 称为别名) , 导致内存泄漏 调用strcpy复…

Qt扫盲-QTreeWidget理论总结

QTreeWidget理论总结1. 简述2. QTreeWidgetItem 简述3. 头标签4. 常用功能5. 槽函数6. 信号1. 简述 QTreeWidget 类是Qt提供了一个标准的树部件&#xff0c;该部件具有经典的基于 Item 的界面&#xff0c;每个Item都是一个 QTreeWidgetItem。这个标准的树控件不需要model/view…

文科女进德国IBM实习做程序媛,我是怎么办到的?

很快我在IBM德国区实习的第一个月就要结束了。 作为一个土生土长的文科生&#xff08;硕士语言学、本科语言学商科&#xff09;&#xff0c;现在竟在欧洲混入跨国科技公司做编程技术类实习生&#xff0c;我自己也挺意外的。 尽管只是一点点微不足道的个人经历&#xff0c;此时…

教你如何优雅的转行Python程序员,一学就会

在实际的工作中&#xff0c;我们经常发现&#xff0c;很多朋友在某一个工作中做了一段时间&#xff0c;发现自己越做越没兴趣&#xff0c;越做越不开心&#xff0c;想跳不敢跳&#xff0c;想辞不敢辞&#xff0c;最后影响了自己的本职工作&#xff0c;陷入两难的窘境。 其实&am…

【Qbot】3.加入内容审核功能

该项目计划长期进行维护更新&#xff0c;欢迎star&#xff1a;https://github.com/zstar1003/Qbot 前言 在ChatGPT上线Q群不久&#xff0c;不少人对其进行了测试&#xff0c;但随着时间的延续&#xff0c;测试话题逐渐走向失控&#xff0c;迫使我不得不紧急暂停。 对同胞素质的…

【Python百日进阶-数据分析】Day134 - plotly饼图:go.pie()实例

文章目录4.2 go.Pie() 的基本饼图4.2.1 基本饼图4.2.2 样式饼图4.2.3 使用 uniformtext 控制文本字体大小4.2.4 控制饼图中的文本方向4.2.5 甜甜圈图4.2.6 从中心拉出扇区4.2.7 子图中的饼图4.2.8 自定义颜色集的子图4.2.9 绘制面积与总计数成比例的图表4.2.10 旭日图4.2.11 Da…

电脑软件、微信多开

因为办公需要在电脑上登录 2 个微信&#xff0c;但是直接双击微信图标只有 1 个登录界面&#xff0c;无法是现实登录 2 个微信。那么怎么才能在 1 个电脑上打开 2 个微信&#xff0c;方法有四种&#xff1a;1、安装&#xff1b;2、Enter&#xff1b;3、连续点击&#xff1b;4、…

『 canvas 特效』一文教你绘制绚丽的星空背景 TS + ES6

介绍 很久没有写关于 canvas 效果的文章了&#xff0c;刚好最近又学到了一个新的特效&#xff0c;使用 canvas 绘制多层次动态星空背景&#xff0c;今天就分享给大家。首先我们依旧来看一下最终实现的效果&#xff0c;如图所示&#xff1a; 由于录制 GIF 造成失帧&#xff0c;…

实战三十三:STAMP算法实现商品推荐实战 代码+数据

1.案例知识点 推荐系统任务描述:通过用户的历史行为(比如浏览记录、购买记录等等)准确的预测出用户未来的行为;好的推荐系统不仅如此,而且能够拓展用户的视野,帮助他们发现可能感兴趣的却不容易发现的item;同时将埋没在长尾中的好商品推荐给可能感兴趣的用户。STAMP推荐…