Linux下基于socket多线程并发通信的实现

在Linux 下开发网络通信系统可以充分发挥Linux 系统出色的网络性能,本文介绍了在Linux 操作系统下基于TCP/IP 协议Socket 套接口的通信机制以及多线程编程知识与技巧,并给出多线程方式实现多用户与服务端(C/S)并发通信模型的详细算法,最后展现了用C 编写的多用户与服务器通信的应用实例并附有运行结果及截图。[喝小酒的网摘]http://blog.hehehehehe.cn/a/9145.htm
关键词:Linux;套接字;多线程;并发服务器;
Socket 是建立在传输层协议(主要是TCP 和UDP)上的一种套接字规范,最初由美国加州Berkley 大学提出,为UNIX 系统开发的网络通信接口,它定义了两台计算机之间通信的规范,socket 屏蔽了底层通信软件和具体操作系统的差异,使得任何两台安装了TCP 协议软件和实现了Socket 规范的计算机之间的通信成为可能,Socket 接口是TCP/IP 网络最为通用的应用接口,也是在Internet 上进行网络程序应用开发最通用的API[1],本文介绍了Socket通信的基本机制以及采用多线程技术实现并发通信的基本原理,并给出实例。

/*************************************************** 
* 文件名:pthread_server.c 
* 文件描述:创建子线程来接收客户端的数据 
***************************************************/  
#include <sys/types.h>  
#include <sys/socket.h>  
#include <stdio.h>  
#include <netinet/in.h>  
#include <arpa/inet.h>  
#include <unistd.h>  
#include <stdlib.h>  
#include <pthread.h>  
void *rec_data(void *fd);  
int main(int argc,char *argv[])  
{  
       int server_sockfd;  
    int *client_sockfd;  
       int server_len, client_len;  
       struct sockaddr_in server_address;  
       struct sockaddr_in client_address;  
       struct sockaddr_in tempaddr;  
       int i,byte;  
       char char_recv,char_send;  
       socklen_t templen;  
       server_sockfd = socket(AF_INET, SOCK_STREAM, 0);//创建套接字  
   
       server_address.sin_family = AF_INET;  
       server_address.sin_addr.s_addr =  htonl(INADDR_ANY);  
       server_address.sin_port = htons(9734);  
       server_len = sizeof(server_address);  
        
       bind(server_sockfd, (struct sockaddr *)&server_address, server_len);//绑定套接字  
       templen = sizeof(struct sockaddr);  
   
       printf("server waiting for connect/n");  
       while(1){  
              pthread_t thread;//创建不同的子线程以区别不同的客户端  
              client_sockfd = (int *)malloc(sizeof(int));  
              client_len = sizeof(client_address);  
              *client_sockfd = accept(server_sockfd,(struct sockaddr *)&client_address, (socklen_t *)&client_len);  
              if(-1==*client_sockfd){  
                     perror("accept");  
                     continue;  
              }  
              if(pthread_create(&thread, NULL, rec_data, client_sockfd)!=0)//创建子线程  
              {  
                     perror("pthread_create");  
                     break;  
              }  
       }  
       shutdown(*client_sockfd,2);  
       shutdown(server_sockfd,2);  
}  
/***************************************** 
* 函数名称:rec_data 
* 功能描述:接受客户端的数据 
* 参数列表:fd——连接套接字 
* 返回结果:void 
*****************************************/  
void *rec_data(void *fd)  
{  
       int client_sockfd;  
       int i,byte;  
       char char_recv[100];//存放数据  
       client_sockfd=*((int*)fd);  
       for(;;)  
       {  
              if((byte=recv(client_sockfd,char_recv,100,0))==-1)  
              {  
                     perror("recv");  
                     exit(EXIT_FAILURE);   
              }  
              if(strcmp(char_recv, "exit")==0)//接受到exit时,跳出循环  
                     break;  
              printf("receive from client is %s/n",char_recv);//打印收到的数据  
       }  
       free(fd);  
       close(client_sockfd);  
       pthread_exit(NULL);  
}  
   
   
/*************************************************** 
* 文件名:pthread_client.c 
* 文件描述:创建子线程来接收客户端的数据 
***************************************************/  
#include <sys/types.h>  
#include <sys/socket.h>  
#include <stdio.h>  
#include <netinet/in.h>  
#include <arpa/inet.h>  
#include <unistd.h>  
#include <stdlib.h>  
int main(int argc,char *argv[])  
{  
       int sockfd;  
       int len;  
       struct sockaddr_in address;     
       int result;  
       int i,byte;  
       char char_send[100] = { 0 };  
    if((sockfd = socket(AF_INET, SOCK_STREAM, 0))==-1)  
       {  
              perror("socket");  
              exit(EXIT_FAILURE);  
       }  
    if(argc != 3){  
      printf("Usage: fileclient <address> <port>/n");//用法:文件名 服务器IP地址 服务器端口地址  
      return 0;  
   }  
   if((sockfd = socket(AF_INET,SOCK_STREAM,0)) == -1){  
       perror("sock");  
       exit(1);  
   }  
   bzero(&address,sizeof(address));  
   address.sin_family = AF_INET;  
   address.sin_port = htons(atoi(argv[2]));  
   inet_pton(AF_INET,argv[1],&address.sin_addr);  
len = sizeof(address);  
   
    if((result = connect(sockfd, (struct sockaddr *)&address, len))==-1)  
       {  
              perror("connect");  
              exit(EXIT_FAILURE);  
       }  
   
       for(;;)  
       {  
             scanf("%s", char_send);//输入发送数据  
              fflush(stdin);//清除输入缓存  
              if(strcmp(char_send, "exit")==0){//如果输入exit,跳出循环  
                     if((byte=send(sockfd,char_send,100,0))==-1)  
                     {  
                            perror("send");  
                            exit(EXIT_FAILURE);  
                     }             
                     break;  
              }                    
              if((byte=send(sockfd,char_send,100,0))==-1)  
              {  
                     perror("send");  
                     exit(EXIT_FAILURE);  
              }  
       }  
    close(sockfd);  
    exit(0);  
}   

编译服务器端程序 pthread_server.c :

$gcc pthread_server.c –o server –lpthread

编译客户端程序 pthread_client.c:

$gcc pthread_client.c –o client

编译在开发板上跑的客户端程序:

$arm-linux-gcc client.c –o arm_client

先启动服务器端的程序 server:

$./server

打开另一个终端,启动客户端的程序 client:

$./client 127.0.0.1 9734

把 arm_client 放到开发板上,启动 arm_client:

$./arm_client 219.222.170.9 9734

结果 :

服务器端:

tongs@tong's-desktop:~/c_c++_program/sock_inet_comm2$ ./server

server waiting for connect

receive from client is client

receive from client is client

receive from client is arm_client

receive from client is arm_client

客户端:

tongs@tong's-desktop:~/c_c++_program/sock_inet_comm2$ ./client 127.0.0.1 9734

client

client

exit

开发板服务器端

[/mnt/yaffs/nfs_share/sock_tcp/thread_socket]./arm_client 219.222.170.9 9734

arm_client

arm_client

exit
[喝小酒的网摘]http://blog.hehehehehe.cn/a/9145.html

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,884评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,347评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,435评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,509评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,611评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,837评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,987评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,730评论 0 267
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,194评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,525评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,664评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,334评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,944评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,764评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,997评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,389评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,554评论 2 349

推荐阅读更多精彩内容