服务器之家:专注于服务器技术及软件下载分享
分类导航

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

延伸 · 阅读

精彩推荐
  • LinuxLinux桌面环境不支持HiDPI显示如何进行设置让其支持

    Linux桌面环境不支持HiDPI显示如何进行设置让其支持

    Linux不能很好的支持HiDPI设备,所以需要将Linux桌面环境进行设置,让Linux系统能够支持HiDPI显示,下面一起来了解下支持HiDPI显示的Linux桌面环境设置 ...

    系统之家3312019-10-24
  • Linuxlinux常用命令的经典使用技巧

    linux常用命令的经典使用技巧

    这篇文章主要介绍linux常用命令的一些经典使用技巧,学习linux的朋友可以参考下 ...

    Linux教程网4942019-12-09
  • Linux有哪些 Linux 平板电脑可供选择

    有哪些 Linux 平板电脑可供选择

    有很多 基于 Linux 的笔记本电脑,也有 预装了 Linux 的迷你 PC,而市场上还有一些 基于 Linux 的智能手机。那平板电脑呢?有没有一些不错的基于 Linux 的平板...

    Linux中国10842021-10-29
  • LinuxLinux安装PHP MongoDB驱动

    Linux安装PHP MongoDB驱动

    在php中使用mongodb你必须使用 mongodb 的 php驱动。本文是小编在部署生产环境时候记录的笔记,特此分享到脚本之家平台供大家参考...

    TODOSomeOne8162021-11-22
  • LinuxAIX突然断电后hacmp无法启动的解决方法

    AIX突然断电后hacmp无法启动的解决方法

    AIX是一款类UNIX操作系统,当AIX突然断电后,会出现hacmp无法启动的情况,关于这个问题的解决方法如下,需要的朋友可以看看 ...

    系统之家2762019-10-15
  • Linuxlinux corosync+pacemaker+drbd+mysql配置安装详解

    linux corosync+pacemaker+drbd+mysql配置安装详解

    这篇文章主要介绍了corosync+pacemaker+drbd+mysql配置安装详解,需要的朋友可以参考下...

    Linux教程网9982021-11-04
  • Linux十个问题理解Linux epoll工作原理

    十个问题理解Linux epoll工作原理

    epoll 是 linux 特有的一个 I/O 事件通知机制。很久以来对 epoll 如何能够高效处理数以百万记的文件描述符很有兴趣。近期学习、研究了 epoll 源码,在这个过...

    腾讯技术工程6542021-06-03
  • Linux一行代码教你如何隐藏Linux进程

    一行代码教你如何隐藏Linux进程

    这篇文章主要介绍了一行代码教你如何隐藏Linux进程,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们...

    dog2503652020-08-21