TCP编程之网卡信息获取和域名解析
1.TCP/IP简介
TCP/IP协议源于1969年,是针对Internet开发的一种体系结构和协议标准,目的在于解决异种计算机网络的通信问题。使得网络在互联时能为用户提供一种通用、一致的通信服务。是Internet采用的协议标准。
TCP/IP是一组通信协议的代名词,是由一系列协议组成的协议簇。它本身指两个协议集:
TCP(传输控制协议)。
IP(互联网络协议)。
TCP/IP协议的基本传输单位是数据包(Datagram)。
TCP/IP传输协议,即传输控制/网络协议,也叫作网络通讯协议。它是在网络的使用中的最基本的通信协议。TCP/IP传输协议对互联网中各部分进行通信的标准和方法进行了规定。并且,TCP/IP传输协议是保证网络数据信息及时、完整传输的两个重要的协议。TCP/IP传输协议是严格来说是一个四层的体系结构,应用层、传输层、网络层和数据链路层都包含其中。
2.获取本地网卡信息
获取本地网卡信息的方法有很多种,在Linux命令行下我们可以通过ifconfig命令查询网卡信息。
2.1 通过ioctl函数获取网卡信息
ioctl(input/output control)是一个专用于设备输入输出操作的系统调用,该调用传入一个跟设备有关的请求码,系统调用的功能完全取决于请求码。
- ioctl函数原型:
#include <sys/ioctl.h>
int ioctl(int fd, unsigned long request, ...);
形参:fd --文件描述符
request --请求命令
第三个参数为可变参数,该参数填写取决于request 命令;
返回值:一般成功返回0
- 网络相关的请求划分为6 类:
- 套接口操作
- 文件操作
- 接口操作
- ARP 高速缓存操作
- 路由表操作
- 流系统
下表列出了网络相关ioctl 请求的request 参数以及arg 地址必须指向的数据类型:
- 获取网卡信息示例:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/ioctl.h>
#include <net/if.h>
#include <arpa/inet.h>
#include <sys/socket.h>
#include <netinet/in.h>
int main()
{
int i=0;
int sockfd;
/*创建网络套接字*/
if((sockfd = socket(AF_INET, SOCK_STREAM,0))<0)
{
printf("socket error\n");
return -1;
}
struct ifreq ifcu_req[4];//网卡信息结构体
struct ifconf ifc=
{
.ifc_len=sizeof(ifcu_req),//缓冲区大小
.ifc_req=ifcu_req,
};
struct ifreq b_addr;
int ret=ioctl(sockfd,SIOCGIFCONF, &ifc);//获取网络接口信息
struct sockaddr boardaddr;
struct sockaddr_in addr;
printf("ret=%d\n",ret);
char mac[6];
int j=0;
int count=ifc.ifc_len/sizeof(struct ifreq);//计算获取到的网卡信息个数
for(i=0;i<count;i++)
{
printf("\n------------%s---------------------\n",ifcu_req[i].ifr_name);
memcpy(&addr,&ifcu_req[i].ifr_addr,sizeof(addr));
printf("ip地址: %s \n",inet_ntoa(addr.sin_addr));
/*广播地址*/
if (ioctl(sockfd, SIOCGIFBRDADDR, &ifcu_req[i]) == 0)
{
memcpy(&addr,&ifcu_req[i].ifr_broadaddr, sizeof(addr));
printf("广播地址: %s \n",inet_ntoa(addr.sin_addr));
}
/*子网掩码*/
if (ioctl(sockfd, SIOCGIFNETMASK, &ifcu_req[i]) == 0)
{
memcpy(&addr,&ifcu_req[i].ifr_broadaddr, sizeof(addr));
printf("子网掩码: %s \n",inet_ntoa(addr.sin_addr));
}
/*MAC地址*/
if (ioctl(sockfd, SIOCGIFHWADDR, &ifcu_req[i]) == 0)
{
memcpy(mac,&ifcu_req[i].ifr_hwaddr.sa_data[0],6);
printf("mac地址:");
for(j=0;j<5;j++)
{
printf("%02X-",mac[j]);
}
printf("%02X\n",mac[j]);
printf("------------------------------------\n");
}
}
}
3.域名解析
域名(英语:Domain Name),又称网域,是由一串用点分隔的名字组成的Internet上某一台计算机或计算机组的名称,用于在数据传输时对计算机的定位标识(有时也指地理位置)。
由于IP地址具有不方便记忆并且不能显示地址组织的名称和性质等缺点,人们设计出了域名,并通过网域名称系统(DNS,Domain Name System)来将域名和IP地址相互映射,使人更方便地访问互联网,而不用去记住能够被机器直接读取的IP地址数串。
- gethostbyname函数接口
#include <netdb.h>
extern int h_errno;
struct hostent *gethostbyname(const char *name);
函数功能:域名解析
形参:name --域名
返回值:
struct hostent {
char *h_name; /* 主机的正式名称 */
char **h_aliases; /* 别名列表*/
int h_addrtype; /* 主机地址类型*/
int h_length; /* 地址长度*/
char **h_addr_list; /*地址列表 */
}
void herror(const char *s); //错误输出
const char *hstrerror(int err);//根据错误编号返回错误消息字符串
- 示例
#include <stdio.h>
#include <netdb.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <string.h>
int main(int argc,char *argv[])
{
if(argc!=2)
{
printf("格式:./app <域名>\n");
return 0;
}
//域名解析
struct hostent *host_info=gethostbyname(argv[1]);
if(host_info==NULL)
{
printf("err=%d,%s\n",h_errno,hstrerror(h_errno));
return 0;
}
int i=0;
struct in_addr addr;
printf("主机名:%s\n",host_info->h_name);
for(i=0;host_info->h_aliases[i]!=0;i++)
{
printf("主机别名:%s\n",host_info->h_aliases[i]);
}
for(i=0;host_info->h_addr_list[i]!=0;i++)
{
memcpy(&addr,host_info->h_addr_list[i],sizeof(addr));
printf("ip地址:%s\n",inet_ntoa(addr));
}
}
获取邦定ip地址信息getsockname
getsockname()函数用于获取一个套接字的名字。它用于一个已捆绑或已连接套接字,本地地址将被返回。本调用特别适用于如下情况:未调用bind()就调用了connect(),这时唯有getsockname()调用可以获知系统内定的本地地址。在返回时,namelen参数包含了名字的实际字节数。
#include <sys/socket.h>
int getsockname(int sockfd, struct sockaddr *addr, socklen_t *addrlen);
函数功能:获取sockfd绑定的ip地址和端口号信息
形 参:sockfd --网络套接字
addr --保存返回的ip地址和端口号信息
addrlen --addr缓冲区大小(必须由用户设置大小,函数调用成功返回实际addr大小)
返回值:成功返回0,失败-1
- 示例:
#include <stdio.h>
#include <sys/types.h> /* See NOTES */
#include <sys/socket.h>
#include <stdlib.h>
#include <unistd.h>
#include <netinet/in.h>
#include <netinet/ip.h> /* superset of previous */
#include <arpa/inet.h>
int main(int argc,char *argv[])
{
if(argc!=3)
{
printf("格式:./app <服务器端口号> <服务器IP>\n");
return 0;
}
/*创建网络套接字*/
int sockfd=socket(AF_INET, SOCK_STREAM,0);
if(sockfd==-1)
{
printf("创建网络套接字失败\n");
return 0;
}
/*连接服务器*/
struct sockaddr_in addr=
{
.sin_family=AF_INET,//IPV4
.sin_port=htons(atoi(argv[1])),//服务器端口号
.sin_addr=
{
.s_addr=inet_addr(argv[2]),//服务器IP
}
};
if(connect(sockfd,(struct sockaddr *)&addr,sizeof(addr)))
{
printf("连接服务器失败\n");
return 0;
}
printf("服务器连接成功\n");
struct sockaddr_in c_addr;
socklen_t addrlen=sizeof(c_addr);
getsockname(sockfd,(struct sockaddr *)&c_addr, &addrlen);
printf("本地信息%s:%d\n",inet_ntoa(c_addr.sin_addr),ntohs(c_addr.sin_port));
while(1)
{
}
}