TCP并发服务器多线程

1.创建线程‐‐pthread_create
int pthread_create( pthread_t *thread, // 线程 ID = 无符号长整型
const pthread_attr_t *attr, // 线程属性, NULL
void *(*start_routine)(void *), // 线程处理函数
void *arg); // 线程处理函数
参数:
pthread: 传出参数,线程创建成功之后,会被设置一个合适的值
attr: 默认传 NULL
start_routine: 子线程的处理函数
arg: 回调函数的参数
返回值:
成功 :0
错误 : 错误号 //perror 不能使用该函数打印错误信息
主线程先退出,子线程会被强制结束
验证线程直接共享全局变量
client

objectivec 复制代码
#include<stdio.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/ip.h> /* superset of previous */
#include<stdlib.h>
#include <strings.h>
#include<unistd.h>

#include <string.h>

#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>

#include <sys/types.h>          /* See NOTES */
#include <sys/socket.h>
//./client 192.168.182.144 5001




#define SERV_PORT 5001
#define SERV_IP_ADDR "192.168.157.34"
#define BUFSIZE 1024
#define QUIT_STR "QUIT"


int main(int argc,char **argv)
{
        int fd = -1;
        if(argc!=3)
        {
        exit(1);
        }
        int port = -1;
        port = atoi(argv[2]);
        fd = socket(AF_INET,SOCK_STREAM,0);
        if(fd<0)
        {

                perror("socket");
                exit(1);
        }

        struct sockaddr_in sin;
        bzero(&sin,sizeof(sin));
        sin.sin_family = AF_INET;
        sin.sin_port = htons(port);
        sin.sin_addr.s_addr = inet_addr(argv[1]);

        if(connect(fd,(struct sockaddr*)&sin,sizeof(sin))<0)
        {
                perror("connect");
                exit(1);
        }
        char buf[BUFSIZE];
        while(1)
        {
                bzero(buf,BUFSIZE);
                if(fgets(buf,BUFSIZE-1,stdin)==NULL)
                {
                        continue;
                }
               write(fd,buf,strlen(buf));
                if(!strncasecmp(buf,QUIT_STR,strlen(QUIT_STR)))
                {

                        break;
                }


        }
        return 0;


}

service

objectivec 复制代码
#include<stdio.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/ip.h> /* superset of previous */
#include<stdlib.h>
#include <strings.h>
#include<unistd.h>
#include <string.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <sys/types.h>          /* See NOTES */
#include <sys/socket.h>
#include<pthread.h>
#define SERV_PORT 5001
#define SERV_IP_ADDR "192.168.157.34"
#define BACKLOG 5
#define BUFSIZE 1024
#define QUIT_STR "QUIT"
void* client_data_handle(void* arg);
int main(int argc,char *argv[])
{
        //socket
        int fd = -1;
        pthread_t tid;//定义pthread_t 类型的参数,成功时指向的内存单元为新创建线程的线程ID
        fd = socket(AF_INET,SOCK_STREAM,0);//三个参数:ipv4网络协议,流式套接字,唯一对应TCP,协议一般填写0。成功返回文件描述符fd
        if(fd < 0)
        {
                perror("socket");
                exit(1);
        }
        /*
           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. 
           struct in_addr {
           uint32_t       s_addr;      address in network byte order 
           };  */

        struct sockaddr_in sin;

        bzero(&sin,sizeof(sin));
        sin.sin_family = AF_INET;
        sin.sin_port = htons(SERV_PORT);
        sin.sin_addr.s_addr = INADDR_ANY;
        //bing
        if(bind(fd,(struct sockaddr *)&sin,sizeof(sin))<0)//绑定函数,与sockaddr_in结构体联系
        {
                perror("bind");
                exit(1);
        }

        //listen

        if(listen(fd,BACKLOG)<0)//监听函数,表示系统允许2*backlog+1个客户端同时进行三次握手
        {
                perror("listen");
                exit(1);
        }
       int newfd = -1;
        char ipv4_addr[16];
        struct sockaddr_in clin;

        socklen_t sockaddr_len = sizeof(clin);
        while(1)
        {
                newfd = accept(fd,(struct sockaddr *)&clin,&sockaddr_len);//持续接收,参数:之前的文件描述符,新定义的结构体,socklen_t 类型的结构体长度。accept函数用法同bind,但结构体不同,结构体内容一样,第三个参数不同,bind 为结构体长度,accept需要为socklen_t 类型的变量,变量值为结构体的长度。返回为另一个文件描述符newfd。即服务器接收到用户端的信息后会返回一个新的文件描述符。
                if(newfd<0)
                {
                        perror("accept");
                        exit(1);
                }
                if(!inet_ntop(AF_INET,(void *)&clin.sin_addr,ipv4_addr,sizeof(clin)))
                {
                        perror("inet_ntop");
                        exit(1);
                }
                printf("client :(%s,%d) is connect!\n",ipv4_addr,ntohs(clin.sin_port));

                pthread_create(&tid,NULL,client_data_handle,(void *)&newfd);//创建线程,线程为tid,成功时tid指向的内存单元为新线程的id,线程描述为NULL,新创建线程从此函数开始运行,子线程函数的参数,有参数时输入参数的地址,当多余一个参数时用结构体传入

        }
        close(fd);
        //read



}

void* client_data_handle(void* arg) //子线程处理函数
{
        int newfd = *(int *)arg; //将传入的参数的值给newfd
        char buf[BUFSIZE];
        int ret = -1;
        printf("handle thread :newfd= %d\n",newfd);
        while(1)
        {

                do{
                        bzero(buf,BUFSIZE);
                        ret = read(newfd,buf,BUFSIZE-1);

                }while(ret<1);
                if(ret<0)
                {
                        perror("read");
                        exit(1);
                }
                if(!ret)
                {
                        break;
                }
                printf("receive data :%s \n",buf);

                if(!strncasecmp(buf,QUIT_STR,strlen(QUIT_STR)))
                {
                        printf("Client is exiting!\n");
                        break;
                }
        }
        close(newfd);
        return NULL;

}

结果

相关推荐
东软吴彦祖13 分钟前
包安装利用 LNMP 实现 phpMyAdmin 的负载均衡并利用Redis实现会话保持nginx
linux·redis·mysql·nginx·缓存·负载均衡
好一点,更好一点14 分钟前
systemC示例
开发语言·c++·算法
不爱学英文的码字机器17 分钟前
[操作系统] 环境变量详解
开发语言·javascript·ecmascript
martian66522 分钟前
第17篇:python进阶:详解数据分析与处理
开发语言·python
五味香26 分钟前
Java学习,查找List最大最小值
android·java·开发语言·python·学习·golang·kotlin
时韵瑶31 分钟前
Scala语言的云计算
开发语言·后端·golang
卷卷的小趴菜学编程35 分钟前
c++之List容器的模拟实现
服务器·c语言·开发语言·数据结构·c++·算法·list
艾杰Hydra38 分钟前
LInux配置PXE 服务器
linux·运维·服务器
多恩Stone42 分钟前
【ubuntu 连接显示器无法显示】可以通过 ssh 连接 ubuntu 服务器正常使用,但服务器连接显示器没有输出
服务器·ubuntu·计算机外设
慵懒的猫mi1 小时前
deepin分享-Linux & Windows 双系统时间不一致解决方案
linux·运维·windows·mysql·deepin