linux socket通讯获取本地的源端口号的实现方法

关于tcp ip网络通讯的资料非常多,tcp ip通过ip数据包模式进行端对端通讯。典型的tcp数据包如下

linux socket通讯获取本地的源端口号的实现方法

可以看到数据包包含了源端口号和目的端口号,客户端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通讯获取本地的源端口号的实现方法,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对网站的支持!

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

原创文章,作者:UIFVF,如若转载,请注明出处:http://www.wangzhanshi.com/n/9123.html

(0)
UIFVF的头像UIFVF
上一篇 2025年1月1日 16:38:47
下一篇 2025年1月1日 16:38:49

相关推荐

  • 手把手教你使用 Gpio 子系统 API

    本文讲解 pinctrl 子系统和 gpio 子系统的 API,以及使用示例。 传统的配置 pin 的方式就是直接操作相应的寄存器,但是这种配置方式比较繁琐、而且容易出问题(比如 …

    2025年1月1日
  • Linux中hostname修改方法详解

    Linux操作系统的hostname是一个kernel变量,可以使用如下两种方式查看 ? # hostname 和 ? # cat /proc/sys/kernel/hostnam…

    Linux 2025年1月1日
  • linux系统下定时执行php脚本的方法

    前言 在windows服务端定时执行某个程序代码还是比较简单的,只要在系统中做定时任务就可以了,目前刚刚接触到linux下的定时任务,这在好多公司的面试中都会提到的,而对于linu…

    Linux 2025年1月1日
  • linux中install命令和cp命令的使用与区别

    前言 install和cp类似,都可以将文件/目录拷贝到指定的地点。但是,install允许你控制目标文件的属性。install通常用于程序的makefile(在RPM的spec里…

    Linux 2025年1月1日
  • linux更改目录显示颜色实例详解

    linux更改目录显示颜色实例详解 用shell列举目录的时候,文件夹都是蓝色的,背景是黑色,使得无法看清蓝色的文件名称,看起来很痛苦。这个已经好几次遇到这个问题了都没有把解决方法…

    Linux 2025年1月1日
  • 解决linux下zip文件解压乱码问题

    原因 由于zip格式并没有指定编码格式,Windows下生成的zip文件中的编码是GBK/GB2312等,因此,导致这些zip文件在Linux下解压时出现乱码问题,因为Linux下…

    Linux 2025年1月1日
  • Linux服务器下安装配置Nginx的教程

    nginx(“engine x”)是一款是由俄罗斯的程序设计师igor sysoev所开发高性能的 web和 反向代理 服务器,也是一个 imap/pop3…

    2025年1月1日
  • linux sudo命令详解

    简单的说,sudo 是一种权限管理机制,管理员可以授权于一些普通用户去执行一些 root 执行的操作,而不需要知道 root 的密码。 严谨些说,sudo 允许一个已授权用户以超级…

    2025年1月1日
  • linux vps服务器常用服务iptables策略

    vps服务器裸奔在公网上,总感觉有点不安全,没办法总得整点措施来加固下服务器呀,安全第一。linux系统自带防火墙必须要好好利用起来,可是我有一年多没写过防火墙策略了,该忘的都忘了…

    Linux 2025年1月1日
  • linux中定时任务crond命令使用方法

    什么是crond? crond是linux用来定期执行命令或指定程序任务的一种服务。安装完操作系统后,默认会启动crond任务调度服务。crond服务会定期检查系统中是否有要执行的…

    2025年1月1日

发表回复

登录后才能评论