首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >Linux网络编程TCP

Linux网络编程TCP

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

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 删除。

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

评论
登录后参与评论
2 条评论
热度
最新
求继续更新
求继续更新
回复回复1举报
666
666
回复回复1举报
推荐阅读
解决移动端适配rem单位在华为和三星手机出现bug
github代码地址:https://github.com/Miofly/mio.git
用户10106350
2022/10/28
7590
移动端页面自适应解决方案:rem 布局篇
假设设计妹妹给我们的设计稿尺寸为750 * 1340。结合网易、淘宝移动端首页html元素上的font-size属性,html5设计稿尺寸以及前端与设计之间协作流程一般分为下面两种。
用户1272076
2019/03/26
2.4K0
手机端页面自适应布局---rem
如果页面的宽度超过了640px,那么页面中html的font-size恒为100px。否则,页面中html的font-size大小为:100*(当前页面宽度 / 640)。
用户1349575
2022/01/27
2K0
H5移动端rem适配
1 /** 2 * 移动端自适应 3 */ 4 <meta name="viewport" 5 content="width=device-width,user-scalable=no,initial-scale=1.0,minimum-scale=1.0,maximum-scale=1.0"> 6 // <!--content--> 7 // <!--width=device-width 可视区域的宽度,值可为数字或关键词de
ProsperLee
2018/10/24
1.5K0
移动端项目设配UI设计图方法
由于要使用UI的设计图,所以需要根据UI设定的尺寸来写页面 下面是移动端适配的方法 在script标签中执行 <script> (function (doc, win) {
用户10106350
2022/10/28
3680
H5页面移动端适配且要根据UI设计图给出的单位出图方法总结
代码已上传至github github代码地址:https://github.com/Miofly/mio.git
用户10106350
2022/10/28
5310
移动端布局攻略
作者:Tolonger 原文链接:https://www.jianshu.com/p/9e7e755ca281 除了百分比流式布局之外,rem布局占据了目前移动端布局的热潮。那么究竟这几种布局差别在哪里,对应的有什么效果,希望本文能给你一些有益的启示。除此之外还有响应式布局,固定宽度布局等。 百分比流式布局 这里面最知名的当属bootstrap框架的思路,他所有的组件以及模板ui均是百分比流式布局,单位为px.并且我们看到的大部分对移动端适配的页面也均是采用这种核心思想去做的,方法简单,多端共用,可以针对
前端教程
2018/03/05
1.6K0
移动端布局攻略
08-移动端开发教程-移动端适配方案
由于移动端的特殊性,屏幕的尺寸碎片化严重,要想很好的适配不同的尺寸的设备,需要我们前端开发相比PC端要做一些基层的适配方案。
老马
2018/02/18
3.8K0
08-移动端开发教程-移动端适配方案
vue及原生html实现列表无缝上下滚动,以及单行滚动
代码已上传至github github代码地址:https://github.com/Miofly/mio.git
用户10106350
2022/10/28
3.9K0
Rem在移动端的适配
<!DOCTYPE html> <html> <head> <meta http-equiv="Content-Type" content="text/html" charset="UTF-8"> <title></title> <!-- 为移动设备添加 viewport --> <meta name="viewport" content="width=device-width, initial-scale=1.0, user-scalable=0" /> <scr
越陌度阡
2020/11/26
6300
canvas合成文字以及对文字进行换行处理
代码已上传至github github代码地址:https://github.com/Miofly/mio.git
用户10106350
2022/10/28
8230
如何写自适应分辨率的网页
方式一:用媒体查询"@media",这种写法好处是可以对不同分辨率的设备,展示完全不同的UI界面,一个页面不同的设备看的时候,展示内容可以不一样,交互方式可以不一样。不过这个不方便用在复杂的地方,而且不同的分辨率都需要对应的重新写样式,同一个页面集合太多的这种写法,最好是分开写两套,降低耦合性。但是这种写法费力不讨好,之前有的网站在PC和手机查看到的样式不一致,用了一些这个技术,但是后来很多都是检测到不同设备,就跳转到不同的网页上去了。
前端小tips
2021/12/11
2.7K0
如何写自适应分辨率的网页
移动端适配终极封装
var adaptive = {}; (function (win, lib) { var doc = win.document; var docEl = doc.documentElement; var devicePixelRatio = win.devicePixelRatio; var dpr = 1; var scale = 1; function setViewport() { var isIPhone = /iphone/gi.
用户10106350
2022/10/28
6000
移动端适配大法
前端代码的编写永远逃不过“兼容”二词,从前PC时代,因为IE的傲娇,导致程序猿们一直在兼容IE的道路上挣扎,如今移动设备的普及,仿佛让我们看到了希望,仿佛马上就要摆脱IE了,可是!一波还未平息,一波又来侵袭~移动端确实不用考虑IE了,各种CSS新特性也用的爽到飞起,但一座大山压了过来,那就是分辨率的适配,移动端由于展示区域比较小,因此对于页面在不同分辨率手机上的展示细节也要求更加严格,这时像PC端有些固定宽高的布局方式显然不适应,我们被要求对于不同大小手机页面能自适应,真是非常有(tong)趣(ku)呢~ 话不多说,下面就总结了一些移动端常用的适配手法:
IMWeb前端团队
2019/12/03
2.9K0
HTML+CSS,PC端/手机端公用部分样式代码整理(建议收藏)
我们一般都会提一个 .overflow {overflow:hidden} 方便布局,也可以有效的阻止移动端上下左右拖动问题(溢出问题)
程序员纬度
2021/03/02
2.2K0
前端学习笔记—移动端web开发
响应式网站:即pc和移动端共用一套网站,只不过在不同宽度的屏幕下,样式会自动适配。配合媒体查询监听,通过判断屏幕宽度来改变样式,以适应不同终端。例如:三星电子官网: www.samsung.com/cn/ 缺点:制作麻烦,需要花很大精力去调兼容性问题
木溪bo
2024/03/23
3840
前端学习笔记—移动端web开发
canvas合成任意数量图片、合成二维码加文字封装处理
代码已上传至github github代码地址:https://github.com/Miofly/mio.git
用户10106350
2022/10/28
7390
前端工程师之移动端布局方案
百分比布局是一种等比例缩放的布局方式,也是移动Web开发中比较常见的布局方式。在CSS代码中需要使用百分比来设置盒子的宽高。
张哥编程
2024/12/13
3640
HTML+CSS,PC端/手机端公用部分样式代码整理(自己收藏)
我们一般都会提一个 .overflow {overflow:hidden} 方便布局,也可以有效的阻止移动端上下左右拖动问题(溢出问题)
用户5997198
2019/08/12
3K0
微信小程序布局单位的使用
rpx单位是微信小程序中css的尺寸单位,rpx可以根据屏幕宽度进行自适应。官方推荐微信小程序可以用iPhone6 作为视觉稿的标准。规定屏幕宽为750rpx。如在 iPhone6 上,屏幕宽度为750 px,则共有个750 物理像素,则750 rpx = 375px = 750 物理像素 例如 : 1rpx = 0.5px = 1物理像素
用户2305175
2018/06/24
3.3K0
推荐阅读
相关推荐
解决移动端适配rem单位在华为和三星手机出现bug
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档