【Linux网络编程】端口绑定问题
00. 目錄
文章目錄
- 00. 目錄
- 01. 概述
- 02. 測試示例
- 03. 附錄
01. 概述
所謂綁定(bind)是指別人連接我只能通過我所綁定的端口,相當(dāng)于,我買了一個手機,別人要想聯(lián)系我,必須要知道我的手機號碼,這時候,我需要怎么辦呢?我需要給手機插上電話卡,固定一個電話號碼,這樣別人就能通過這個電話號碼聯(lián)系我。手機插上電話卡,固定一個電話號碼,類似于綁定(bind)的過程,綁定(bind)為了固定一個端口號,別的網(wǎng)絡(luò)程序就可以找到這個端口號,找到這個端口號就能找到這個端口號所對應(yīng)的網(wǎng)絡(luò)應(yīng)用程序。
在網(wǎng)絡(luò)編程里,通常都是在服務(wù)器里綁定(bind)端口,這并不是說客戶端里不能綁定(bind)端口,但這里需要注意的是,一個網(wǎng)絡(luò)應(yīng)用程序只能綁定一個端口( 一個套接字只能 綁定一個端口 )。
02. 測試示例
一個套接字不能同時綁定多個端口,如下:
#include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <sys/socket.h> #include <netinet/in.h> #include <arpa/inet.h>int main(int argc, char *argv[]) {char server_ip[30] = "10.221.20.12";int sockfd;sockfd = socket(AF_INET, SOCK_DGRAM, 0); //創(chuàng)建UDP套接字if(sockfd < 0){perror("socket");exit(-1);}// 初始化本地網(wǎng)絡(luò)信息struct sockaddr_in my_addr;bzero(&my_addr, sizeof(my_addr));my_addr.sin_family = AF_INET;my_addr.sin_port = htons(8000);my_addr.sin_addr.s_addr = htonl(INADDR_ANY);// 第一次綁定端口8000int err_log;err_log = bind(sockfd, (struct sockaddr*)&my_addr, sizeof(my_addr));if(err_log != 0){perror("bind 8000");close(sockfd); exit(-1);}// 又一次綁定別的端口9000, 會綁定失敗my_addr.sin_port = htons(9000);err_log = bind(sockfd, (struct sockaddr*)&my_addr, sizeof(my_addr));if(err_log != 0){perror("bind 9000");close(sockfd); exit(-1);}close(sockfd);return 0; }測試結(jié)果:
deng@itcast:/mnt/hgfs/LinuxHome/code.bak2$ gcc 1.c deng@itcast:/mnt/hgfs/LinuxHome/code.bak2$ ./a.out bind 9000: Invalid argument deng@itcast:/mnt/hgfs/LinuxHome/code.bak2$如果客戶端想綁定端口號,一定要調(diào)用發(fā)送信息函數(shù)之前綁定( bind )端口,因為在發(fā)送信息函數(shù)( sendto, 或 write ),系統(tǒng)會自動給當(dāng)前網(wǎng)絡(luò)程序分配一個隨機端口號,這相當(dāng)于隨機綁定了一個端口號,這里只會分配一次,以后通信就以這個隨機端口通信,我們再綁定端口號的話,就會綁定失敗。如果我們放在發(fā)送信息函數(shù)( sendto, 或 write )之前綁定,那樣程序?qū)⒁晕覀兘壎ǖ亩丝谔柊l(fā)送信息,不會再隨機分配一個端口號。
綁定失敗例子( UDP )如下:
#include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <sys/socket.h> #include <netinet/in.h> #include <arpa/inet.h>int main(int argc, char *argv[]) {char server_ip[30] = "10.221.20.12";int sockfd;sockfd = socket(AF_INET, SOCK_DGRAM, 0); //創(chuàng)建UDP套接字if(sockfd < 0){perror("socket");exit(-1);}struct sockaddr_in dest_addr;bzero(&dest_addr, sizeof(dest_addr));dest_addr.sin_family = AF_INET;dest_addr.sin_port = htons(8080); // 服務(wù)器的端口inet_pton(AF_INET, server_ip, &dest_addr.sin_addr);char send_buf[512] = "this is for test";// 如果前面沒有綁定端口,sendto()系統(tǒng)會隨機分配一個端口sendto(sockfd, send_buf, strlen(send_buf), 0, (struct sockaddr*)&dest_addr, sizeof(dest_addr));//發(fā)送數(shù)據(jù)// 初始化本地網(wǎng)絡(luò)信息struct sockaddr_in my_addr;bzero(&my_addr, sizeof(my_addr));my_addr.sin_family = AF_INET;my_addr.sin_port = htons(8000);my_addr.sin_addr.s_addr = htonl(INADDR_ANY);// sendto()后面綁定端口,綁定失敗int err_log;err_log = bind(sockfd, (struct sockaddr*)&my_addr, sizeof(my_addr));if(err_log != 0){perror("bind 8000");close(sockfd); exit(-1);}close(sockfd);return 0; }測試結(jié)果:
deng@itcast:/mnt/hgfs/LinuxHome/code.bak2$ gcc 1.c deng@itcast:/mnt/hgfs/LinuxHome/code.bak2$ ./a.out bind 8000: Invalid argument deng@itcast:/mnt/hgfs/LinuxHome/code.bak2$03. 附錄
總結(jié)
以上是生活随笔為你收集整理的【Linux网络编程】端口绑定问题的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 【Linux网络编程】UDP编程
- 下一篇: linux 其他常用命令