Linux TCP编程流程

news2024/12/30 2:33:38

一、TCP编程流程

TCP 提供的是面向连接的、可靠的、字节流服务。TCP的服务器端和客户端编程流程如下:

在这里插入图片描述

1.socket()方法

用来创建一个套接字,有了套接字就可以通过网络进行数据的收发。这也是为什么进行网络通信的程序首先要创建一个套接字。创建套接字时要指定使用的服务类型,使用基于TCP协议的流式服务(SOCK_STREAM)。

2.bind()方法

用来指定套接字使用的IP地址和端口。IP地址就是自己主机的地址,如果主机没有接入网络,测试程序时可以使用回环地址“127.0.0.1”。端口是一个16位的整形值,一般0-1024 为知名端口,如HTTP使用的80号端口。这类端口一般用户不能随便使用。其次,1024-4096 为保留端口,用户一般也不使用。4096以上为临时端口,用户可以使用。在Linux 上,1024 以内的端口号,只有root用户可以使用。

3.listen()方法

用来创建监听队列。监听队列有两种,一个是存放未完成三次握手的连接,一种是存放已完成三次握手的连接。listen()第二个参数就是指定已完成三次握手队列的长度。

4.accept()方法

处理存放在 listen 创建的已完成三次握手的队列中的连接。每处理一个连接,则accept()返回该连接对应的套接字描述符。如果该队列为空,则accept阻塞。

5.connect()方法

一般由客户端程序执行,需要指定连接的服务器端的IP地址和端口。该方法执行后,会进行三次握手, 建立连接。

在这里插入图片描述

6.send()方法

向TCP连接的对端发送数据。send()执行成功,只能说明将数据成功写入到发送端的发送缓冲区中,并不能说明数据已经发送到了对端。send()的返回值为实际写入
到发送缓冲区中的数据长度。

7.recv()方法

接收TCP连接的对端发送来的数据。recv()从本端的接收缓冲区中读取数据,如果接收缓冲区中没有数据,则recv()方法会阻塞。返回值是实际读到的字节数,如果
recv()返回值为 0, 说明对方已经关闭了TCP连接。
close()方法用来关闭TCP连接。此时,会进行四次挥手。

在这里插入图片描述
二、服务器端和客户端连通

服务器端代码ser.c如下:

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<arpa/inet.h>

int main()
{
    //1.创建套接字
    int sockfd =socket(AF_INET,SOCK_STREAM,0);
    //第一个参数:协议族,AF_INET代表IPV4网络协议
    //第二个参数:套接字的服务类型,SOCK_STREAM代表基于TCP协议的流式服务的套接字
    //第三个参数:0,表示使用默认协议

    if(sockfd==-1)
    {
        printf("创建失败\n");
        exit(1);
    }

    struct sockaddr_in saddr,caddr;//定义服务端和客户端的套接字地址
    memset(&saddr,0,sizeof(saddr));//套接字在使用之前必须清空
    saddr.sin_family=AF_INET;
    saddr.sin_port=htons(6000);//短整型主机字节序转网络字节序
    saddr.sin_addr.s_addr = inet_addr("127.0.0.1");//ip地址

    //2.指定套接字地址
    int res=bind(sockfd,(struct sockaddr*)&saddr/*将专用的套接字地址强转为通用的地址*/,sizeof(saddr));
    //第一个参数:需要绑定的套接字描述符
    //第二个参数:指向结构体变量saddr,并强转为struct sockaddr类型,然后将ip和端口传给bind函数进行绑定
    //第三个参数:第二个参数所指向的结构体的大小,即套接字地址的长度
    if(res==-1)
    {
        printf("绑定失败\n");
        exit(1);
    }

    //3.创建监听队列,存放要连接的客户端
    res = listen(sockfd,5);
    //第一个参数:被监听的套接字描述符
    //第二个参数:表示处于完全连接状态的套接字的上限
    if(res==-1)
    {
        exit(1);
    }

    while(1)
    {
        socklen_t len=sizeof(caddr);
        //4.接受客户端的连接
        int c=accept(sockfd,(struct sockaddr*)&caddr,&len);
        //如果accept成功,返回一个新的套接字描述符c与客户端通信,这个新的套接字描述符是内核自动生成的
        //第一个参数:是服务器端的套接字描述符
        //第二个参数:用于返回客户端的套接字地址cadrr
        //第三个参数:客户端套接字地址的长度

        if(c<0)
        {
            continue;
        }

        printf("accept c=%d,ip=%s,port=%d\n",c,inet_ntoa(caddr.sin_addr),ntohs(caddr.sin_port));
        char buff[128]={0};

        //5.接收客户端的数据
        recv(c,buff,127,0);
        //第一个参数:已连接客户端的那个新的套接字描述符
        //第二个参数:指定接收客户端数据的位置
        //第三个参数:指定接收客户端发来的数据的大小
        //第四个参数:一般设置为0

        printf("buff=%s\n",buff);

        //6.向客户端发送反馈数据
        send(c,"ok",2,0);
        //第一个参数:已连接客户端的那个新的套接字描述符
        //第二个参数:指定向客户端发送数据的位置,也可以直接指定内容
        //第三个参数:指定向客户端发送数据的大小
        //第四个参数:一般设置为0

        //7.关闭与客户端通信的套接字
        close(c);
    }



}

客户端代码cli.c如下:

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<arpa/inet.h>

int main()
{
    //1.创建套接字
    int sockfd=socket(AF_INET,SOCK_STREAM,0);
    if(sockfd==-1)
    {
        exit(1);
    }

    struct sockaddr_in saddr;
    memset(&saddr,0,sizeof(saddr));
    saddr.sin_family=AF_INET;
    saddr.sin_port=htons(6000);
    saddr.sin_addr.s_addr=inet_addr("127.0.0.1");

    //2.向服务器端发起连接
    int res=connect(sockfd,(struct sockaddr*)&saddr,sizeof(saddr));
    //第一个参数:客户端套接字的描述符
    //第二个参数:服务器套接字的地址
    //第三个参数:服务器套接字地址的大小
    if(res==-1)
    {
        printf("连接失败\n");
        exit(1);
    }

    printf("输入:");

    char buff[128]={0};
    fgets(buff,128,stdin);

    //3.向服务器端发送数据
    send(sockfd,buff,strlen(buff),0);
    //第一个参数:已被服务器端连接的客户端的套接字描述符
    //第二个参数:指定向服务器端发送的数据的位置
    //第三个参数:指定向服务器端发送的数据的大小
    //第四个参数:一般设置为0
    memset(buff,0,sizeof(buff));

    //4.接收服务器反馈回来数据
    recv(sockfd,buff,127,0);
    //第一个参数:已被服务器端连接的客户端的套接字描述符
    //第二个参数:指定服务器端反馈回来的数据的位置
    //第三个参数:指定服务器端反馈回来的数据的大小
    //第四个参数:一般设置为0

    printf("buff=%s\n",buff);

    //5.关闭连接
    close(sockfd);

    exit(0);
}

运行结果:

先编译运行ser.c,使服务器端启动:

在这里插入图片描述

再打开另一个终端,编译运行cli.c,使客户端启动,此时服务器端显示已经接收到了客户端的连接:

在这里插入图片描述

然后客户端向服务器端发送数据:

在这里插入图片描述
如上图所示,客户端向服务端发送信息"hello",服务器端向客户端反馈信息"ok"。

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

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

相关文章

SqlServer2019—解决SQL Server 无法连接127.0.0.1的问题

1、打开SQL Server 2019配置管理器 2、SQL Servere 网络配置(启用 Named Pipes 和 TCP/IP) 3、修改TCP/IP协议(右键选择属性—IP地址)&#xff0c;具体如下图所示&#xff1a; 4、重启SQL Server服务

QT通过ODBC连接GBase 8s数据库(Windows)示例

示例环境&#xff1a; 操作系统&#xff1a;Windows 10 64位数据库及CSDK版本&#xff1a;GBase 8s V8.8_3.0.0_1 64位QT&#xff1a;5.12.0 64位 1&#xff0c;CSDK安装及ODBC配置 1.1&#xff0c;免安装版CSDK 下载免安装版的CSDK驱动&#xff0c;地址&#xff1a;https:…

盛最多水的容器——力扣11

int maxArea(vector<int>& height) {int l=0, r=height.size()

企业如何通过CRM系统获得更多商机?

商机是销售的基础&#xff0c;如何获取更多的商机&#xff0c;是每个企业都关心的问题。通过传统的营销方式来吸引潜在客户&#xff0c;往往成本高、效果低、难以管理。那么有没有一种方式&#xff0c;既节省时间和成本&#xff0c;又可以获取更多商机&#xff1f;当然有&#…

AUTOSAR配置与实践(配置篇) 如何条件控制PDU外发

AUTOSAR配置与实践(配置篇)如何条件控制PDU外发 一、需求1.1 需求简要分析1.2 需求进一步分析二、流程实现和具体配置一、需求 需要针对特定的PDU(外发)进行条件控制,这里要通过不同配置字进行PDU是否外发的控制 1.1 需求简要分析 正常PDU分组时分为两组,接收报文组和…

Javase(中)

一、面向对象进阶 1、static&#xff08;静态是共享的&#xff09; 静态变量 特点&#xff1a;修饰的成员变量被该类的所有对象共享 静态方法 静态方法只能调用静态的东西&#xff0c;非静态方法可以调用所有 static 的注意事项 static的内存图 重识main方法 2、工具类、测…

23.8.11.用apifox端口号与java接口链接的时候少了个/导致连接不成功。

用apifox端口号与java接口链接的时候少了个/导致连接不成功。 原因分析&#xff0c;因为拼接的位置少了个/ 如图所示

电工-三相异步电动机启动电流很大的原因与不利影响

电工-三相异步电动机启动电流很大的原因与不利影响 所谓电动机的启动是指电动机从接入电源开始转动起&#xff0c;到达到额定转速为止的这一过程。 根据理论分析和实际测定&#xff0c;异步电动机启动瞬间&#xff1b;定子绕组启动电流很大&#xff0c;可达额定值的4~7倍。为什…

家庭智慧管控中心——酷开系统全时AI

智能家居入口在哪&#xff1f; 或许你家已经有了&#xff01;举个例子&#xff0c;智能电视就是其中的一种&#xff01;智能家电的变革进度逐渐加深&#xff0c;人们对于智慧生活的需求也逐渐提高。伴随着人工智能、物联网、大数据等信息技术的发展&#xff0c;家电产品的智能化…

Camunda 7.x 系列【23】身份服务 IdentityService

有道无术,术尚可求,有术无道,止于术。 本系列Spring Boot 版本 2.7.9 本系列Camunda 版本 7.19.0 源码地址:https://gitee.com/pearl-organization/camunda-study-demo 文章目录 1. 概述2. 相关表3. IdentityService1. 概述 在之前我们演示了一个简单的请假流程,接下来…

上篇——税收大数据应用研究

财税是国家治理的基础和重要支柱&#xff0c;税收是国家治理体系的重要组成部分。我们如何利用税收数据深入挖掘包含的数据价值&#xff0c;在进行数据分析&#xff0c;提升税收治理效能&#xff0c;推进税收现代化。 1. 定义与特点 对于“大数据”&#xff08;Big data&#…

使用秘籍|如何实现图数据库 NebulaGraph 的高效建模、快速导入、性能优化

本文整理自 NebulaGraph PD 方扬在「NebulaGraph x KubeBlocks」meetup 上的演讲&#xff0c;主要包括以下内容&#xff1a; NebulaGraph 3.x 发展历程NebulaGraph 最佳实践 建模篇导入篇查询篇 NebulaGraph 3.x 的发展历程 NebulaGraph 自 2019 年 5 月开源发布第一个 alp…

论AI与大数据之间的关系

前言 在21世纪&#xff0c;"AI"和"大数据"已经成为科技领域的热门词汇。它们不仅是创新的代名词&#xff0c;更是现代技术发展的双翼。然而&#xff0c;很多人对于AI与大数据之间的关系仍然停留在表面的理解。本文旨在深入探讨这两者之间的深厚关系&#…

【Python机器学习】实验14 手写体卷积神经网络

文章目录 LeNet-5网络结构&#xff08;1&#xff09;卷积层C1&#xff08;2&#xff09;池化层S1&#xff08;3&#xff09;卷积层C2&#xff08;4&#xff09;池化层S2&#xff08;5&#xff09;卷积层C3&#xff08;6&#xff09;线性层F1&#xff08;7&#xff09;线性层F2 …

C# 反编译dll文件,将dll文件转为cs文件

工具名&#xff1a;reflector&#xff0c;可免费使用 工具下载地址使用方法 工具下载地址 下载地址&#xff1a; https://www.red-gate.com/products/reflector/ 使用方法 下载安装&#xff1b; 打开软件 打开dll文件 侧边栏找打刚打开的dll文件&#xff0c;直接就能导…

操作教程|通过1Panel开源Linux面板快速安装DataEase

DataEase开源数据可视化分析工具&#xff08;dataease.io&#xff09;的在线安装是通过在服务器命令行执行Linux命令来进行的。但是在实际的安装部署过程中&#xff0c;很多数据分析师或者业务人员经常会因为不熟悉Linux操作系统及命令行操作方式&#xff0c;在安装DataEase的过…

论文及代码详解——Restormer

文章目录 论文详解Overall pipelineMulti-Dconv Head Transposed AttentionGated-Dconv Feed-Forward Network 代码详解 论文&#xff1a;《Restormer: Efficient Transformer for High-Resolution Image Restoration》 代码&#xff1a;https://github.com/swz30/Restormer 论…

LRU淘汰策略执行过程

1 介绍 Redis无论是惰性删除还是定期删除&#xff0c;都可能存在删除不尽的情况&#xff0c;无法删除完全&#xff0c;比如每次删除完过期的 key 还是超过 25%&#xff0c;且这些 key 再也不会被客户端访问。 这样的话&#xff0c;定期删除和堕性删除可能都彻底的清理掉。如果…

Nodejs 第十三章(os)

Nodejs os 模块可以跟操作系统进行交互 var os require("node:os")序号API作用1os.type()它在 Linux 上返回 Linux&#xff0c;在 macOS 上返回 Darwin&#xff0c;在 Windows 上返回 Windows_NT2os.platform()返回标识为其编译 Node.js 二进制文件的操作系统平台的…

AI流程图教程,小白也能轻松画!

流程图是一种图形化的工具&#xff0c;用于展示和描述一个过程中的各个步骤、活动和决策。它通过使用不同的图形符号和箭头表示&#xff0c;清晰地展示了流程的顺序和流动路径。流程图通常用于可视化和沟通复杂的业务流程、系统操作流程或项目流程。它帮助人们更好地理解和分析…