linux_socket网络套接字函数(TCP通信)-socket函数-bind函数-listen函数-accept函数-connect函数-send函数-recv函数

接上一篇:linux_网络通信-套接字通信socket-网络字节序-IP地址转换函数-inet_pton函数-htonl函数-htons函数-ntohl函数-ntohs函数

  今天来分享网络套接字需要用到的函数以及怎样用tcp来建立网络通信,开始上菜:

此博主在CSDN发布的文章目录:我的CSDN目录,作为博主在CSDN上发布的文章类型导读

1.TCP-socket模型创建流程图

在这里插入图片描述

2.socket函数

函数作用:
  创建一个通信端点,并返回一个文件描述符。
头文件:
  #include <sys/socket.h>
  #include <sys/types.h>
函数原型:
  int socket(int domain, int type, int protocol);
函数参数:
domain:
  AF_INET 这是大多数用来产生socket的协议,使用TCP或UDP来传输,用IPv4的地址
  AF_INET6 与上面类似,不过是来用IPv6的地址
  AF_UNIX 本地协议,使用在Unix和Linux系统上,一般都是当客户端和服务器在同一台及其上的时候使用
type:
  SOCK_STREAM 这个协议是按照顺序的、可靠的、数据完整的基于字节流的连接。这是一个使用最多的socket类型,这个socket是使用TCP来进行传输。
  SOCK_DGRAM 这个协议是无连接的、固定长度的传输调用。该协议是不可靠的,使用UDP来进行它的连接。
  SOCK_SEQPACKET该协议是双线路的、可靠的连接,发送固定长度的数据包进行传输。必须把这个包完整的接受才能进行读取。
  SOCK_RAW socket类型提供单一的网络访问,这个socket类型使用ICMP公共协议。(ping、traceroute使用该协议)
  SOCK_RDM 这个类型是很少使用的,在大部分的操作系统上没有实现,它是提供给数据链路层使用,不保证数据包的顺序
protocol:
  传0 表示使用默认协议。
返回值:
  成功:返回指向新创建的socket的文件描述符;
  失败:返回-1,设置errno。
注意:
  socket()打开一个网络通讯端口,如果成功的话,就像open()一样返回一个文件描述符,应用程序可以像读写文件一样用read/write在网络上收发数据,如果socket()调用出错则返回-1。对于IPv4,domain参数指定为AF_INET。
  对于TCP协议,type参数指定为SOCK_STREAM,表示面向流的传输协议。
  如果是UDP协议,则type参数指定为SOCK_DGRAM,表示面向数据报的传输协议。protocol参数的介绍从略,指定为0即可。

3.bind函数

函数作用:
  将addr指定的地址分配给由文件描述符sockfd引用的套接字。
头文件:
  #include <sys/socket.h>
  #include <sys/types.h>
函数原型:
  int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
函数参数:
  sockfd:socket文件描述符
  addr:构造出IP地址加端口号
  addrlen:sizeof(addr)长度
返回值:
  成功:返回0;
  失败:返回-1,设置errno。
注意:
  服务器程序所监听的网络地址和端口号通常是固定不变的,客户端程序得知服务器程序的地址和端口号后就可以向服务器发起连接,因此服务器需要调用bind绑定一个固定的网络地址和端口号。
  bind()的作用是将参数sockfd和addr绑定在一起,使sockfd这个用于网络通讯的文件描述符监听addr所描述的地址和端口号。
  struct sockaddr *是一个通用指针类型,addr参数实际上可以接受多种协议的sockaddr结构体,而它们的长度各不相同,所以需要第三个参数addrlen指定结构体的长度。如:

struct sockaddr_in servaddr;
bzero(&servaddr, sizeof(servaddr));
servaddr.sin_family = AF_INET;
servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
servaddr.sin_port = htons(6666);

  首先将整个结构体清零,然后设置地址类型为AF_INET,网络地址为INADDR_ANY,这个宏表示本地的任意IP地址,因为服务器可能有多个网卡,每个网卡也可能绑定多个IP地址,这样设置可以在所有的IP地址上监听,直到与某个客户端建立了连接时才确定下来到底用哪个IP地址,端口号为6666。

4.listen函数

函数作用:
  允许建立三次握手的最大连接数。
头文件:
  #include <sys/socket.h>
  #include <sys/types.h>
函数原型:
  int listen(int sockfd, int backlog);
函数参数:
  sockfd:socket文件描述符
  backlog:排队建立3次握手队列和刚刚建立3次握手队列的链接数和
  查看系统默认backlog:cat /proc/sys/net/ipv4/tcp_max_syn_backlog
返回值:
  成功:返回0;
  失败:返回-1,设置errno。
注意:
  典型的服务器程序可以同时服务于多个客户端,当有客户端发起连接时,服务器调用的accept()返回并接受这个连接。
  如果有大量的客户端发起连接而服务器来不及处理,尚未accept的客户端就处于连接等待状态,listen()声明sockfd处于监听状态,并且最多允许有backlog个客户端处于连接待状态,如果接收到更多的连接请求就忽略。

5.accept函数

函数作用:
  提取侦听套接字的挂起连接队列中的第一个连接请求,并且返回一个引用该套接字的新文件描述符。
头文件:
  #include <sys/socket.h>
  #include <sys/types.h>
函数原型:
  int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen);
函数参数:
  sockdf:socket文件描述符
  addr:传出参数,返回链接客户端地址信息,含IP地址和端口号
  addrlen1:传入传出参数(值-结果),传入sizeof(addr)大小,函数返回时返回真正接收到地址结构体的大小
返回值:
  成功:返回一个新的socket文件描述符,用于和客户端通信;
  失败:返回-1,设置errno。
注意:
  三方握手完成后,服务器调用accept()接受连接,如果服务器调用accept()时还没有客户端的连接请求,就阻塞等待直到有客户端连接上来。
  addr是一个传出参数,accept()返回时传出客户端的地址和端口号。、addrlen参数是一个传入传出参数(value-result argument),传入的是调用者提供的缓冲区addr的长度以避免缓冲区溢出问题,传出的是客户端地址结构体的实际长度(有可能没有占满调用者提供的缓冲区)。如果给addr参数传NULL,表示不关心客户端的地址。

6.connect函数

函数作用:
  根据地址结构链接指定服务器。
头文件:
  #include <sys/socket.h>
  #include <sys/types.h>
函数原型:
  int connect(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
函数参数:
  sockdf:socket文件描述符
  addr:传入参数,指定服务器端地址信息,含IP地址和端口号
  addrlen:传入参数,传入sizeof(addr)大小
返回值:
  成功:返回0;
  失败:返回-1,设置errno。
注意:
  客户端需要调用connect()连接服务器,connect和bind的参数形式一致,区别在于bind的参数是自己的地址,而connect的参数是对方的地址。

7.read()函数

函数作用:
  向fd所指文件中读取数据。
头文件:
  #include <unistd.h>
函数原型:
  ssize_t read(int fd, void *buf, size_t count);
函数参数:
  fd:文件描述符
  buf:缓冲区
  count:最多读取的字节
返回值:
  成功:返回读取的字节数(零表示结束文件);
  失败:返回-1,并适当设置errno。

8.write()函数

函数作用:
  向fd所指文件中写入数据。
头文件:
  #include <unistd.h>
函数原型:
  ssize_t write(int fd, const void *buf, size_t count);
函数参数:
  fd:文件描述符。
  buf:数据缓冲区首地址。
  count:最多写入的字节。
返回值:
  成功:返回写入的字节数;
  失败:返回-1,并适当设置errno。

9.send函数

函数作用:
  发送数据到sockfd所指套接字中。
头文件:
  #include <sys/socket.h>
函数原型:
  ssize_t send(int sockfd, const void *buf, size_t len, int flags);
函数参数:
  sockfd:网络套接字描述符。
  buf:数据缓冲区首地址。
  len:buf的大小。
  flags:控制标志,默认可设置为0。
返回值:
  成功:返回发送的字节数;
  失败:返回-1,并适当设置errno。

10.recv函数

函数作用:
  接收sockfd所指套接字中的数据。
头文件:
  #include <sys/socket.h>
函数原型:
   ssize_t recv(int sockfd, void *buf, size_t len, int flags);
函数参数:
  sockfd:网络套接字描述符。
  buf:数据缓冲区首地址。
  len:buf的大小。
  flags:控制标志,默认可设置为0。
返回值:
  成功:返回接收的字节数;
  失败:返回-1,并适当设置errno。

11.TCP网络通信-C语言代码

11.1.服务器端server.c

#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <strings.h>
#include <string.h>
#include <ctype.h>
#include <arpa/inet.h>
#define SERV_PORT 9527
int main(void)
{
    int sfd, cfd;
    int len, i;
    char buf[BUFSIZ], clie_IP[BUFSIZ];
    struct sockaddr_in serv_addr, clie_addr;
    socklen_t clie_addr_len;
    /*创建一个socket 指定IPv4协议族 TCP协议*/
    sfd = socket(AF_INET, SOCK_STREAM, 0);
    /*初始化一个地址结构 man 7 ip 查看对应信息*/
    bzero(&serv_addr, sizeof(serv_addr));           //将整个结构体清零
    serv_addr.sin_family = AF_INET;                 //选择协议族为IPv4
    serv_addr.sin_addr.s_addr = htonl(INADDR_ANY);  //监听本地所有IP地址
    serv_addr.sin_port = htons(SERV_PORT);          //绑定端口号    
    /*绑定服务器地址结构*/
    bind(sfd, (struct sockaddr *)&serv_addr, sizeof(serv_addr));
    /*设定链接上限,注意此处不阻塞*/
    listen(sfd, 64);                                //同一时刻允许向服务器发起链接请求的数量
    printf("等待连接中 ...\n");
    /*获取客户端地址结构大小*/ 
    clie_addr_len = sizeof(clie_addr_len);
    cfd = accept(sfd, (struct sockaddr *)&clie_addr, &clie_addr_len);           /*监听客户端链接, 会阻塞*/

    printf("客户端IP:%s\t端口:%d\n", 
            inet_ntop(AF_INET, &clie_addr.sin_addr.s_addr, clie_IP, sizeof(clie_IP)), 
            ntohs(clie_addr.sin_port));
    while (1) {
        /*读取客户端发送数据*/
        len = read(cfd, buf, sizeof(buf));
        write(STDOUT_FILENO, buf, len);
        /*处理客户端数据*/
        for (i = 0; i < len; i++)
            buf[i] = toupper(buf[i]);//将小写字母转换为大写字母
        /*处理完数据回写给客户端*/
        write(cfd, buf, len); 
    }
    /*关闭链接*/
    close(sfd);
    close(cfd);
    return 0;
}

11.2.客户端client.c

#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#define SERV_IP "127.0.0.1"
#define SERV_PORT 9527
int main(void)
{
    int sfd, len;
    struct sockaddr_in serv_addr;
    char buf[BUFSIZ]; 
    /*创建一个socket 指定IPv4 TCP*/
    sfd = socket(AF_INET, SOCK_STREAM, 0);
    /*初始化一个地址结构:*/
    bzero(&serv_addr, sizeof(serv_addr));                       //清零
    serv_addr.sin_family = AF_INET;                             //IPv4协议族
    inet_pton(AF_INET, SERV_IP, &serv_addr.sin_addr.s_addr);    //指定IP 字符串类型转换为网络字节序
    serv_addr.sin_port = htons(SERV_PORT);                      //指定端口 本地转网络字节序
    /*根据地址结构链接指定服务器进程*/
    connect(sfd, (struct sockaddr *)&serv_addr, sizeof(serv_addr));
    while (1) {
        /*从标准输入获取数据*/
        fgets(buf, sizeof(buf), stdin);
        /*将数据写给服务器*/
        write(sfd, buf, strlen(buf));       //写给服务器
        /*从服务器读回转换后数据*/
        len = read(sfd, buf, sizeof(buf));
        /*写至标准输出*/
        write(STDOUT_FILENO, buf, len);
    }
    /*关闭链接*/
    close(sfd);
    return 0;
}

  以上就是本次的分享了,希望对大家有所帮助,欢迎关注博主一起学习更多的新知识!

  • 1
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

futureCode.

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值