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

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容