Socket | C语言 | Linux | 网络编程 | 实战(一)

socket 网络编程和接口详解 | Windows实例 | Linux实例_我是一块小石头-CSDN博客
Socket | C语言 | Linux | 网络编程 | 实战(一)_我是一块小石头-CSDN博客
Socket | C语言 | Linux | 网络编程 | 实战(二)_我是一块小石头-CSDN博客
Socket | C语言 | Linux | 网络编程 | 实战(三)_我是一块小石头-CSDN博客

实例

实现了client单向(send发)发消息给server(recv收),quit断开socket链接

Client

#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <string.h>
#include <unistd.h>
#include <errno.h>
#include <arpa/inet.h>

#define MAX_READ_LINE 1024

int main(void) {
    char *server_ip_addr = "127.0.0.1";
    int server_ip_port = 10004;
    char send_message[MAX_READ_LINE];

    //建立socket连接,AF_INET : TCP/IP-IPV4,SOCK_STREAM : TCP类型
    int socket_fd = socket(AF_INET, SOCK_STREAM, 0);
    if (socket_fd < 0) {
        fprintf(stderr, "socket error %s errno: %d\n", strerror(errno), errno);
    }
    printf("创建SOCKET成功!\n");

    struct sockaddr_in t_sockaddr;
    memset(&t_sockaddr, 0, sizeof(struct sockaddr_in));
    t_sockaddr.sin_family = AF_INET;
    t_sockaddr.sin_port = htons(server_ip_port);
    //inet_pton是一个IP地址转换函数,可以在将点分文本的IP地址转换为二进制网络字节序”的IP地址,
    //而且inet_pton和inet_ntop这2个函数能够处理ipv4和ipv6
    inet_pton(AF_INET, server_ip_addr, &t_sockaddr.sin_addr);

    //TCP,客户端主动连接服务器
    if((connect(socket_fd, (struct sockaddr*)&t_sockaddr, sizeof(struct sockaddr))) < 0 ) {
        fprintf(stderr, "connect error %s errno: %d\n", strerror(errno), errno);
        return 0;
    }
    printf("连接成功!\n");
    while(1){
        //TCP发送数据
        memset(send_message, 0, 255);
        printf("发送:");
        //get input from keyboard using fileinput
        fgets(send_message,sizeof(send_message), stdin);
        strtok(send_message, "\n");
        if(strcmp(send_message, " ") == 0 || strcmp(send_message, "") == 0)
        {
        printf("Your input is empty.");
        continue;
        }
        if((send(socket_fd, send_message, strlen(send_message), 0)) < 0) {
            fprintf(stderr, "send message error: %s errno : %d", strerror(errno), errno);
            return 0;
        }
        if(strcmp(send_message, "quit") == 0||strcmp(send_message, "allquit") == 0)
        {
            printf("End conversation.\n");
            break;
        }
    }
    close(socket_fd);
    socket_fd = -1;

    return 0;
}


Server

#include <stdio.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <string.h>
#include <netinet/in.h>
#include <stdlib.h>
#include <errno.h>
#include <unistd.h>
#include <stdbool.h>
#include <arpa/inet.h>

#define MAX_READ_LINE 1024

int main(void) {
    int recv_len = -1;
    int conn_fd = -1;
    int ret = -1;
    int server_ip_port = 10004;

    // sockaddr_in在头文件#include<netinet/in.h>或#include <arpa/inet.h>中定义,该结构体解决了sockaddr的缺陷,
    // 把port和addr 分开储存在两个变量中,如下:
    struct sockaddr_in t_sockaddr;
    memset(&t_sockaddr, 0, sizeof(t_sockaddr));
    t_sockaddr.sin_family = AF_INET;//AF_INET(TCP/IP – IPv4)
    t_sockaddr.sin_addr.s_addr = htonl(INADDR_ANY);//ip
    t_sockaddr.sin_port = htons(server_ip_port);//port

    //SOCK_STREAM TCP类型
    //AF_INET(TCP/IP – IPv4)
    //初始化创建socket对象
    int listen_fd = socket(AF_INET, SOCK_STREAM, 0);
    if (listen_fd < 0) {
        fprintf(stderr, "socket error %s errno: %d\n", strerror(errno), errno);
    }
    printf("创建SOCKET成功!\n");
    //将初始化的socket对象绑定特定ip、端口上
    ret = bind(listen_fd,(struct sockaddr *) &t_sockaddr,sizeof(t_sockaddr));
    if (ret < 0) {
        fprintf(stderr, "bind socket error %s errno: %d\n", strerror(errno), errno);
    }
    printf("绑定成功!\n");
    //启动监听服务
    ret = listen(listen_fd, 1024);
    if (ret < 0) {
        fprintf(stderr, "listen error %s errno: %d\n", strerror(errno), errno);
    }
    printf("监听成功!\n");
    bool flag = true;
    while(flag) {
        struct sockaddr_in client_addr;
        socklen_t client_addrlength = sizeof(client_addr);
 
        //TCP
        //从已经完成连接队列返回下一个简历成功的连接,如果队列为空进入阻塞态
        conn_fd = accept(listen_fd, (struct sockaddr*)&client_addr, &client_addrlength);
        if(conn_fd < 0) {
            fprintf(stderr, "accpet socket error: %s errno :%d\n", strerror(errno), errno);
            continue;
        }
        printf("有客户端连接进来了.IP:%s, port:%d\n", inet_ntoa(client_addr.sin_addr), ntohs(client_addr.sin_port));
        //TCP类型的数据接收,返回数据大小
        char buff[MAX_READ_LINE];
        while(1){
            memset(buff, 0, sizeof(buff));
            recv_len = recv(conn_fd, buff, MAX_READ_LINE, 0);
            if (recv_len < 0) {
                fprintf(stderr, "recv error %s errno: %d\n", strerror(errno), errno);
                continue;
            }
            //添加char[]结束标志
            buff[recv_len] = '\0';
            fprintf(stdout, "recv message from client: %s\n", buff);
            if(strcmp(buff,"quit")==0){
                printf("Conversation will be closed.\n");
                break;
            }

            if(strcmp(buff,"allquit")==0){
                printf("Server will be closed.\n");
                flag = false;
                break;
            }
            printf(">> %s\n",buff);
        }
        close(conn_fd);
        conn_fd = -1;
    }

    close(listen_fd);
    listen_fd = -1;

    return 0;
}


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值