linux中關於socket通訊取得本地的來源連接埠號碼的實作方法
這篇文章主要介紹了linux socket通訊取得本地的來源連接埠號碼的相關資料,需要的朋友可以參考下
關於TCP IP網路通訊的資料非常多,TCP IP透過IP封包模式進行端對端通訊。典型的TCP封包如下
可以看到封包包含了來源埠號和目的埠號,當客戶端socket向服務端發起連線時,系統會給socket隨機分配一個來源連接埠號,我們可以透過getsocketname來取得連接成功的socket的原始連接埠資訊。
參數原型
#include <sys/socket.h> int getsockname(int sockfd, struct sockaddr *addr, socklen_t *addrlen);
參數:
sockfd socket連接的句柄
#addr網路位址指針,用來儲存本地端socket位址訊息,
addrlen addr的空間大小
返回結果,如果呼叫成功,返回0,並將本地網路位址資訊存放在addr裡面,失敗返回-1,並透過errno反應錯誤訊息。
source_port.cpp
#include <cstring> #include <cstdio> #include <cstdlib> #include <sys/socket.h> #include <sys/types.h> #include <netinet/in.h> #include <netinet/ip.h> #include <netdb.h> #include <errno.h> #include <unistd.h> #include <arpa/inet.h> void safe_close(int &sock); int main(int argc, char *argv[]) { int sockfd = 0, n = 0; socklen_t len = 0; char host[512] = {0}; char buf[1024] = {0}; struct hostent *server; struct sockaddr_in serv_addr, loc_addr; if (argc < 2) { printf("Please input host name\n"); exit(-1); } strncpy(host, argv[1], sizeof(host)); server = gethostbyname(host);// 判断输入的域名是否正确 if (NULL == server) { printf("find host: %s failed.\n", host); exit(-1); } if (-1 == (sockfd = socket(AF_INET, SOCK_STREAM, 0))) {// 创建socket memset(buf, 0, sizeof(buf)); snprintf(buf, sizeof(buf), "new socket failed. errno: %d, error: %s", errno, strerror(errno)); perror(buf); exit(-1); } memset(&serv_addr, 0, sizeof(serv_addr)); serv_addr.sin_family = AF_INET; serv_addr.sin_port = htons(80);// http标准端口号 memcpy(&serv_addr.sin_addr.s_addr, server->h_addr, server->h_length); if (-1 == inet_pton(AF_INET, host, &serv_addr.sin_addr)) { memset(buf, 0, sizeof(buf)); snprintf(buf, sizeof(buf), "inet_pton failed. errno: %d, error: %s", errno, strerror(errno)); perror(buf); exit(-1); } if (-1 == connect(sockfd, (struct sockaddr *)&serv_addr, sizeof(serv_addr))) {// 连接socket memset(buf, 0, sizeof(buf)); snprintf(buf, sizeof(buf), "connect socket failed. errno: %d, error: %s", errno, strerror(errno)); perror(buf); exit(-1); } printf("connect to %s success.\n", host); len = sizeof(sizeof(loc_addr)); memset(&loc_addr, 0, len); if (-1 == getsockname(sockfd, (struct sockaddr *)&loc_addr, &len)) {// 获取socket绑定的本地address信息 memset(buf, 0, sizeof(buf)); snprintf(buf, sizeof(buf), "get socket name failed. errno: %d, error: %s", errno, strerror(errno)); perror(buf); safe_close(sockfd); exit(-1); } if (loc_addr.sin_family == AF_INET) {// 打印信息 printf("local port: %u\n", ntohs(loc_addr.sin_port)); } safe_close(sockfd); return 0; } void safe_close(int &sock) { if (-1 != sock) { shutdown(sock, SHUT_RDWR); sock = -1; } }
本程式首先會啟動一個socket連接一個普通的http伺服器(baidu,qq,163,csdn),當socket連通時就透過getsocketname取得連線綁定的本機位址,並透過該位址取得來源埠號。
終端機1: 編譯及執行
$ g++ source_port.cpp $ ./a.out www.baidu.com connect to www.baidu.com success. local port: 39702
終端2:透過tcpdump抓包驗證
$ sudo tcpdump host www.baidu.com -v tcpdump: listening on eth0, link-type EN10MB (Ethernet), capture size 65535 bytes 18:38:32.381448 IP (tos 0x0, ttl 64, id 35033, offset 0, flags [DF], proto TCP (6), length 60) icentos.39702 > 220.181.111.188.http: Flags [S], cksum 0x8cd2 (incorrect -> 0x596a), seq 2381397554, win 29200, options [mss 1460,sackOK,TS val 3513497323 ecr 0,nop,wscale 7], length 0 18:38:32.425904 IP (tos 0x0, ttl 55, id 35033, offset 0, flags [DF], proto TCP (6), length 60) 220.181.111.188.http > icentos.39702: Flags [S.], cksum 0xc315 (correct), seq 3561856904, ack 2381397555, win 8192, options [mss 1424,sackOK,nop,nop,nop,nop,nop,nop,nop,nop,nop,nop,nop,wscale 5], length 0 18:38:32.425930 IP (tos 0x0, ttl 64, id 35034, offset 0, flags [DF], proto TCP (6), length 40)
對比終端一和終端二表示取得的來源連接埠位址是正確的。
總結
#以上是linux中關於socket通訊取得本地的來源連接埠號碼的實作方法的詳細內容。更多資訊請關注PHP中文網其他相關文章!

熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

AI Hentai Generator
免費產生 AI 無盡。

熱門文章

熱工具

記事本++7.3.1
好用且免費的程式碼編輯器

SublimeText3漢化版
中文版,非常好用

禪工作室 13.0.1
強大的PHP整合開發環境

Dreamweaver CS6
視覺化網頁開發工具

SublimeText3 Mac版
神級程式碼編輯軟體(SublimeText3)

熱門話題

無法以 root 身份登錄 MySQL 的原因主要在於權限問題、配置文件錯誤、密碼不符、socket 文件問題或防火牆攔截。解決方法包括:檢查配置文件中 bind-address 參數是否正確配置。查看 root 用戶權限是否被修改或刪除,並進行重置。驗證密碼是否準確無誤,包括大小寫和特殊字符。檢查 socket 文件權限設置和路徑。檢查防火牆是否阻止了 MySQL 服務器的連接。

C語言條件編譯是一種根據編譯時條件選擇性編譯代碼塊的機制,入門方法有:使用#if和#else指令根據條件選擇代碼塊。常用條件表達式包括STDC、_WIN32和linux。實戰案例:根據操作系統打印不同消息。根據系統位數使用不同的數據類型。根據編譯器支持不同的頭文件。條件編譯增強了代碼的可移植性和靈活性,使其適應編譯器、操作系統和CPU架構變化。

Linux的五個基本組件是:1.內核,管理硬件資源;2.系統庫,提供函數和服務;3.Shell,用戶與系統交互的接口;4.文件系統,存儲和組織數據;5.應用程序,利用系統資源實現功能。

MySQL安裝報錯的解決方法是:1.仔細檢查系統環境,確保滿足MySQL的依賴庫要求,不同操作系統和版本需求不同;2.認真閱讀報錯信息,根據提示(例如缺少庫文件或權限不足)採取對應措施,例如安裝依賴或使用sudo命令;3.必要時,可嘗試源碼安裝並仔細檢查編譯日誌,但這需要一定的Linux知識和經驗。最終解決問題的關鍵在於仔細檢查系統環境和報錯信息,並參考官方文檔。

MySQL啟動失敗的原因有多種,可以通過檢查錯誤日誌進行診斷。常見原因包括端口衝突(檢查端口占用情況並修改配置)、權限問題(檢查服務運行用戶權限)、配置文件錯誤(檢查參數設置)、數據目錄損壞(恢復數據或重建表空間)、InnoDB表空間問題(檢查ibdata1文件)、插件加載失敗(檢查錯誤日誌)。解決問題時應根據錯誤日誌進行分析,找到問題的根源,並養成定期備份數據的習慣,以預防和解決問題。

MySQL無法直接在Android上運行,但可以通過以下方法間接實現:使用輕量級數據庫SQLite,由Android系統自帶,無需單獨服務器,資源佔用小,非常適合移動設備應用。遠程連接MySQL服務器,通過網絡連接到遠程服務器上的MySQL數據庫進行數據讀寫,但存在網絡依賴性強、安全性問題和服務器成本等缺點。

MySQL安裝失敗的原因主要有:1.權限問題,需以管理員身份運行或使用sudo命令;2.依賴項缺失,需安裝相關開發包;3.端口衝突,需關閉佔用3306端口的程序或修改配置文件;4.安裝包損壞,需重新下載並驗證完整性;5.環境變量配置錯誤,需根據操作系統正確配置環境變量。解決這些問題,仔細檢查每個步驟,就能順利安裝MySQL。

無法從終端訪問 MySQL 可能是由於:MySQL 服務未運行;連接命令錯誤;權限不足;防火牆阻止連接;MySQL 配置文件錯誤。
