Linux進程間通信――使用流套接字
睿豐德科技 專注RFID識別技術和條碼識別技術與管理軟件的集成項目。質量追溯系統、MES系統、金蝶與條碼系統對接、用友與條碼系統對接
前面說到的進程間的通信,所通信的進程都是在同一臺計算機上的,而使用socket進行通信的進程可以是同一臺計算機的進程,也是可以是通過網絡連接起來的不同計算機上的進程。通常我們使用socket進行網絡編程,這里將會簡單地講述如何使用socket進行簡單的網絡編程。
一、什么是socket
socket,即套接字是一種通信機制,憑借這種機制,客戶/服務器(即要進行通信的進程)系統的開發工作既可以在本地單機上進行,也可以跨網絡進行。也就是說它可以讓不在同一臺計算機但通過網絡連接計算機上的進程進行通信。也因為這樣,套接字明確地將客戶端和服務器區分開來。
二、套接字的屬性
套接字的特性由3個屬性確定,它們分別是:域、類型和協議。
1、套接字的域
它指定套接字通信中使用的網絡介質,最常見的套接字域是AF_INET,它指的是Internet網絡。當客戶使用套接字進行跨網絡的連接時,它就需要用到服務器計算機的IP地址和端口來指定一臺聯網機器上的某個特定服務,所以在使用socket作為通信的終點,服務器應用程序必須在開始通信之前綁定一個端口,服務器在指定的端口等待客戶的連接。另一個域AF_UNIX表示UNIX文件系統,它就是文件輸入/輸出,而它的地址就是文件名。
2、套接字類型
因特網提供了兩種通信機制:流(stream)和數據報(datagram),因而套接字的類型也就分為流套接字和數據報套接字。這里主要講流套接字。
流套接字由類型SOCK_STREAM指定,它們是在AF_INET域中通過TCP/IP連接實現,同時也是AF_UNIX中常用的套接字類型。流套接字提供的是一個有序、可靠、雙向字節流的連接,因此發送的數據可以確保不會丟失、重復或亂序到達,而且它還有一定的出錯后重新發送的機制。
與流套接字相對的是由類型SOCK_DGRAM指定的數據報套接字,它不需要建立連接和維持一個連接,它們在AF_INET中通常是通過UDP/IP協議實現的。它對可以發送的數據的長度有限制,數據報作為一個單獨的網絡消息被傳輸,它可能會丟失、復制或錯亂到達,UDP不是一個可靠的協議,但是它的速度比較高,因為它并一需要總是要建立和維持一個連接。
3、套接字協議
只要底層的傳輸機制允許不止一個協議來提供要求的套接字類型,我們就可以為套接字選擇一個特定的協議。通常只需要使用默認值。
三、套接字地址
每個套接字都有其自己的地址格式,對于AF_UNIX域套接字來說,它的地址由結構sockaddr_un來描述,該結構定義在頭文件sys/un.h中,它的定義如下:
[cpp] view plaincopyprint?
在本例子中,我們啟動了一個服務器程序和三個客戶程序,從運行的結果來看,客戶端發送給服務器程序的所有請求都得到了處理,即把A變成了B。對于服務器和客戶程序之間使用的read和write系統調用跟使用命名管道時阻塞的read、write系統調用一樣。例如客戶程序調用read時,如果服務器程序沒有向指定的客戶程序的socket中寫入信息,則read調用會一直阻塞。
七、流式套接字給我印象
給我的感覺是流式套接字很像命名管道,但是它卻可以使不在同一臺計算機而通過網絡連接的不同計算機上的進程進行通信,功能真是非常的強大。RFID管理系統集成商 RFID中間件 條碼系統中間層 物聯網軟件集成
- struct sockaddr_un{
- sa_family_t sun_family;//AF_UNIX,它是一個短整型
- char sum_path[];//路徑名
- };
- struct sockaddr_in{
- short int sin_family;//AF_INET
- unsigned short int sin_port;//端口號
- struct in_addr sin_addr;//IP地址
- };
- struct in_addr{
- unsigned long int s_addr;
- };
- int socket(int domain, int type, int protocol);
- int bind( int socket, const struct sockaddr *address, size_t address_len);
- int listen(int socket, int backlog);
- int accept(int socket, struct sockaddr *address, size_t *address_len);
- int connect(int socket, const struct sockaddr *address, size_t address_len);
- #include <unistd.h>
- #include <sys/types.h>
- #include <sys/socket.h>
- #include <netinet/in.h>
- #include <signal.h>
- #include <stdio.h>
- #include <stdlib.h>
- int main()
- {
- int server_sockfd = -1;
- int client_sockfd = -1;
- int client_len = 0;
- struct sockaddr_in server_addr;
- struct sockaddr_in client_addr;
- //創建流套接字
- server_sockfd = socket(AF_INET, SOCK_STREAM, 0);
- <span style="white-space:pre"> </span>//設置服務器接收的連接地址和監聽的端口
- server_addr.sin_family = AF_INET;//指定網絡套接字
- server_addr.sin_addr.s_addr = htonl(INADDR_ANY);//接受所有IP地址的連接
- server_addr.sin_port = htons(9736);//綁定到9736端口
- //綁定(命名)套接字
- bind(server_sockfd, (struct sockaddr*)&server_addr, sizeof(server_addr));
- //創建套接字隊列,監聽套接字
- listen(server_sockfd, 5);
- //忽略子進程停止或退出信號
- signal(SIGCHLD, SIG_IGN);
- while(1)
- {
- char ch = '\0';
- client_len = sizeof(client_addr);
- printf("Server waiting\n");
- //接受連接,創建新的套接字
- client_sockfd = accept(server_sockfd, (struct sockaddr*)&client_addr, &client_len);
- if(fork() == 0)
- {
- //子進程中,讀取客戶端發過來的信息,處理信息,再發送給客戶端
- read(client_sockfd, &ch, 1);
- sleep(5);
- ch++;
- write(client_sockfd, &ch, 1);
- close(client_sockfd);
- exit(0);
- }
- else
- {
- //父進程中,關閉套接字
- close(client_sockfd);
- }
- }
- }
- #include <unistd.h>
- #include <sys/types.h>
- #include <sys/socket.h>
- #include <netinet/in.h>
- #include <arpa/inet.h>
- #include <stdio.h>
- #include <stdlib.h>
- int main()
- {
- int sockfd = -1;
- int len = 0;
- struct sockaddr_in address;
- int result;
- char ch = 'A';
- //創建流套接字
- sockfd = socket(AF_INET, SOCK_STREAM, 0);
- //設置要連接的服務器的信息
- address.sin_family = AF_INET;//使用網絡套接字
- address.sin_addr.s_addr = inet_addr("127.0.0.1");//服務器地址
- address.sin_port = htons(9736);//服務器所監聽的端口
- len = sizeof(address);
- //連接到服務器
- result = connect(sockfd, (struct sockaddr*)&address, len);
- if(result == -1)
- {
- perror("ops:client\n");
- exit(1);
- }
- //發送請求給服務器
- write(sockfd, &ch, 1);
- //從服務器獲取數據
- read(sockfd, &ch, 1);
- printf("char form server = %c\n", ch);
- close(sockfd);
- exit(0);
- }
