激情久久久_欧美视频区_成人av免费_不卡视频一二三区_欧美精品在欧美一区二区少妇_欧美一区二区三区的

服務器之家:專注于服務器技術及軟件下載分享
分類導航

Linux|Centos|Ubuntu|系統進程|Fedora|注冊表|Bios|Solaris|Windows7|Windows10|Windows11|windows server|

服務器之家 - 服務器系統 - Linux - linux socket通訊獲取本地的源端口號的實現方法

linux socket通訊獲取本地的源端口號的實現方法

2022-02-17 20:18二流小寶 Linux

這篇文章主要介紹了linux socket通訊獲取本地的源端口號的相關資料,需要的朋友可以參考下

關于tcp ip網絡通訊的資料非常多,tcp ip通過ip數據包模式進行端對端通訊。典型的tcp數據包如下

linux socket通訊獲取本地的源端口號的實現方法

可以看到數據包包含了源端口號和目的端口號,客戶端socket向服務端發起連接時,系統會給socket隨機分配一個源端口號,我們可以通過getsocketname來獲取連接成功的socket的原端口信息。

函數原型

?
1
2
#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

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
#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: 編譯及運行

?
1
2
3
4
$ g++ source_port.cpp
$ ./a.out www.baidu.com
connect to www.baidu.com success.
local port: 39702

終端2: 通過tcpdump抓包驗證

?
1
2
3
4
5
6
7
$ 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通訊獲取本地的源端口號的實現方法,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對服務器之家網站的支持!

原文鏈接:http://blog.csdn.net/sweettool/article/details/78078750

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 亚洲国产精品99 | 斗罗破苍穹在线观看免费完整观看 | 欧美乱论 | 一区二区三区在线视频观看58 | 一级国产免费 | 国产一区二区高清在线 | 久久国产精品久久精品国产演员表 | 一区二区三区欧美日韩 | 操嫩草 | 国产精品久久久久永久免费 | 国产免费传媒av片在线 | 久久最新网址 | 日韩不卡一区二区 | 激情小说区 | 在线观看国产 | 色.com| 午夜精品久久久久久久99热浪潮 | 九色p| 一区二区免费网站 | 久久成人午夜视频 | 午夜国内精品a一区二区桃色 | 国产精品久久久久久久久久 | a一级黄色大片 | 欧美不卡在线 | 羞羞答答tv | 国产91九色视频 | 精品久久久一二三区播放播放播放视频 | 国产va在线观看免费 | 羞羞视频在线免费 | 久久精品中文字幕一区 | 麻豆一二区 | 国产亚洲精品久久久久久久久久 | 成人短视频在线播放 | 毛片在线免费观看视频 | 久久久久久久九九九九 | 国产成人aⅴ | 黄污视频在线看 | 国产98色| 亚洲午夜免费电影 | 草碰人人 | 精品成人免费一区二区在线播放 |