程序笔记   发布时间:2022-05-30  发布网站:大佬教程  code.js-code.com
大佬教程收集整理的这篇文章主要介绍了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,sizeof(buf)); 
    snprintf(buf,sizeof(buf),"new socket Failed. errno: %d,error: %s",errno,strerror(errno)); 
    perror(buf); 
    exit(-1); 
  } 
  memset(&serv_addr,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,"inet_pton Failed. errno: %d,strerror(errno)); 
    perror(buf); 
    exit(-1); 
  } 
  if (-1 == connect(sockfd,(struct sockaddr *)&serv_addr,sizeof(serv_addr))) {// 连接socket 
    memset(buf,"connect socket Failed. errno: %d,strerror(errno)); 
    perror(buf); 
    exit(-1); 
  } 
  printf("connect to %s success.\n",host); 
  len = sizeof(sizeof(loc_addr)); 
  memset(&loc_addr,len); 
  if (-1 == getsockname(sockfd,(struct sockaddr *)&loc_addr,&len)) {// 获取socket绑定的本地address信息 
    memset(buf,"get socket name Failed. errno: %d,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,length 60)
220.181.111.188.http > icentos.39702: Flags [s.],cksum 0xc315 (correct),seq 3561856904,ack 2381397555,win 8192,options [mss 1424,wscale 5],length 0
18:38:32.425930 IP (tos 0x0,ID 35034,length 40)

对比终端一和终端二表明获取的源端口地址是正确的。

总结

以上所述是小编给大家介绍的linux socket通讯获取本地的源端口号的实现方法,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对编程小技巧网站的支持!

大佬总结

以上是大佬教程为你收集整理的linux socket通讯获取本地的源端口号的实现方法全部内容,希望文章能够帮你解决linux socket通讯获取本地的源端口号的实现方法所遇到的程序开发问题。

如果觉得大佬教程网站内容还不错,欢迎将大佬教程推荐给程序员好友。

本图文内容来源于网友网络收集整理提供,作为学习参考使用,版权属于原作者。
如您有任何意见或建议可联系处理。小编QQ:384754419,请注明来意。