域名系统 DNS

news2024/10/6 14:29:12

DNS 概述

        域名系统 DNS(Domain Name System)是因特网使用的命名系统,用来把便于人们使用的机器名字转换成为 IP 地址。域名系统其实就是名字系统。为什么不叫“名字”而叫“域名”呢?这是因为在这种因特网的命名系统中使用了许多的“域(domain)”,因此就出现了“域名”这个名词。“域名系统”明确地指明这种系统是应用在因特网中。我们都知道,IP 地址是由 32 位的二进制数字组成的。用户与因特网上某台主机通信时,显然不愿意使用很难记忆的长达 32 位的二进制主机地址。即使是点分十进制 IP 地址也并不太容易记忆。相反,大家愿意使用比较容易记忆的主机名字。但是,机器在处理 IP 数据报时,并不是使用域名而是使用 IP 地址。 这是因为 IP 地址长度固定,而域名的长度不固定,机器处理起来比较困难。
        例如,www.baidu.com 就是一个域名,那么域名解析的过程就是:
www.baidu.com-->DNS 服务器(把域名转成 IP)-->转换为 IP(http://14.215.177.39/)

DNS 协议运行在 UDP 协议上面,是一个 UDP 的“回显”程序,使用 53 号端口

因特网的域名结构

从语法上讲,每一个域名都是有标号(label)序列组成,而各标号之间用点(小数点)隔开。如下例子所示:

        上图这是中央电视台用于手法电子邮件的计算机的域名,它由三个标号组成,其中标号com 是顶级域名,标号 cctv 是二级域名,标号 mail 是三级域名。
        DNS 规定,域名中的标号都有英文和数字组成, 每一个标号不超过 63 个字符 ( 为了记忆方 便,一般不会超过 12 个字符 ) ,也不区分大小写字母。
        级别最低的域名写在最左边,而级别最高的字符写在最右边。由多个标号组成的完整域名总共不超过 255 个字符。
        DNS 既不规定一个域名需要包含多少个下级域名,也不规定每一级域名代表什么意思。各级域名由其上一级的域名管理机构管理,而最高的顶级域名则由 ICANN 进行管理。用这种方法可使每一个域名在整个互联网范围内是唯一的,并且也容易设计出一种查找域名的机制。

DNS 协议

首部格式

        DNS 请求与响应的格式是一致的,其头部分为 Header Question Answer Authority
Additional5 部分,如下图所示:

Header 部分是一定有的,长度固定为 12 个字节;其余 4 部分可能有也可能没有,并且长度也不一定,这个在 Header 部分中有指明。 Header 的结构如下:

下面说明一下各个字段的含义 :
1. 标识符:占 16
        ID:占 16 位。该值由发出 DNS 请求的程序生成,DNS 服务器在响应时会使用该 ID,这样便于请求程序区分不同的 DNS 响应。
2. 标志:占 16
        QR:占 1 位。指示该消息是请求还是响应。0 表示请求;1 表示响应。
        OPCODE:占 4 位。指示请求的类型,有请求发起者设定,响应消息中复用该值。0 表示标准查询;1 表示反转查询;2 表示服务器状态查询。3~15 目前保留,以备将来使用。
        AA(Authoritative Answer,权威应答):占 1 位。表示响应的服务器是否是权威DNS 服务器。只在响应消息中有效。
        TC(TrunCation,截断):占 1 位。指示消息是否因为传输大小限制而被截断。
        RD(Recursion Desired,期望递归):占 1 位。该值在请求消息中被设置,响应消息复用该值。如果被设置,表示希望服务器递归查询。但服务器不一定支持递归查询。
        RA(Recursion Available,递归可用性):占 1 位。该值在响应消息中被设置或被清除,以表明服务器是否支持递归查询。
        Z:占 3 位。保留备用。
        RCODE(Response code):占 4 位。该值在响应消息中被设置。取值及含义如下:
                0:No error condition,没有错误条件;
                1:Format error,请求格式有误,服务器无法解析请求;
                2:Server failure,服务器出错。
                3:Name Error,只在权威 DNS 服务器的响应中有意义,表示请求中的域名不存在。
                4:Not Implemented,服务器不支持该请求类型。
                5:Refused,服务器拒绝执行请求操作。
                6~15:保留备用。QDCOUNT:占 16 位(无符号)。指明 Question 部分的包含的实体数量。
        ANCOUNT:占 16 位(无符号)。指明 Answer 部分的包含的 RR(Resource Record)数 量。
        NSCOUNT:占 16 位(无符号)。指明 Authority 部分的包含的 RR(Resource Record)数量。
        ARCOUNT:占 16 位(无符号)。指明 Additional 部分的包含的 RR(Resource Record)数量。

数据区域

查询名:
        查询名部分长度不定,一般为要查询的域名(也会有 IP 的时候,即反向查询)。 此部分由一个或者多个标示符序列组成,每个标示符以首字节数的计数值来说明该标示符长度,每个名字以 0 结束。计数字节数必须是 0~63 之间。该字段无需填充字节。还是借个例子来说明更直观些,查询名为 http://gemini.tuc.noao.edu 的话,查询名字段如下
查询类型:

查询类:
        一般为 IN(枚举值 1) ,即 Internet 数据

DNS 客户端程序实例

#include <sys/types.h> /* See NOTES */
#include <sys/socket.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <linux/socket.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <stdlib.h>
#include <sys/types.h>
#include <dirent.h>
#include <unistd.h>

int sock;
struct sockaddr_in sa;

/*
generate_question:解析域名数据
*/
void generate_question(char*dns_name,char*buf,int*len)
{
    char*pos = dns_name;//指向域名当前位置
    char*ptr = buf;
    int n = 0;
    *len = 0;
    while(1)
    {
            n = strlen(pos) - (strstr(pos , ".") ? strlen(strstr(pos , ".")) : 0);//在 pos 中查找"."子串
        //返回"."在 pos 中第一次出现的地址,若没有找到返回 NULL
        printf("%d\n", n);
        *ptr++ = (unsigned char)n;
        memcpy(ptr,pos,n);
        printf("%lu\n",strlen(ptr));
        printf("%s\n", ptr);
        *len += n + 1;
        ptr += n;
        if(!strstr(pos , "."))
        {
            *ptr = (unsigned char)0;
            ptr ++;
            *len += 1;
            break;
        }
        pos += n + 1;
        printf("%s\n", pos);
    }
}

/*
send_dns_request:解析域名数据,并发送数据包
*/
void send_dns_request(char * dns_name)
{
    unsigned char request[256] = {0}; //保存整个请求报文
    unsigned char *ptr = request;
    unsigned char question[128];//存储域名解析数据
    int question_len;
    //产生请求(把字符串表示的域名转换成 DNS 要求的格式)
    generate_question(dns_name , question ,&question_len);
    printf("%s\n", question);
    printf("%d\n", question_len);
    *((unsigned short*)ptr) = 1; //会话标识 ID
    ptr += 2;
    *((unsigned short*)ptr) = htons(0x0100); //flags
    ptr += 2;
    *((unsigned short*)ptr) = htons(1); //Quetions 问题数,通常为 1
    ptr += 2;
    *((unsigned short*)ptr) = 0;
    ptr += 2;
    *((unsigned short*)ptr) = 0;
    ptr += 2;
    *((unsigned short*)ptr) = 0;
    ptr += 2;
    //把域名(www.baidu.com)装换为相应的问题格式保存在此处,以\0 结尾
    memcpy(ptr , question , question_len);
    ptr += question_len;
    *((unsigned short*)ptr) = htons(1); //获取 IPV4 地址,查询类型
    ptr += 2;
    *((unsigned short*)ptr) = htons(1); //指互联网地址,查询类,一般为 1,表明是 Internet 数据
    int re = sendto(sock, request,question_len+16 ,0, (structsockaddr*)&sa, sizeof(sa));
    printf("re = %d\n",re);
    int i;
    for(i=0;i<re;i++)//以 16 进制格式打印出来
    {
        printf("%02X ",(unsigned char)request[i]);
    } 
    printf("===================\n");
}

/*
    recv_dns_response:获取域名所对应的 IP 地址,并打印
*/
void recv_dns_response()
{
    struct sockaddr_in src_addr;
    socklen_t addrlen = sizeof(src_addr);
    char buf[255] = {0};
    int r = recvfrom(sock, buf, 255, 0, (struct sockaddr*)&src_addr,&addrlen);
    printf("r == %d\n", r);
    if(r > 0)//收到大于 0 的数据
    {
        int i;
        for(i=r-4;i<r;i++)//以 16 进制格式打印出来
        {
            if(i != r-1)
                printf("%d.",(unsigned char)buf[i]);
            else
                printf("%d\n",(unsigned char)buf[i]);
        } 
    }
}

//例./main www.baidu.com
int main(int argc, char *argv[])
{
    if(argc != 2)
    {
        printf("Usage : %s <domain name>\n",argv[0]);
        return -1;
    }
    //step 1: 创建一个套接字
    sock = socket(AF_INET, SOCK_DGRAM, 0);
    if (sock == -1)
    {
        perror("socket error:");
        return -1;
    }

    //step 2: 绑定一个地址(ip+端口号)
    memset(&sa, 0,sizeof(sa));
    sa.sin_family = AF_INET;
    sa.sin_port = htons(53); //按"网络字节序"来保存一个整数
    sa.sin_addr.s_addr = inet_addr("114.114.114.114");
    //发送解析请求
    send_dns_request(argv[1]);
    //接收分析结果
    recv_dns_response();
    close(sock);
    return 0;
}

可以看到当按正确的报文形式发送给 DNS 服务器后,接收到的报文后面四个字节存储了解析的 IP 地址

域名解析函数(gethostbyname)

【头文件】
#include <netdb.h>
#include <sys/socket.h>
extern int h_errno;
【函数原型】
struct hostent *gethostbyname(const char *name);
【函数功能】
        使用域名或主机名获取地址
【参数含义】
        [name]: 待解析的域名或主机名
【返回值】
        失败返回 NULL 指针
        成功返回的非空指针指向如下的 hostent 结构体指针
struct hostent
{
    char *h_name; /* 主机正式名称 */
    char **h_aliases; /* 别名列表。 */
    int h_addrtype; /* 主机地址类型。*/
    int h_length; /* 地址的长度。 */
    char **h_addr_list; /*来自名称服务器的地址列表。 */
    #ifdef __USE_MISC
    # define h_addr h_addr_list[0] /* 地址,用于向后兼容。*/
    #endif
};
【示例】
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netdb.h>
#include <stdio.h>
extern int h_errno;

int main(int argc, char **argv)
{
    if (argc != 2) {
        printf("Use example: %s www.google.com\n", *argv);
        return -1;
    }
    char *name = argv[1];
    struct hostent *hptr;
    hptr = gethostbyname(name);
    if (hptr == NULL) {
        printf("gethostbyname error for host: %s: %s\n", name, 
        hstrerror(h_errno));
        return -1;
    }
    //输出主机的规范名
    printf("\tofficial: %s\n", hptr->h_name);
    //输出主机的别名
    char **pptr;
    char str[INET_ADDRSTRLEN];
    for (pptr=hptr->h_aliases; *pptr!=NULL; pptr++) {
        printf("\ttalias: %s\n", *pptr);
    }
    //输出 ip 地址
    switch (hptr->h_addrtype) {
        case AF_INET:
            pptr = hptr->h_addr_list;
            for (; *pptr!=NULL; pptr++) {
                printf("\taddress: %s\n",inet_ntop(hptr->h_addrtype, hptr->h_addr, str, sizeof(str)));
        }
            break;
        default:
            printf("unknown address type\n");
            break;
    }
    return 0;
}

获取本地主机名(gethostname)

【头文件】
#include <unistd.h>
【函数原型】
int gethostname(char *name, size_t len);
【函数功能】
        获取本地主机名
【参数含义】
        [name]: 保存获取的主机名
        [len]: naem 的最大长度
【返回值】
        成功返回 0,失败返回-1
【示例】
char buf[256] = {0};
int t = gethostname(buf,256);
if(t == 0)
{
    printf("%s\n",buf);
}

设置本地主机名(sethostname)

【头文件】
#include <unistd.h>
【函数原型】
int sethostname(const char *name, size_t len);
【函数功能】
        设置本地主机名
【参数含义】
        [name]: 设置的主机名
        [len]: naem 的最大长度
【返回值】
        成功返回 0,失败返回-1
【示例】
sethostname ( "jiuyue" , 6 );

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1139108.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【数字IC验证半科班历程:芯片概括_2023.10.20】

前言 选择行业需深入了解&#xff1a;行业的前景&#xff0c;是否适合……在知乎浏览n多帖子&#xff0c;千人千面&#xff0c;褒贬不一&#xff0c;只能黑人问号脸。且帖子虽说明学习路线&#xff0c;但甚至面对缩写名词&#xff0c;百度后仍是一知半解&#xff0c;不知确切内…

el-form组件如何清除校验提示(前端技能提升)

错误效果 错误描述 在切换radio切换的时候校验提示提示出来了&#xff0c;本身不应该出来但是如何取消呢&#xff1f;因为在切换时候我们置空但是并没有取消校验&#xff0c;所以从通过到拒绝置空时候肯定会出现提示语&#xff0c;那么我们把提示校验的方法去掉就行了。 错误代…

启动两个线程,用另一个线程以通知的终止另一个线程

要求&#xff1a; 在main方法中启动两个线程第一个线程循环随机打印100以内的整数直到第二个线程从键盘读取了”Q“命令 实现代码如下&#xff1a; public class Homework01 {public static void main(String[] args) {THk tHk new THk();tHk.start();Thk2 thk2 new Thk2(…

【Linux】操作系统以及虚拟机的安装与配置

&#x1f973;&#x1f973;Welcome Huihuis Code World ! !&#x1f973;&#x1f973; 接下来看看由辉辉所写的关于Linux的相关操作吧 目录 &#x1f973;&#x1f973;Welcome Huihuis Code World ! !&#x1f973;&#x1f973; 一.操作系统的介绍 二.VMWare虚拟机的安装…

【算法题】割后面积最大的蛋糕

题目&#xff1a; 矩形蛋糕的高度为 h 且宽度为 w&#xff0c;给你两个整数数组 horizontalCuts 和 verticalCuts&#xff0c;其中&#xff1a; horizontalCuts[i] 是从矩形蛋糕顶部到第 i 个水平切口的距离 verticalCuts[j] 是从矩形蛋糕的左侧到第 j 个竖直切口的距离 请你…

宏定义实现offsetof

在C语言中&#xff0c;有这样一个特殊的宏&#xff0c;叫offsetof&#xff0c;它的功能是啥呢&#xff1f; 我们来看看它的介绍 它的功能是&#xff1a;返回一个结构体的成员的大小&#xff08;相较于起始地址的偏移量&#xff09; 引用代码&#xff1a;http://t.csdnimg.cn…

CS224W1.1——图机器学习介绍

文章目录 1. 介绍2. 主要问题3. 深度学习如何应用在图结构中4. 课程大纲 学习一下斯坦福CS224W的图机器学习&#xff08;2021年&#xff09;&#xff0c;并做一下学习笔记&#xff0c;主要是研究方向与图神经网络相关。这次是第一次笔记&#xff0c;图片很多都是从斯坦福的PPT里…

【java学习—九】抽象类和抽象方法(3)

文章目录 1. 相关概念2. 如何定义抽象方法和抽象类3. 抽象类的作用4. 练习题4.1. 问题14.2. 问题2 1. 相关概念 抽象类&#xff1a;随着继承层次中一个个新子类的定义&#xff0c;类变得越来越具体&#xff0c;而父类则更一般&#xff0c;更通用。类的设计应该保证父类和子类能…

如何在群晖Synology+Office实现多人编辑一个文件?

使用群晖Synology Office提升生产力&#xff1a;多人同时编辑一个文件 文章目录 使用群晖Synology Office提升生产力&#xff1a;多人同时编辑一个文件本教程解决的问题是&#xff1a;1. 本地环境配置2. 制作本地分享链接3. 制作公网访问链接4. 公网ip地址访问您的分享相册5. 制…

潮玩宇宙:大逃杀模式的利与弊

在当今的游戏市场中&#xff0c;潮玩宇宙的概念越来越受到关注。这个以潮流玩具为主题的宇宙&#xff0c;不仅包含了众多类型的玩具和收藏品&#xff0c;还融合了各种独特的文化形式和娱乐内容。其中&#xff0c;大逃杀模式作为一种流行的游戏模式&#xff0c;在潮玩宇宙中占据…

倒计时 1 天!神策 2023 数据驱动大会「参会指南」,请收藏

「新旅程、新经营&#xff0c;决胜数字化」&#xff0c;神策 2023 数据驱动大会明日召开&#xff01;大会聚焦客户经营与客户旅程主题&#xff0c;特邀数十位海内外专家、知名企业代表、资深投资人&#xff0c;分享行业观点与趋势洞察&#xff0c;共同推动中国企业数字化客户经…

Mysql8.1.0 windows 绿色版安装

Mysql8.1.0 windows 绿色版安装 目录 Mysql8.1.0 windows 绿色版安装1、下载mysql8.1.0_windows&#xff08;mysql-8.1.0-winx64.zip&#xff09;2、解压到安装目录3、添加环境变量4、新建mysql配置文件5、安装mysql服务6、初始化数据文件7、启动mysql服务8、进入mysql管理模式…

深度学习:张量 介绍

张量[1]是向量和矩阵到 n 维的推广。了解它们如何相互作用是机器学习的基础。 简介 虽然张量看起来是复杂的对象&#xff0c;但它们可以理解为向量和矩阵的集合。理解向量和矩阵对于理解张量至关重要。 向量是元素的一维列表&#xff1a; 矩阵是向量的二维列表&#xff1a; 下标…

JAVA面试题简单整理

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、重载和重写的区别一、&和&&的区别一、get和post请求的区别 delete、put一、cookie和session的区别一、Autowired和Resource区别一、”和equals…

wait 和 notify 这个为什么要在 synchronized 代码块中?

wait 和 notify wait 和 notify 用来实现多线程之间的协调&#xff0c;wait 表示让线程进入到阻塞状态&#xff0c;notify 表示让阻塞的线程唤醒。 wait 和 notify 必然是成对出现的&#xff0c;如果一个线程被 wait()方法阻塞&#xff0c;那么必然需要另外一个线程通过 noti…

C语言_文件_进程_进程间通讯 常用函数/命令 + 实例

文件相关命令&#xff1a; ps -aux|grep init? //搜索包含init名称的进程 top //linux下的资源管理器&#xff08;动态&#xff09;//open 返回的int 是给后面的读/写/光标移动 用的fd&#xff0c;没有open就不能进行后面的操作&#xff1b; int op…

大数据Flink(一百零三):SQL 表值聚合函数(Table Aggregate Function)

文章目录 SQL 表值聚合函数(Table Aggregate Function) SQL 表值聚合函数(Table Aggregate Function) Python UDTAF,即 Python TableAggregateFunction。Python UDTAF 用来针对一组数据进行聚合运算,比如同一个 window 下的多条数据、或者同一个 key 下的多条数据等,与…

【micro ros】快速上手:在 RT-Thread上运行 micro ros

文章目录 快速上手micro ros && RT-Thread&#xff08;serial和udp方式&#xff09;1.背景介绍2.工程准备工作2.1 克隆 RT-Thread主仓2.2 克隆 env-windows 3.编译准备工作3.1 python & cmake安装3.2 scons工具安装3.3 GNU make安装3.4 Fastgithub安装 4.工程配置4…

[SQL开发笔记]LIKE操作符:在 WHERE 子句中搜索列中的指定模式

一、功能描述&#xff1a; LIKE操作符&#xff1a;用于在 WHERE 子句中搜索列中的指定模式。 二、LIKE操作符语法详解&#xff1a; LIKE 语法 SELECT column1, column2,…FROM table_nameWHERE column LIKE pattern; 参数说明&#xff1a; &#xff08;1&#xff09;colum…

全球生物气候产品2.5m和30s分辨率

简介 生物气候是指生物和气候相互作用的结果&#xff0c;包括植物和动物对气候的影响&#xff0c;以及气候对生物的影响。生物气候研究的是生物、气候、土地和水等自然要素之间相互作用的过程&#xff0c;旨在探讨它们是如何互动并导致生态系统的变化的。生物气候对于理解全球…