Loading [MathJax]/jax/input/TeX/config.js
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >Linux网络编程TCP

Linux网络编程TCP

作者头像
DeROy
发布于 2021-11-16 08:30:03
发布于 2021-11-16 08:30:03
5.9K01
代码可运行
举报
文章被收录于专栏:编程学习基地编程学习基地
运行总次数:1
代码可运行

TCP/IP协议

TCP/IP 协议栈是一系列网络协议(protocol)的总和,是构成网络通信的核心骨架,它定义了电子设备如何连入因特网,以及数据如何在它们之间进行传输。

OSI 7层模型和TCP/IP四层网络模型对应关系

计算机网路基础的知识不过多讲解,主要是让大家明白接下来的Linux网络编程数据流属于那一层,具体如下图

TCP/IP协议数据流示意图

我们接下来讲解的Linux网络编程Tcp协议是属于传输层的协议

Linux Socket 网络编程

TCP协议

TCP是面向连接的可靠的传输层协议。

TCP编程

Linux中的网络编程是通过socket接口来进行的。socket是一种特殊的I/O接口,它也是一种文件描述符。常用于不同机器上的进程之间的通信,当然也可以实现本地机器上的进程之间通信。

使用TCP协议的流程图

根据流程图逐一讲解API接口.

服务端API接口

socket
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
#include <sys/socket.h>
int socket(int family //协议簇 一般 AF_INET PF_INET
 ,int type   //套接口类型 SOCK_STREAM(字节流套接口)
 ,int protocol);  //非原始套接口,参数为 0

套接口类型:

- SOCK_STREAM(字节流套接口)

- SOCK_DGRAM(数据报套接口)

- SOCK_RAW(原始套接口)

示例:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
listenfd = socket(AF_INET,SOCK_STREAM,0);
bind

为套接字分配一个本地IP和协议端口

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
#include <sys/socket.h>
int bind(int socket
, const struct sockaddr *address//协议族地址
,socklen_t address_len); //协议族长度  

- address: 协议族地址,通用的socket地址

通用的socket地址不是很好用,所以Linux为各个协议族提供了专门的socket地址结构体

UNIX本地协议族
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
struct sockaddr_un {
 sa_family_t sa_family;
 char sun_path[100];
}
TCP/IP协议族

TCP/IP协议族有sockaddr_insockaddr_in6两个专用的socket地址结构体,分别对应IPv4IPv6

IPv4对应的协议族sockaddr_in定义如下

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
struct sockaddr_in {
 sa_family_t sin_family; /*地址族:AF_INET*/
 in_port_t sin_port; /*网络字节序表示的端口号*/
 struct in_addr sin_addr; /*ipv4地址*/
}
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
/* Internet address. */
struct in_addr {
    uint32_t       s_addr;     /* address in network byte order */
};

ipv6用的比较少就不单独介绍定义了

常用的有sockaddr_in(网络地址),sockaddr_un(本地地址), 传入参数时要强制转换为sockaddr*指针类型,示例如下。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
struct sockaddr_in servaddr;
/*(2) 设置服务器协议族sockaddr_in结构*/
bzero(&servaddr,sizeof(servaddr));
servaddr.sin_family = AF_INET;//必须和套接字的创建fimile一致
servaddr.sin_addr.s_addr = htonl(INADDR_ANY);//表明可接受任意IP地址
servaddr.sin_port = htons(8888);
bind(listenfd,(struct sockaddr*)&servaddr,sizeof(servaddr));
listen

listen函数仅被TCP服务器调用

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
#include <sys/socket.h>
int listen(int sockfd //socket函数返回的套接口描述字
 ,int backlog);   //则此值表示listen时的队列大小,最大连接个数
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
listen(listenfd,2);
accept
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
#include <sys/socket.h>         
int accept(int listenfd   //socket 函数返回的套接口描述字 监听句柄
 , struct sockaddr *client //协议族地址
 , socklen_t * addrlen);  //客户端 套接字
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
accept(listenfd , (struct sockaddr *)&cliaddr , &clilen);

用于接收一个新的连接。

客户端API接口

connect
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
#include <sys/socket.h>      
  int connect(int sockfd
   , const struct sockaddr * addr //协议族地址
   , socklen_t addrlen);   //协议族长度  
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
connect(sockfd , (struct sockaddr *)&servaddr , sizeof(servaddr))

用于连接到服务器

send/recv
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
#include <sys/types.h>
#include < sys/socket.h >         
ssize_t send(int sockfd, const void *buf, size_t len, int flags); 
ssize_t recv(int sockfd, void *buf, size_t len, int flags);

前3个参数与read()相同,参数flags是传输控制标志,UDP再做纤细介绍.

TCP案例

服务端
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
#include<stdio.h>
#include<stdlib.h>
#include<sys/socket.h>
#include<sys/types.h>
#include<unistd.h>
#include<string.h>
#include<netinet/in.h>
#include<netdb.h>
#include<arpa/inet.h>
#define MAX_BUFF 1024
#define MAX_LISTEN 10
int main(int argc,char *argv[])
{
  int defaule_port = 8000;
  int optch = 0;
  while((optch = getopt(argc, argv, "s:p:")) != -1)
  {
    switch (optch)
    {
      case 'p':
          defaule_port = atoi(optarg);
          printf("port: %s\n", optarg);
          break;
      case '?':
          printf("Unknown option: %c\n",(char)optopt);    
          break;
      default:
          break;
    }
  }
 /*声明服务器地址和客户链接地址*/
 struct sockaddr_in server_addr,client_addr;
 socklen_t client_len;

 /*声明服务器监听套接字和客户端链接套接字*/
 int listen_fd,connect_fd;

  /*(1) 初始化监听套接字listenfd*/
 listen_fd = socket(AF_INET, SOCK_STREAM,0);
 if(listen_fd == -1)
 {
   perror("Socket Error:");
   return 0;
 }
    
 /*(2) 设置服务器sockaddr_in结构*/
 bzero(&server_addr,sizeof(server_addr));
 server_addr.sin_family = AF_INET;
 server_addr.sin_addr.s_addr = htonl(INADDR_ANY);    //任意地址
 server_addr.sin_port = htons(defaule_port);

 /*(3) 绑定套接字和端口*/
 if(bind(listen_fd,(struct sockaddr*)&server_addr,sizeof(server_addr))==-1)
 {
   perror("Bind error:");
   return 0;
 }

 /*(4) 监听客户请求*/
 if(listen(listen_fd,MAX_LISTEN)==-1)
 {
   perror("Listen error:");
   return 0;
 }
 
 /*(5) 接受客户请求*/
 for(;;)
 {
   client_len = sizeof(client_addr);
   connect_fd = accept(listen_fd,(struct sockaddr*)&client_addr,&client_len);
   if(connect_fd < 0)
   {
     perror("accept error");
     return 0;
   }

   printf("Connect from %s:%u...\n",inet_ntoa(client_addr.sin_addr),ntohs(client_addr.sin_port));
   /*声明缓冲区,向客户端发送数据*/
   char buff[MAX_BUFF] = "hello\n";
   if(-1 == write(connect_fd,buff,strlen(buff)))
   {
     perror("Send error\n");
     return 0;
   }
   printf("Send success...\n");
   /*清空缓冲区,阻塞等待读取客户端发过来的数据*/
   memset(buff,'\0',sizeof(buff));
   if(-1 == read(connect_fd,buff,MAX_BUFF))
   {
     perror("read error\n");
     return 0;
   }
   write(1,buff,strlen(buff));
   close(connect_fd);
  }
  close(listen_fd);
  return 0;
}

编译运行,默认8000端口,-p 指定端口

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
gcc -o server.c server
./server -p 8020
客户端
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <unistd.h>
#include <errno.h>
#include <netinet/in.h>
#include <netdb.h>
#include <arpa/inet.h>
const int MAX_LINE = 2048;
int main(int argc ,char**argv)
{
    /*声明套接字和链接服务器地址*/
  int sockfd;
  int optch,ret = -1;
  const char*server_addr;
  int default_port = 8000;

  struct sockaddr_in servaddr;
  /*判断是否为合法输入 必须传入一个参数:服务器Ip*/
  if(argc<3)
  {
    printf("usage:tcpcli <IPaddress>");
    return 0;
  }
  while((optch = getopt(argc, argv, "s:p:")) != -1)
  {
    switch (optch)
    {
        case 's':
            server_addr = optarg;
            break;
        case 'p':
            default_port = atoi(optarg);
            printf("port: %s\n", optarg);
            break;
        case '?':
            printf("Unknown option: %c\n",(char)optopt);    
            break;
        default:
            break;
    }
  }
  /*(1) 创建套接字*/
  sockfd =socket(AF_INET,SOCK_STREAM,0);
  if(sockfd==-1)
  {
      perror("socket error");
      return 0;
  }
  /*(2) 设置链接服务器地址结构*/
  bzero(&servaddr,sizeof(servaddr));
  servaddr.sin_family =AF_INET;
  servaddr.sin_port = htons(default_port);
  if(inet_pton(AF_INET , server_addr , &servaddr.sin_addr) < 0)
  {
    printf("inet_pton error for %s\n",server_addr);
    return 0;
  }
  /*  (3) 发送链接服务器请求  */
  if( (ret = connect(sockfd , (struct sockaddr *)&servaddr , sizeof(servaddr))) < 0)
  {
    perror("connect error");
    return 0;
  }
  printf("connect seccess,ret:%d..\n",ret);
  struct sockaddr_in c_addr;
  memset(&c_addr, 0, sizeof(c_addr));
  socklen_t len = sizeof(c_addr);

  char buf[MAX_LINE];
  while (1)
  {
      /* code */
      memset(buf,'\0',sizeof(buf));
      len = read(sockfd,buf,sizeof(buf)-1);
      if(len == 0)
      {
          printf("server close..\n");
          return 0;
      }
      printf("recv from server:%s",buf);

      memset(buf,'\0',sizeof(buf));
      printf("please enter:\n");
      ssize_t len = read(0,buf,sizeof(buf)-1);
      if(len>0)
      {
          if(strcmp(buf,"quit")==0)
          {
              printf("quit\n");
              break;
          }
          buf[len - 1]='\0';
          write(sockfd,buf,strlen(buf));
      }
  }
  close(sockfd);
  return 0;
}

编译运行,默认8000端口,-s 指定连接的服务器 -p 指定端口

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
$ gcc -o client client.c
$ ./client -s 0.0.0.0 -p 8020
connect seccess,ret:0..
recv from server:hello
please enter:
hello too
server close..

简单 tcp服务器和客户端就到这里,下期介绍多线程技术,实现一个多线程的聊天室程序。

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2021-11-05,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 编程学习基地 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
UDP&TCP Linux网络应用编程详解
暂时想不出什么好的应用场景, 目前想到目标就是实现让两个设备通过网络传输数据, 比如开发板和Linux主机之间传数据, 以后就可以实现开发板通过网络上报数据或者主机通过网络控制开发板。
韦东山
2020/09/30
5.9K0
UDP&TCP Linux网络应用编程详解
linux网络编程系列(三)--tcp和udp的基本函数调用过程及如何选择
TCP是TCP/IP体系中面向连接的传输层协议,它提供全双工和可靠交付的服务。它采用许多机制来确保端到端结点之间的可靠数据传输,如采用序列号、确认重传、滑动窗口等。
cpp加油站
2021/04/16
1K0
linux网络编程系列(三)--tcp和udp的基本函数调用过程及如何选择
提升性能的必备技术:Linux网络IO与select详解
IO 即“Input”和“Output”的组合,即输入/输出,IO用来处理设备之间的数据传输。socket/fd也是一种IO。
Lion 莱恩呀
2024/08/10
1930
提升性能的必备技术:Linux网络IO与select详解
TCP网络编程中connect()、listen()和accept()三者之间的关系 ( 非常重要!!)
基于 TCP 的网络编程开发分为服务器端和客户端两部分,常见的核心步骤和流程如下:
战神伽罗
2019/09/04
3K0
TCP网络编程中connect()、listen()和accept()三者之间的关系 ( 非常重要!!)
[置顶] linux网络编程之socket(十四):基于UDP协议的网络程序
一、下图是典型的UDP客户端/服务器通讯过程 下面依照通信流程,我们来实现一个UDP回射客户/服务器   #include <sys/types.h>  #include <sys/socket.h
s1mba
2017/12/28
1.9K0
[置顶]
        linux网络编程之socket(十四):基于UDP协议的网络程序
【Socket】简单的回声服务器实现
网络通信 Socket通信的3要素 通信的目的地址 使用的端口号 使用的传输层协议(如TCP、UDP) ---- Socket通信模型 简单的回声服务器实现 服务端 #include <stdio.h> #include <unistd.h> #include <sys/types.h> #include <sys/socket.h> #include <string.h> #include <ctype.h> #include <arpa/inet.h> #define SERVER_PORT
半生瓜的blog
2023/05/13
6200
【Socket】简单的回声服务器实现
day01-从一个基础的socket服务说起
在linux中,一切都是文件,所有文件都有一个int类型的编号,称为文件描述符。服务端和客户端通信本质是在各自机器上创建一个文件,称为socket(套接字),然后对该socket文件进行读写。
会玩code
2022/04/24
1.3K2
day01-从一个基础的socket服务说起
【网络编程】Linux网络编程基础与实战第二弹——Socket编程
Socket本身有“插座”的意思,在Linux环境下,用于表示进程间网络通信的特殊文件类型。本质为内核借助缓冲区形成的伪文件。
洁洁
2023/10/19
5020
【网络编程】Linux网络编程基础与实战第二弹——Socket编程
socket网络编程(一)——初识socket
出现一项技术,首先我们弄懂一下,为什么要出现。那么为什么要出现socket这玩意呢?可以很简单的用一句话来概括:
一点sir
2024/01/10
4490
socket网络编程(一)——初识socket
1-UNIX网络编程-Socket套接字编程简介
触发学习UNIX网络编程的动力在于前段时间需要开发一个接入服务,需要考虑比较高的并发处理能力,且尽量少占用的机器资源,选用了JAVA的Netty框架,学习过程产生不少疑问,限于基础知识太薄弱无法理解原理,所以开始关注UNIX编程。
zoujunjie202
2022/07/27
1.3K0
1-UNIX网络编程-Socket套接字编程简介
SOCKET网络编程 (通俗易懂入门篇)
struct sockaddr :很多网络编程函数的出现早于IPV4协议,为了向前兼容,现在sockaddr都退化成(void *)结构了。 传递一个地址给函数,然后由函数内部再强制类型转换为所需的地址类型。
看、未来
2020/08/25
1.2K0
SOCKET网络编程 (通俗易懂入门篇)
Linux网络编程之搭建服务器和客户端(一)
说明:这里我们可以看到IP地址点分十进制表示方法被转换成一般形式(这里打印出的是十六进制表示法)。
用户6280468
2022/03/21
1.4K0
Linux网络编程之搭建服务器和客户端(一)
【网络编程系列】二:socket通信原理及实践
我们深谙信息交流的价值,那网络中进程之间如何通信,如我们每天打开浏览器浏览网页时,浏览器的进程怎么与web服务器通信的?当你用QQ聊天时,QQ进程怎么与服务器或你好友所在的QQ进程通信?这些都得靠socket?那什么是socket?socket的类型有哪些?还有socket的基本函数,这些都是本文想介绍的。本文的主要内容如下: 1、网络中进程之间如何通信? 2、Socket是什么? 3、socket的基本操作 3.1、socket()函数 3.2、bind()函数 3.3、l
老白
2018/03/19
1.8K0
【网络编程系列】二:socket通信原理及实践
socket网络编程(五)——粘包拆包问题
假设一个这样的场景,客户端要利用send()函数发送字符“asd”到服务端,连续发送3次,但是服务端休眠10秒之后再去缓冲池中接收。那么请问10秒之后服务端从缓冲区接收到的信息是“asd”还是“asdasdasd”呢?如果大家有去做实验的话,可以知道服务端收到的是“asdasdasd”,为什么会这样呢?按正常的话,服务端收到的应该是“asd”,剩下的两个asd要不就是收不到要不就是下次循环收到,怎么会一次性收到“asdasdasd”呢?如果要说罪魁祸首的话就是那个休眠10秒,导致数据粘包了!
一点sir
2024/01/10
3740
socket网络编程(五)——粘包拆包问题
实现基于TCP的服务端/客户端
客户端调用connect函数后服务端接收连接请求(记录到等待队列)或发生断网等异常情况而中断连接请求才会返回(完成函数调用)。客户端的IP地址和端口号在调用connect函数时由内核自动分配,无需调用标记的bind函数进行分配。
玖柒的小窝
2021/12/15
1.2K0
实现基于TCP的服务端/客户端
服务器线程并发和进程并发
进程和线程的使用在前面博文已经讲述完毕,在完成一个最简单的服务器之后,就是要考虑下如何实现并发服务器了。 要实现服务的并发,只能通过进程和线程两种方式。 之前提到过listen_fd和connect_fd,listen用于监听是否有客户端连接,维护两个fd队列,没完成握手的和完成就绪的。 connect从就绪队列取描述符,这个connect_fd描述符将用于数据通信,所以要实现并发,就是将connect_fd分发到线程或进程上,由他们去独立完成通信。 在实际并发服务器应用场合,在IO层大多通过两个
Aichen
2018/05/18
3.2K0
socket网络编程(六)——心跳包问题
问大家一个问题,如果客户端和服务端长时间没有相互发送数据的话,那么我们怎么来判断这个连接是否存在的呢?有些人可能很自然地说直接send一下不就可以了,确实可以这样进行判断,那么我们发送的时候可以选择发送任何东西,所以一般都是发送一个空包,这个就是心跳包。
一点sir
2024/01/10
5330
Linux下TCP网络编程-创建服务器与客户端
互联网概念诞生于20世纪60年代末,从9几年中国接入互联网开始到现在,生活的每个角落都能看到网络的使用。现在物联网时代、共享经济的到来,生活中不仅仅电脑、手机可以接入网络,身边的各个设备也能接入互联网了。 比如:市政路灯、污水井盖、家用电器,汽车等等。
DS小龙哥
2022/05/11
7.1K0
Linux下TCP网络编程-创建服务器与客户端
网络编程-一个简单的echo程序(0)
在上一篇《网络编程-从TCP连接的建立说起》中简单介绍了TCP连接的建立,本文暂时先抛开TCP更加详细的介绍,来看看如何实现一个简单的网络程序。
编程珠玑
2019/07/12
5990
网络编程-一个简单的echo程序(0)
Linux实验八:流式套接字编程
1、深入理解 TCP/IP 模型相关概念; 2、掌握流式套接字 socket 相关的数据结构,如 sokkaddr,sockaddr_in 等; 3、掌握流式套接字通讯时常用的函数,如 socket,connect,listen,accept 等。
Francek Chen
2025/01/23
3660
Linux实验八:流式套接字编程
推荐阅读
相关推荐
UDP&TCP Linux网络应用编程详解
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验