struct sockaddr { u_charsa_len;//長度 u_short sa_family;//協議 char sa_data[14];//數據 };
struct sockaddr_in { u_char sin_len;//長度 u_short sin_family;//協議 u_short sin_port;//端口 structin_addr sin_addr;//ip地址 char sin_zero[8];//數據 };
structin_addr { u_longs_addr; };
用來表示一個32位的IPv4地址,其字節順序爲網絡順序。linux
功能:建立一個新的套接字,返回套接字描述符編程
參數說明:緩存
domain:域類型,指明使用的協議棧,如TCP/IP使用的是 PF_INET ,其餘還有AF_INET六、AF_UNIX服務器
type:指明須要的服務類型, 如網絡
SOCK_DGRAM:數據報服務,UDP協議dom
SOCK_STREAM:流服務,TCP協議socket
protocol:通常都取0(由系統根據服務類型選擇默認的協議)函數
功能:爲套接字指明一個本地端點地址指針
TCP/IP協議使用sockaddr_in結構,包含IP地址和端口號,服務器使用它來指明熟知的端口號,而後等待鏈接code
參數說明:
sockfd:套接字描述符,指明建立鏈接的套接字
my_addr:本地地址,IP地址和端口號
addrlen:地址長度
套接口中port=0表示由內核指定端口號,設定sin_addr爲INADDR_ANY,由內核指定IP地址。
功能:
面向鏈接的套接字使用它將一個套接字置爲被動模式,並準備接收傳入鏈接。用於服務器,指明某個套接字鏈接是被動的
參數說明:
Sockfd:套接字描述符,指明建立鏈接的套接字
input_queue_size:該套接字使用的隊列長度,指定在請求隊列中容許的最大請求數
功能:獲取傳入鏈接請求,返回新的鏈接的套接字描述符。
爲每一個新的鏈接請求建立了一個新的套接字,服務器只對新的鏈接使用該套接字,原來的監聽套接字接收其餘的鏈接請求。新的鏈接上傳輸數據使用新的套接字,使用完畢,服務器將關閉這個套接字。
參數說明:
Sockfd:套接字描述符,指明正在監聽的套接字
addr:提出鏈接請求的主機地址
addrlen:地址長度
功能: 同遠程服務器創建主動鏈接,成功時返回0,若鏈接失敗返回-1。
參數說明:
Sockfd:套接字描述符,指明建立鏈接的套接字
Server_addr:指明遠程端點:IP地址和端口號
sockaddr_len :地址長度
功能:
在TCP鏈接上發送數據,返回成功傳送數據的長度,出錯時返回-1。send會將外發數據複製到OS內核中,也可使用send發送面向鏈接的UDP報文。
參數說明:
sockfd:套接字描述符
data:指向要發送數據的指針
data_len:數據長度
flags:一般爲0
若是send()函數的返回值小於len的話,則你須要再次發送剩下的數據。802.3,MTU爲1492B,若是包小於1K,那麼send()通常都會一次發送光的。
功能:
從TCP接收數據,返回實際接收的數據長度,出錯時返回-1。
服務器使用其接收客戶請求,客戶使用它接受服務器的應答。若是沒有數據,將阻塞。若是TCP收到的數據大於(/小於)緩存的大小,只抽出可以填滿緩存的足夠數據(/抽出全部數據並返回它實際接收的字節數)。也可使用recv接收面向鏈接的UDP的報文,若緩存不能裝下整個報文,填滿緩存後剩下的數據將被丟棄。
參數說明:
Sockfd:套接字描述符
Buf:指向內存塊的指針
Buf_len:內存塊大小,以字節爲單位
flags:通常爲0(MSG_WAITALL接收到指定長度數據時才返回),設置爲 MSG_DONTWAIT爲非阻塞
功能:
撤銷套接字.若是隻有一個進程使用,當即終止鏈接並撤銷該套接字,若是多個進程共享該套接字,將引用數減一,若是引用數降到零,則關閉鏈接並撤銷套接字。
參數說明:
#include <sys/types.h> #include <sys/socket.h> #include <stdio.h> #include <netinet/in.h> #include <arpa/inet.h> #include <unistd.h> #include <string.h> #include <stdlib.h> #include <fcntl.h> #include <sys/shm.h> #define MYPORT 8887 #define QUEUE 20 #define BUFFER_SIZE 1024 int main() { ///定義sockfd int server_sockfd = socket(AF_INET,SOCK_STREAM, 0); ///定義sockaddr_in struct sockaddr_in server_sockaddr; server_sockaddr.sin_family = AF_INET; server_sockaddr.sin_port = htons(MYPORT); server_sockaddr.sin_addr.s_addr = htonl(INADDR_ANY); ///bind,成功返回0,出錯返回-1 if(bind(server_sockfd,(struct sockaddr *)&server_sockaddr,sizeof(server_sockaddr))==-1) { perror("bind"); exit(1); } printf("監聽%d端口\n",MYPORT); ///listen,成功返回0,出錯返回-1 if(listen(server_sockfd,QUEUE) == -1) { perror("listen"); exit(1); } ///客戶端套接字 char buffer[BUFFER_SIZE]; struct sockaddr_in client_addr; socklen_t length = sizeof(client_addr); printf("等待客戶端鏈接\n"); ///成功返回非負描述字,出錯返回-1 int conn = accept(server_sockfd, (struct sockaddr*)&client_addr, &length); if(conn<0) { perror("connect"); exit(1); } printf("客戶端成功鏈接\n"); while(1) { memset(buffer,0,sizeof(buffer)); int len = recv(conn, buffer, sizeof(buffer),0); //客戶端發送exit或者異常結束時,退出 if(strcmp(buffer,"exit\n")==0 || len<=0) break; printf("來自客戶端數據:%s\n",buffer); send(conn, buffer, len, 0); printf("發送給客戶端數據:%s\n",buffer); } close(conn); close(server_sockfd); return 0; }
#include <sys/types.h> #include <sys/socket.h> #include <stdio.h> #include <netinet/in.h> #include <arpa/inet.h> #include <unistd.h> #include <string.h> #include <stdlib.h> #include <fcntl.h> #include <sys/shm.h> #define MYPORT 8887 #define BUFFER_SIZE 1024 char* SERVER_IP = "127.0.0.1"; int main() { ///定義sockfd int sock_cli = socket(AF_INET,SOCK_STREAM, 0); ///定義sockaddr_in struct sockaddr_in servaddr; memset(&servaddr, 0, sizeof(servaddr)); servaddr.sin_family = AF_INET; servaddr.sin_port = htons(MYPORT); ///服務器端口 servaddr.sin_addr.s_addr = inet_addr(SERVER_IP); ///服務器ip printf("鏈接%s:%d\n",SERVER_IP,MYPORT); ///鏈接服務器,成功返回0,錯誤返回-1 if (connect(sock_cli, (struct sockaddr *)&servaddr, sizeof(servaddr)) < 0) { perror("connect"); exit(1); } printf("服務器鏈接成功\n"); char sendbuf[BUFFER_SIZE]; char recvbuf[BUFFER_SIZE]; while (fgets(sendbuf, sizeof(sendbuf), stdin) != NULL) { printf("向服務器發送數據:%s\n",sendbuf); send(sock_cli, sendbuf, strlen(sendbuf),0); ///發送 if(strcmp(sendbuf,"exit\n")==0) break; recv(sock_cli, recvbuf, sizeof(recvbuf),0); ///接收 printf("從服務器接收數據:%s\n",recvbuf); memset(sendbuf, 0, sizeof(sendbuf)); memset(recvbuf, 0, sizeof(recvbuf)); } close(sock_cli); return 0; }
好比要產看socket()函數的用法,能夠用man socket 指令:
man socket
若是以爲linux下看的不習慣,能夠導入到文件中用notepad++打開
man socket > ~/socket.txt //將socket的man手冊導出到當前用戶根目錄下的socket.txt文件中