socket回显服务器练习

前言

什么是回显服务器(echo server)呢?

回显服务器接收客户端发送的任何数据,并将这些数据原封不动地发送回客户端。回显服务器在连接成功的基础上只需要知道如何在客户端将收到的信息打印输出到控制台即可。我接下来会使用两种方法来输出,一种是printf(),另一种是fputs(),详见客户端信息传输部分。

服务端

c 复制代码
#include <stdio.h>
#include <string.h>
#include <sys/errno.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <stdlib.h>
#include <ctype.h>
#define PORT 8888
#define BUFFSIZE 1024
#define ERRORCODE -1

int main(int argc, char *argv[])
{
    char buf[BUFSIZ],client_addr_INFO[BUFFSIZE];
    int accept_fd, listen_fd;
    socklen_t client_len;
    struct sockaddr_in listen_addr, accept_addr;
    char buffer[BUFFSIZE];
    int buffer_len,i;
    int on = 1;
    //创建socket
    listen_fd = socket(AF_INET, SOCK_STREAM, 0);
    if (listen_fd == -1)
    {
        printf("创建socket error: %s \n", strerror(errno));
        return ERRORCODE;
    }
    //bind
    if(setsockopt(listen_fd,SOL_SOCKET,SO_REUSEADDR,&on,sizeof(on)))
    {
        printf("setsockopt error: %s \n", strerror(errno));
        return ERRORCODE;
    }
    listen_addr.sin_family = AF_INET;
    listen_addr.sin_port = htons(PORT);
    listen_addr.sin_addr.s_addr = htonl(INADDR_ANY);
    if (bind(listen_fd, (struct sockaddr *)&listen_addr, sizeof(listen_addr)) < 0)
    {
        printf("bind error: %s \n", strerror(errno));
        return ERRORCODE;
    }

    //listen
    if (listen(listen_fd, 5) == -1)
    {
        printf("listen error: %s \n", strerror(errno));
        return ERRORCODE;
    }
    printf("服务端创建成功,等待连接\n");
    //accept

    client_len = sizeof(accept_addr);
    
    /* code */
    accept_fd = accept(listen_fd, (struct sockaddr *)&accept_addr, &client_len);
    if (accept_fd == -1)
    {
        printf("accept error: %s \n", strerror(errno));
        return ERRORCODE;
    }
    printf ("accept ip : %s,port : %d\n",
    inet_ntop(AF_INET,&accept_addr.sin_addr.s_addr,client_addr_INFO,sizeof(client_addr_INFO)),
    ntohs(accept_addr.sin_port));
    printf("accept ip : %s,port : %d\n",inet_ntoa(accept_addr.sin_addr),ntohs(accept_addr.sin_port));
   
    
    while (1)
    {
        memset(buffer,0,sizeof(buffer));
        buffer_len = read(accept_fd,buffer,sizeof(buffer));
        fputs(buffer,stdout);
     
        write(accept_fd,buffer,buffer_len);

        /* code */
    }
    
    close(accept_fd);
    close(listen_fd);
    return 0;
}

nc测试

在服务端做好了之后,我们先不急于写客户端,我们可以先用nc(netcat)工具来测试服务端是否可以正常运作。

先把服务端跑起来,然后在终端连接,可以看到回显是能够实现的。

客户端

c 复制代码
#include <stdio.h>
#include <string.h>
#include <sys/errno.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <stdlib.h>
#include <ctype.h>

#define BUFFSIZE 1024
#define ERRORCODE -1
#define PORT 8888
int main(int argc, char *argv[])
{
    
    char send_buffer[BUFFSIZE],recv_buffer[BUFFSIZE];
    
    int buffer_len;
    int client_fd;
    struct sockaddr_in connect_addr;
    //创建socket
    client_fd = socket(AF_INET, SOCK_STREAM, 0);
    if (client_fd == -1)
    {
        printf("socket error: %s \n", strerror(errno));
        return ERRORCODE;
    }
    //connect

    inet_pton(AF_INET,"127.0.0.1",&connect_addr.sin_addr.s_addr);
    connect_addr.sin_family = AF_INET;
    connect_addr.sin_port = htons(PORT);
    if (connect(client_fd, (struct sockaddr *)&connect_addr, sizeof(connect_addr)) == -1)
    {
        printf("connect error: %s \n", strerror(errno));
        return ERRORCODE;
    }
    printf("连接成功 , 请输入内容:\n");
    while (fgets(send_buffer,sizeof(send_buffer),stdin) != NULL)
    {
        
        write(client_fd,send_buffer,sizeof(send_buffer));
        //sleep(1);
        read(client_fd,recv_buffer,sizeof(recv_buffer));

        //fputs(recv_buffer,stdout);
        printf("回显内容: %s\n",recv_buffer);
        memset(send_buffer,0,sizeof(send_buffer));
        memset(recv_buffer,0,sizeof(recv_buffer));
        
    }
    close(client_fd);
    return 0;
}

运行效果


注:关于客户端与服务端连接的部分我在我的其它文章有写。

socket通信基本流程函数 -- 连接

相关推荐
Linux运维技术栈4 分钟前
Ansible(自动化运维)环境搭建及ansible-vault加密配置
运维·自动化·ansible
YashanDB1 小时前
【YashanDB知识库】XMLAGG方法的兼容
数据库·yashandb·崖山数据库
独行soc1 小时前
#渗透测试#漏洞挖掘#红蓝攻防#护网#sql注入介绍11基于XML的SQL注入(XML-Based SQL Injection)
数据库·安全·web安全·漏洞挖掘·sql注入·hw·xml注入
Bessssss1 小时前
centos权限大集合,覆盖多种权限类型,解惑权限后有“. + t s”问题!
linux·运维·centos
苹果醋31 小时前
Golang的文件加密工具
运维·vue.js·spring boot·nginx·课程设计
jwensh2 小时前
【Jenkins】Declarative和Scripted两种脚本模式有什么具体的区别
运维·前端·jenkins
风间琉璃""2 小时前
bugkctf 渗透测试1超详细版
数据库·web安全·网络安全·渗透测试·内网·安全工具
drebander2 小时前
SQL 实战-巧用 CASE WHEN 实现条件分组与统计
大数据·数据库·sql
IvorySQL2 小时前
IvorySQL 4.0 发布:全面支持 PostgreSQL 17
数据库·postgresql·开源数据库·国产数据库·ivorysql
18号房客2 小时前
高级sql技巧进阶教程
大数据·数据库·数据仓库·sql·mysql·时序数据库·数据库架构