linux支持并发的服务器回射程序实例

实例一:不支持并发,单服务器---单客户端

/*************************************************************************
	> File Name: ser01.c
	> Author:
	> Mail:
	> Created Time: 2015年11月22日 星期日 10时22分17秒
 ************************************************************************/

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

int main()
{
   //int socket(int domain, int type, int protocol);
    int sockfd;
    sockfd = socket(AF_INET, SOCK_STREAM, 0);

    struct sockaddr_in seraddr;
    seraddr.sin_family = AF_INET;
    seraddr.sin_port = htons(8001);
    seraddr.sin_addr.s_addr = inet_addr("127.0.0.1");

    //int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
    bind(sockfd, (struct sockaddr *)&seraddr, sizeof(seraddr));

    listen(sockfd, 128);

    struct sockaddr cliaddr;
    socklen_t clilen;
    int connfd;
    connfd = accept(sockfd, (struct sockaddr *)&cliaddr, &clilen);

    char recvbuf[2014] = {0};
    int recvcount = 0;

    while(1){
        recvcount = read(connfd, recvbuf,1024);
        if (recvcount == 0){
            printf("read is over.\n");
            printf("the client is over.\n");
            exit(0);
        }
        else if (recvcount == -1){
            perror("read error.\n");
            exit(EXIT_FAILURE);
        }
        fputs(recvbuf, stdout);
        write(connfd, recvbuf, strlen(recvbuf));
        memset(recvbuf, 0, 1024);
    }

    exit(0);
}
/*************************************************************************
	> File Name: cli01.c
	> Author:
	> Mail:
	> Created Time: 2015年11月22日 星期日 11时56分05秒
 ************************************************************************/

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

int main()
{

    int sockfd, ret;
    struct sockaddr_in seraddr;
    seraddr.sin_family = AF_INET;
    seraddr.sin_port = htons(8001);
    seraddr.sin_addr.s_addr = inet_addr("127.0.0.1");

    sockfd = socket(AF_INET, SOCK_STREAM, 0);

    ret = connect(sockfd, (struct sockaddr*)&seraddr, sizeof(seraddr));
    if(ret == -1){
        perror("bind error.\n");
        exit(EXIT_FAILURE);
    }

    char recvbuf[1024] = {0};
    char sendbuff[1024] = {0};

    while(fgets(sendbuff, 1024, stdin) != NULL){
        write(sockfd, sendbuff, strlen(sendbuff));
        read(sockfd, recvbuf, 1024);
        fputs(recvbuf, stdout);
        memset(recvbuf, 0, 1024);
        memset(sendbuff, 0, 1024);
    }

    exit(0);
}

server:

client:

实例二:支持多并发,单服务器----多客户端

/*************************************************************************
	> File Name: ser02.c
	> Author:
	> Mail:
	> Created Time: 2015年11月22日 星期日 10时22分17秒
 ************************************************************************/

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

int main()
{
   //int socket(int domain, int type, int protocol);
    int sockfd;
    sockfd = socket(AF_INET, SOCK_STREAM, 0);

    struct sockaddr_in seraddr;
    seraddr.sin_family = AF_INET;
    seraddr.sin_port = htons(8001);
    seraddr.sin_addr.s_addr = inet_addr("127.0.0.1");

    //int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
    bind(sockfd, (struct sockaddr *)&seraddr, sizeof(seraddr));

    listen(sockfd, 128);

    struct sockaddr_in cliaddr;
    socklen_t clilen;

    int connfd;
    char recvbuf[2014] = {0};
    int recvcount = 0;

    while(1){
        connfd = accept(sockfd, (struct sockaddr *)&cliaddr, &clilen);
        if(connfd == -1){
            perror("accept error.\n");
            exit(EXIT_FAILURE);
        }
        pid_t pid = fork();

        if(pid == 0){
            while(1){
                recvcount = read(connfd, recvbuf,1024);
                if (recvcount == 0){
                    printf("%s:%d is over.\n", inet_ntoa(cliaddr.sin_addr), ntohs(cliaddr.sin_port));
                    exit(0);
                }
                else if (recvcount == -1){
                    perror("read error.\n");
                    exit(EXIT_FAILURE);
                }
                printf("%d receive from %s:%d - ", getpid(), inet_ntoa(cliaddr.sin_addr), ntohs(cliaddr.sin_port));
                fputs(recvbuf, stdout);
                write(connfd, recvbuf, strlen(recvbuf));
                memset(recvbuf, 0, 1024);
            }
        }
        else if(pid > 0){

        }
        else if(pid < 0){
            perror("fork error.\n");
            exit(EXIT_FAILURE);
        }
    }

    exit(0);
}
/*************************************************************************
	> File Name: cli02.c
	> Author:
	> Mail:
	> Created Time: 2015年11月22日 星期日 11时56分05秒
 ************************************************************************/

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

int main()
{

    int sockfd, ret;
    struct sockaddr_in seraddr;
    seraddr.sin_family = AF_INET;
    seraddr.sin_port = htons(8001);
    seraddr.sin_addr.s_addr = inet_addr("127.0.0.1");

    sockfd = socket(AF_INET, SOCK_STREAM, 0);

    ret = connect(sockfd, (struct sockaddr*)&seraddr, sizeof(seraddr));
    if(ret == -1){
        perror("bind error.\n");
        exit(EXIT_FAILURE);
    }

    char recvbuf[1024] = {0};
    char sendbuff[1024] = {0};

    while(fgets(sendbuff, 1024, stdin) != NULL){
        write(sockfd, sendbuff, strlen(sendbuff));
        read(sockfd, recvbuf, 1024);
        printf("receive from %s:%d - ", inet_ntoa(seraddr.sin_addr), ntohs(seraddr.sin_port));
        fputs(recvbuf, stdout);
        memset(recvbuf, 0, 1024);
        memset(sendbuff, 0, 1024);
    }

    exit(0);
}

server:

client1:

client2:

client3:

server(此时三个客户端退出):

时间: 2024-11-08 15:50:22

linux支持并发的服务器回射程序实例的相关文章

unp TCP 客户端服务器回射程序中对SIGCHLD信号的处理

第五章中,有一个例子模拟客户端并发的终止TCP连接,服务器捕捉并处理SIGCHLD信号并调用waitpid函数防止僵死进程的出现.信号处理函数中核心的一句是: 1 while ( (pid = waitpid(-1, &statloc, WNOHANG)) > 0 ) 2 { 3 printf("wait child pid:%ld\n",(long)pid); 4 } 这是在使用wait函数时不能解决N个子进程同时终止时导致只有1个子进程被wait而其他N-1个子进程变

Socket 入门- 客户端回射程序

结果输出:------------------------------------------------------客户端:[email protected]:~/Public/C$ ./postBackCli.out 127.0.0.1connect OKaaaaaabbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbcc#close OK[email protected]:~/Public/C$ ---------------

回射程序改进3——消息的群发

前文列表: 简单的回射程序 回射程序改进1 回射程序改进2--群发消息(fork)错误的尝试 目的: 设计一个C/S程序,客户端发送/接收消息,服务端将从客户端接收到的消息群发给其它已连接套接字,产生 类似群聊的效果 相对于之前的改进: 1.客户端可以在服务端终止后得到通知 2.客户端使用shutdown()函数处理批量输入产生的问题 3.服务端使用select()函数管理套接字(单进程),而非使用fork()让每个子进程管理一个套接字(多进程) select函数介绍 程序代码: 客户端: 1

服务器客户端回射程序-自己设计包的结构

这次是个点对点,不过我自己设计包,包中包括发送的字符串的长度,和实际的字符串,使用结构体来表示. 客户端跟服务器在接收报文时,首先接收字符串的长度这一数值,然后将这一数值作为参数传入readn接收固定长度的字节数字符串. 看代码,首先是服务器端: 1 /*使用发送固定字节数报文的点对点聊天程序*/ 2 #include<stdio.h> 3 #include<unistd.h> 4 #include<sys/types.h> 5 #include<sys/sock

第五十六节,python实现支持并发、断点续传的Ftp程序

一.要求 1.用户md5认证 2.支持多用户同时登陆(并发) 3.进入用户的命令行模式,支持cd切换目录,ls查看目录子文件 4.执行命令(ipconfig) 5.传输文件: a.支持断点续传 b.传输中显示进度条 二.思路 1.客户端用户登录和注册: a.客户端仅提供用户名和密码,选择登录或注册,b.服务器端进行注册并将加密后的密码写入文件,最后返回给客户端是否登录或注册成功 2.ls和cd命令 a.客户端输入命令,服务器端处理并返回给客户端 3.执行命令: a.客户端发送需要执行的命令b.服

第十篇:基于TCP的一对回射客户/服务器程序及其运行过程分析( 上 )

前言 本文将讲解一对经典的客户/服务器回射程序,感受网络编程的大致框架( 该程序稍作改装即可演变成各种提供其他服务的程序 ):同时,还将对其运行过程加以分析,观察程序背后协议的执行细节,学习调试网络程序的技巧. 客户端 1 #include "unp.h" 2 3 void str_cli(FILE *fp, int sockfd); 4 5 int 6 main(int argc, char **argv) 7 { 8 int sockfd; 9 struct sockaddr_in

第十二篇:并发回射服务器的具体实现及其中僵尸子进程的清理( 上 )

前言 本文将分为两个部分,第一部分具体实现一对并发回射服务器/客户程序( 看过前面那篇文章的这部分可不看 重复了 ):第二部分为服务器添加僵尸子进程自动清理机制. 那么服务器具体怎么实现并发?怎么会有僵尸进程?僵尸进程又是什么?如何处理这些僵尸进程 ... 本文将为你一一解惑. 回射并发服务器 功能:接收用户发送过来的数据后再发送回用户,且能同时处理多个用户请求. 大体思路:每当收到用户请求,服务器就fork一个子进程,让子进程去处理客户请求. 实现代码: 1 #include "unp.h&q

第十一篇:基于TCP的一对回射客户/服务器程序及其运行过程分析( 下 )

执行分析 1. 打开服务器进程: 2. 执行netstat -a命令观察当前的连接状态: 第1条连接记录说明:绑定了本地主机的任意IP,端口为9877,目前处于监听状态. 3. 打开客户进程: 4. 执行netstat -a命令观察当前的连接状态,发现了两个新的连接: 以及 上面一个连接说明一个连接到服务器的连接,客户端临时端口是32818,目的端口正是先前的9877,连接状态为已建立,对应已连接套接字: 下面一个连接说明一个连接到客户端的连接,服务端端口为9877,目的端口是32818,这个连

UNIX网络编程卷1 回射客户程序 TCP客户程序设计范式

本文为senlie原创,转载请保留此地址:http://blog.csdn.net/zhengsenlie 下面我会介绍同一个使用 TCP 协议的客户端程序的几个不同版本,分别是停等版本.select 加阻塞式 I/O 版本. 非阻塞式 I/O 版本.fork 版本.线程化版本.它们都由同一个 main 函数调用来实现同一个功能,即回射程序客户端. 它从标准输入读入一行文本,写到服务器上,读取服务器对该行的回射,并把回射行写到标准输出上. 其中,非阻塞式 I/O 版本是所有版本中执行速度最快的,