Linux网络编程1-简单的CS通信程序

Linux网络编程1-简单的CS通信程序


1.Socket相关API说明

1.1字节序转换函数:用于ip和port转换

网络字节序是大端字节序:低位地址存放高位数据, 高位地址存放低位数据。一般主机字节序是小端字节序:低位地址存放低位数据, 高位地址存放高位数据。

0x12345678
大端存储:12 34 56 78
小端存储: 78 56 34 12

小端字节序需要转换为大端字节序:

整形的转换函数:一般用于端口htons。

cpp 复制代码
#include <arpa/inet.h>
// 主机到网络
uint32_t htonl(uint32_t hostlong); 
uint16_t htons(uint16_t hostshort);
// 网络到主机
uint32_t ntohl(uint32_t netlong);
uint16_t ntohs(uint16_t netshort);

IP地址转换函数:

cpp 复制代码
int inet_pton(int af, const char *src, void *dst); // 点分十进制转换为网络字节序 
// inet_pton(AF_INET, "127.0.0.1", &serv.sin_addr.s_addr); 127.0.0.1 -> [100007f]
cpp 复制代码
const char *inet_ntop(int af, const void *src, char *dst, socklen_t size); // 网络字节序转换为点分十进制IP地址

1.2sockaddr结构

cpp 复制代码
struct sockaddr {
    typedef unsigned short int sa_family_t;
    sa_family_t sa_family; // AF_INET
    char     sa_data[14];  // 地址信息:2个字节存放port,4个字节存放ip,剩下8个字节空闲
}

使用sockaddr结构不方便,一般使用sockaddr_in结构,两个结构体大小相同。

cpp 复制代码
struct sockaddr_in {
    sa_family_t    sin_family; /* address family: AF_INET */
    in_port_t      sin_port;   /* port in network byte order */
    struct in_addr sin_addr;   /* internet address */
};

/* Internet address.  */
typedef uint32_t in_addr_t;
struct in_addr {
    in_addr_t s_addr;
};

如果函数需要sockaddr结构,可以使用sockaddr_in变量,然后强转成sockaddr结构:

cpp 复制代码
struct sockaddr_in serv;
serv.sin_family = AF_INET;
serv.sin_port = htons(6789);
// 服务器的ip地址:点分十进制ip地址转换为网络ip
inet_pton(AF_INET, "127.0.0.1", &serv.sin_addr.s_addr);

int ret = connect(cfd, (struct sockaddr *)&serv, sizeof(serv));

1.3socket函数 以及两个队列

cpp 复制代码
int socket(int domain, int type, int protocol);

/*
domain: 协议版本
AF_INET IPV4
AF_INET6 IPV6
AF_UNIX AF_LOCAL本地套接字使用

type:协议类型
SOCK_STREAM 流式, 默认使用的协议是TCP协议
SOCK_DGRAM  报式, 默认使用的是UDP协议

protocal: 
一般填0, 表示使用对应类型的默认协议.

成功: 返回一个大于0的文件描述符
失败: 返回-1, 并设置errno
*/
int cfd = socket(AF_INET, SOCK_STREAM, 0);

当调用socket函数以后, 返回一个文件描述符, 内核会提供与该文件描述符相对应的读和写缓冲区, 同时还有两个队列, 分别是请求连接队列已连接队列.

1.4bind listen connect accept

bind:将socket文件描述符和IP,PORT绑定

cpp 复制代码
int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
/*
成功: 返回0
失败: 返回-1, 并设置errno
*/
struct sockaddr_in serv;
bzero(&serv, sizeof(serv));
serv.sin_family = AF_INET;
serv.sin_port = htons(6789); // 主机到网络字节序
serv.sin_addr.s_addr = htonl(INADDR_ANY); // 使用本机任意有效的可用IP
int ret = bind(lfd, (struct sockaddr *) &serv, sizeof(serv));
if (ret < 0)
{
    perror("bind error");
    return -1;
}

listen:将套接字由主动态变为被动态:

cpp 复制代码
int listen(int sockfd, int backlog);
/*
backlog: 同时请求连接的最大个数(还未建立连接) 

成功: 返回0
失败: 返回-1, 并设置errno
*/
ret = listen(lfd, 128); // 请求连接的最大个数128
if(ret < 0)
{
    perror("listen error");
    return -1;
}

accept:获得一个连接,,若当前没有连接则会阻塞等待.

cpp 复制代码
int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen);
/*
addr: 传出参数, 保存客户端的地址信息
addrlen: 传入传出参数,  addr变量所占内存空间大小

成功: 返回一个新的文件描述符,用于和客户端通信
失败: 返回-1, 并设置errno值.
*/
struct sockaddr_in client;
socklen_t len = sizeof(client);
int cfd = accept(lfd, (struct sockaddr *)&client, &len);
if(cfd < 0)
{
    perror("accpet error");
    return -1;
}

accept函数是一个阻塞函数, 若没有新的连接请求, 则一直阻塞。从已连接队列中获取一个新的连接, 并获得一个新的文件描述符, 该文件描述符用于和客户端通信. (内核会负责将请求队列中的连接拿到已连接队列中)

调用accept函数不是说新建一个连接,而是从已连接队列中取出一个可用连接.

connect:连接服务器

cpp 复制代码
int connect(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
/*
addr: 服务端的地址信息

成功: 返回0
失败: 返回-1, 并设置errno值
*/
int ret = connect(cfd, (struct sockaddr *)&serv, sizeof(serv));
if(ret<0)
{
    perror("connect error");
    return -1;
}

1.5收发数据

可以使用write和read函数进行读写操作。还可以使用recv和send函数。

cpp 复制代码
// 读取数据和发送数据:
ssize_t read(int fd, void *buf, size_t count);
ssize_t write(int fd, const void *buf, size_t count);

// 对应recv和send这两个函数flags直接填0就可以了.
ssize_t recv(int sockfd, void *buf, size_t len, int flags);
ssize_t send(int sockfd, const void *buf, size_t len, int flags);	

注意: 如果写缓冲区已满, write也会阻塞, read读操作的时候, 若读缓冲区没有数据会引起阻塞.

2.服务器和客户端程序代码流程

3.服务器端

cpp 复制代码
#include <iostream>

#include <sys/types.h>
#include <sys/socket.h> // socket
#include <netinet/in.h> // sockaddr_in
#include <cstring> // bzero
#include <unistd.h> // read

int main()
{
    std::cout << "STR_TEST: " << STR_TEST << std::endl;
    // 1.创建监听套接字
    int lfd = socket(AF_INET, SOCK_STREAM, 0);
    if (lfd < 0) {
        perror("socket error");
        return -1;
    }

    // 2.绑定本地ip和port
    struct sockaddr_in serv;
    bzero(&serv, sizeof(serv));
    serv.sin_family = AF_INET;
    serv.sin_port = htons(6789); // 主机到网络字节序
    serv.sin_addr.s_addr = htonl(INADDR_ANY); // 使用本机任意有效的可用IP
    int ret = bind(lfd, (struct sockaddr *) &serv, sizeof(serv));
    if (ret < 0)
    {
        perror("bind error");
        return -1;
    }

    // 3.监听端口
    ret = listen(lfd, 128); // 请求连接的最大个数128
    if(ret < 0)
    {
        perror("listen error");
        return -1;
    }

    // 4.从连接队列中取出一个通信
    struct sockaddr_in client;
    socklen_t len = sizeof(client);
    int cfd = accept(lfd, (struct sockaddr *)&client, &len);
    if(cfd < 0)
    {
        perror("accpet error");
        return -1;
    }

    // 5.读写数据
    while(1)
    {
        char buf[1024];
        bzero(buf, sizeof(buf));

        ssize_t n = read(cfd, buf, sizeof(buf));
        if(n <= 0)
        {
            std::cout << "read error or client close, n=" << n << std::endl;
            break;
        }
        std::cout << "n=" << n << ", buf=" << buf << std::endl;

        // 将接收的数据转为大写,然后发送给客户端
        for(int i= 0;i<n;++i)
        {
            buf[i] = toupper(buf[i]);
        }
        n = write(cfd, buf, n);
        if(n < 0)
        {
            perror("write error");
            break;
        }
    }

    close(cfd);
    close(lfd);

    return 0;
}

4.客户端

cpp 复制代码
#include <iostream>

#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h> // inet_pton
#include <cstring>
#include <unistd.h> // read write

int main(int argc, char* argv[])
{
    std::cout << "STR_TEST: " << STR_TEST << std::endl;
    // 1.创建socket
    int cfd = socket(AF_INET, SOCK_STREAM, 0);
    if(cfd<0)
    {
        perror("socket error");
        return -1;
    }
    struct in_addr
    // 2.连接服务端
    struct sockaddr_in serv;
    serv.sin_family = AF_INET;
    serv.sin_port = htons(6789);
    // 服务器的ip地址:点分十进制ip地址转换为网络ip
    inet_pton(AF_INET, "127.0.0.1", &serv.sin_addr.s_addr);
    //printf("[%x]\n", serv.sin_addr.s_addr); [100007f]
    int ret = connect(cfd, (struct sockaddr *)&serv, sizeof(serv));
    if(ret<0)
    {
        perror("connect error");
        return -1;
    }

    while(1)
    {
        char buf[256];
        //读标准输入数据
        bzero(buf, sizeof(buf));
        int n = read(STDIN_FILENO, buf, sizeof(buf));

        //发送数据
        write(cfd, buf, n);

        //读服务端发来的数据
        bzero(buf, sizeof(buf));
        n = read(cfd, buf, sizeof(buf));
        if(n<=0)
        {
            std::cout << "read error or server closed, n=" << n << std::endl;
            break;
        }
        std::cout << "n=" << n << ", buf=" << buf << std::endl;

    }

    //关闭套接字cfd
    close(cfd);

    return 0;
}

5.测试accept不是建立连接而是从已连接的队列中取出一个通信

测试过程中可以使用netstat命令查看监听状态和连接状态

netstat命令:

a表示显示所有,

n表示显示的时候以数字的方式来显示

p表示显示进程信息(进程名和进程PID)

accept函数是从已连接队列中取出一个通信,可以在调用accept函数之前sleep一分钟,在此期间,执行客户端连接到服务器,查看端口监听状态,sleep结束后会accept,然后再查看端口监听状态。

sh 复制代码
1.执行Server accpet前等待30s
2.执行Client
3.netstat -anp | grep 6789 查看端口状态 连接已建立。
4.30s过后
5.再次执行netstat -anp | grep 6789 查看端口状态
相关推荐
lulu_gh_yu43 分钟前
数据结构之排序补充
c语言·开发语言·数据结构·c++·学习·算法·排序算法
ULTRA??1 小时前
C加加中的结构化绑定(解包,折叠展开)
开发语言·c++
凌云行者2 小时前
OpenGL入门005——使用Shader类管理着色器
c++·cmake·opengl
凌云行者2 小时前
OpenGL入门006——着色器在纹理混合中的应用
c++·cmake·opengl
~yY…s<#>2 小时前
【刷题17】最小栈、栈的压入弹出、逆波兰表达式
c语言·数据结构·c++·算法·leetcode
可均可可3 小时前
C++之OpenCV入门到提高004:Mat 对象的使用
c++·opencv·mat·imread·imwrite
白子寰3 小时前
【C++打怪之路Lv14】- “多态“篇
开发语言·c++
小芒果_013 小时前
P11229 [CSP-J 2024] 小木棍
c++·算法·信息学奥赛
gkdpjj4 小时前
C++优选算法十 哈希表
c++·算法·散列表
王俊山IT4 小时前
C++学习笔记----10、模块、头文件及各种主题(一)---- 模块(5)
开发语言·c++·笔记·学习