首页
学习
活动
专区
工具
TVP
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往

linux C语言循环监听

在Linux环境下使用C语言进行循环监听通常涉及到网络编程,特别是服务器端程序需要持续监听客户端的连接请求。以下是关于“Linux C语言循环监听”的基础概念、优势、类型、应用场景以及可能遇到的问题和解决方法:

基础概念

循环监听指的是服务器端程序使用一个无限循环来不断地检查是否有客户端的连接请求。当检测到连接请求时,服务器会接受这个连接,并可能创建一个新的线程或进程来处理与该客户端的通信。

优势

  1. 实时响应:服务器可以实时响应客户端的连接请求。
  2. 持续服务:只要服务器程序在运行,就可以持续提供服务。
  3. 灵活性:可以根据需要调整监听的端口、监听的客户端数量等。

类型

  1. 单线程监听:服务器在单个线程中处理所有客户端的连接请求和数据交换。
  2. 多线程/多进程监听:服务器为每个客户端连接创建一个新的线程或进程来处理。
  3. 异步IO监听:使用如selectpollepoll等系统调用来异步处理多个客户端连接。

应用场景

  • Web服务器
  • 数据库服务器
  • 聊天服务器
  • 文件传输服务器

可能遇到的问题及解决方法

  1. 资源耗尽:如果使用多线程或多进程模型,可能会因为客户端连接过多而导致系统资源耗尽。
    • 解决方法:使用线程池或进程池来限制同时处理的客户端连接数量;使用异步IO模型来减少线程或进程的使用。
  • 阻塞问题:在单线程模型中,如果某个客户端的操作阻塞了,整个服务器都会受到影响。
    • 解决方法:使用非阻塞IO或异步IO来避免阻塞。
  • 效率问题:在大量客户端连接的情况下,单线程模型的效率可能较低。
    • 解决方法:使用多线程、多进程或异步IO模型来提高效率。

示例代码(使用select进行异步监听)

代码语言:txt
复制
#include <sys/socket.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <fcntl.h>
#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#define PORT 8080
#define MAX_CLIENTS 10

int main() {
    int server_fd, new_socket, max_sd, activity, valread, sd;
    int client_sockets[MAX_CLIENTS] = {0};
    struct sockaddr_in address;
    int addrlen = sizeof(address);
    char buffer[1024] = {0};

    // 创建socket文件描述符
    if ((server_fd = socket(AF_INET, SOCK_STREAM, 0)) == 0) {
        perror("socket failed");
        exit(EXIT_FAILURE);
    }

    // 设置socket选项
    int opt = 1;
    if (setsockopt(server_fd, SOL_SOCKET, SO_REUSEADDR | SO_REUSEPORT,
                   &opt, sizeof(opt))) {
        perror("setsockopt");
        exit(EXIT_FAILURE);
    }

    address.sin_family = AF_INET;
    address.sin_addr.s_addr = INADDR_ANY;
    address.sin_port = htons(PORT);

    // 绑定
    if (bind(server_fd, (struct sockaddr *)&address, sizeof(address)) < 0) {
        perror("bind failed");
        exit(EXIT_FAILURE);
    }

    // 监听
    if (listen(server_fd, 3) < 0) {
        perror("listen");
        exit(EXIT_FAILURE);
    }

    fd_set readfds;
    while (1) {
        FD_ZERO(&readfds);
        FD_SET(server_fd, &readfds);
        max_sd = server_fd;

        // 添加子socket到集合
        for (int i = 0; i < MAX_CLIENTS; i++) {
            sd = client_sockets[i];
            if (sd > 0)
                FD_SET(sd, &readfds);
            if (sd > max_sd)
                max_sd = sd;
        }

        // 等待活动
        activity = select(max_sd + 1, &readfds, NULL, NULL, NULL);
        if ((activity < 0) && (errno != EINTR)) {
            printf("select error");
        }

        // 如果有新的连接
        if (FD_ISSET(server_fd, &readfds)) {
            if ((new_socket = accept(server_fd, (struct sockaddr *)&address,
                                     (socklen_t*)&addrlen)) < 0) {
                perror("accept");
                exit(EXIT_FAILURE);
            }

            // 添加新的socket到数组
            for (int i = 0; i < MAX_CLIENTS; i++) {
                if (client_sockets[i] == 0) {
                    client_sockets[i] = new_socket;
                    printf("Adding to list of sockets as %d\n", i);
                    break;
                }
            }
        }

        // 处理客户端数据
        for (int i = 0; i < MAX_CLIENTS; i++) {
            sd = client_sockets[i];
            if (FD_ISSET(sd, &readfds)) {
                if ((valread = read(sd, buffer, 1024)) == 0) {
                    // 客户端断开连接
                    getpeername(sd, (struct sockaddr*)&address, \
                                (socklen_t*)&addrlen);
                    printf("Host disconnected, ip %s, port %d\n", \
                           inet_ntoa(address.sin_addr), ntohs(address.sin_port));
                    close(sd);
                    client_sockets[i] = 0;
                } else {
                    // 处理数据
                    buffer[valread] = '\0';
                    printf("Received: %s\n", buffer);
                }
            }
        }
    }

    return 0;
}

这个示例代码展示了如何使用select系统调用来进行异步监听,从而可以同时处理多个客户端连接。

页面内容是否对你有帮助?
有帮助
没帮助

相关·内容

领券