互联网私有IP地址列表
最近因为业务需要,要判断用户的IP是否私有IP,
以前知道的私有IP,基本上只有如下几个(注意:这不是正确答案):
- 10.0.0.0/8(10.0.0.0-10.255.255.255)
- 172.16.0.0/12(172.16.0.0-172.31.255.255)
- 192.168.0.0/16(192.168.0.0-192.168.255.255)
- 再加一个本地IP:
127.0.0.0/8(127.0.0.0-127.255.255.255)
localhost - 以前考MCSE的时候,还知道一个Windows在DHCP未获取到IP时,会自动分配的:
169.254.0.0/16(169.254.0.0-169.254.255.255)
之前知道的就这些了,问了一下ChatGPT老师,说还有一些,如图:
基于以往印象中,GPT老师的不靠谱情况,去google查了一些资料,最后搜索到官方的定义文档,对互联网保留地址做了一个清单说明,还贴心的提供了csv格式下载:
https://www.iana.org/assignments/iana-ipv4-special-registry/iana-ipv4-special-registry.xhtml
这个网址是iana的官方网站,iana负责全球的IP地址分配,具体说明参考百度百科:
IANA ( Internet Assigned Numbers Authority,互联网号码分配机构)负责协调全球DNS的根域名、IP地址分配和其他互联网协议资源。
不过iana给的清单还有一些重复,维基百科帮它做了一下合并,整理为17项,参考地址:
https://en.wikipedia.org/wiki/Reserved_IP_addresses
IP地址段 | IP范围 | IP个数 | 范围 | 说明 |
---|---|---|---|---|
0.0.0.0/8 | 0.0.0.0–0.255.255.255 | 16,777,216 | Software | 一般用于当前网络或路由等特殊用途 |
10.0.0.0/8 | 10.0.0.0–10.255.255.255 | 16,777,216 | Private network | 用于大型的私有网络本地通信 |
100.64.0.0/10 | 100.64.0.0–100.127.255.255 | 4,194,304 | Private network | 运营商NAT用,即ISP与客户间通信的地址 |
127.0.0.0/8 | 127.0.0.0–127.255.255.255 | 16,777,216 | Host | 本地主机环回地址。 |
169.254.0.0/16 | 169.254.0.0–169.254.255.255 | 65,536 | Subnet | 一般在DHCP无法获取到地址时,为本机分配的地址 |
172.16.0.0/12 | 172.16.0.0–172.31.255.255 | 1,048,576 | Private network | 用于中型的私有网络本地通信。 |
192.0.0.0/24 | 192.0.0.0–192.0.0.255 | 256 | Private network | IETF协议分配。 |
192.0.2.0/24 | 192.0.2.0–192.0.2.255 | 256 | Documentation | 分配为TEST-NET-1,做文档和示例 |
192.88.99.0/24 | 192.88.99.0–192.88.99.255 | 256 | Internet | 保留。以前用于从IPv6到IPv4中继 |
192.168.0.0/16 | 192.168.0.0–192.168.255.255 | 65,536 | Private network | 用于小型的私有网络本地通信。 |
198.18.0.0/15 | 198.18.0.0–198.19.255.255 | 131,072 | Private network | 用于对2个独立子网间的通信通过基准测试 |
198.51.100.0/24 | 198.51.100.0–198.51.100.255 | 256 | Documentation | 分配为TEST-NET-2,做文档和示例 |
203.0.113.0/24 | 203.0.113.0–203.0.113.255 | 256 | Documentation | 分配为TEST-NET-2,做文档和示例 |
224.0.0.0/4 | 224.0.0.0–239.255.255.255 | 268,435,456 | Internet | 用于IP多播(以前的D类网络) |
233.252.0.0/24 | 233.252.0.0–233.252.0.255 | 256 | Documentation | 指定为 MCAST-TEST-NET,文档和示例 |
240.0.0.0/4 | 240.0.0.0–255.255.255.254 | 268,435,455 | Internet | 保留以备将来使用(以前的E类网络) |
255.255.255.255/32 | 255.255.255.255 | 1 | Subnet | 保留,用于“有限广播”目标地址 |
注:
IPv6的保留地址参考这里:
https://www.iana.org/assignments/iana-ipv6-special-registry/iana-ipv6-special-registry.xhtml
# 转换代码
文章的最后,贴一些转换用的代码吧:
Javascript版本的代码
有5个方法,可以看下面代码的注释:
/*** 判断给定的ip,是否属于 IANA定义的保留地址(即私有地址)* @param ip 给定的ip* @returns {boolean} 是否私有地址*/
function isPrivateIpAddr(ip) {const privateIp = [['0.0.0.0', '0.255.255.255'], // 0.0.0.0/8['10.0.0.0', '10.255.255.255'], // 10.0.0.0/8['100.64.0.0', '100.127.255.255'], // 100.64.0.0/10['127.0.0.0', '127.255.255.255'], // 127.0.0.0/8['169.254.0.0', '169.254.255.255'], // 169.254.0.0/16['172.16.0.0', '172.31.255.255'], // 172.16.0.0/12['192.0.0.0', '192.0.0.255'], // 192.0.0.0/24['192.0.2.0', '192.0.2.255'], // 192.0.2.0/24['192.88.99.0', '192.88.99.255'], // 192.88.99.0/24['192.168.0.0', '192.168.255.255'], // 192.168.0.0/16['198.18.0.0', '198.19.255.255'], // 198.18.0.0/15['198.51.100.0', '198.51.100.255'], // 198.51.100.0/24['203.0.113.0', '203.0.113.255'], // 203.0.113.0/24['224.0.0.0', '239.255.255.255'], // 224.0.0.0/4['233.252.0.0', '233.252.0.255'], // 233.252.0.0/24['240.0.0.0', '255.255.255.254'], // 240.0.0.0/4['255.255.255.255', '255.255.255.254'], // 255.255.255.255/32];for (const [startIP, endIP] of privateIp) {if (inIpAddrRange(ip, startIP, endIP)) {return true;}}return false;
}/*** 把 x.x.x.x形式的IP地址,转换为整数返回* 注:转换为无符号整型,如 '255.255.255.255'转为4294967295* 即: (255*256*256*256) + (255*256*256) + (255*256) + 255* @param ip 字符串格式的ip* @returns {number} 无符号整数*/
function ipaddrToNumber(ip) {//return (ip.split('.').reduce((acc, cur) => (acc << 8) + parseInt(cur), 0)) >>> 0;const ipArr = ip.split('.');if (ipArr.length !== 4)throw new Error('IP地址格式不对,应该有3个小数点');let ret = 0;// 验证每个项是否小于等于255for (let i = 0; i < 4; i++) {const number = parseInt(ipArr[i], 10);if (isNaN(number) || number > 255 || number < 0)throw new Error('IP地址中的每个项都应在0~255之间');ret = (ret << 8) + number;}// 转无符号数,避免负数返回return ret >>> 0;
}/*** 把一个无符号整数,转换为ip地址返回* @param number ipaddrToNumber方法计算得出的无符号整数* @returns {string} ip地址*/
function numberToIpAddr(number) {if (number < 0 || number > 4294967295) {throw new Error('参数应在0~4294967295之间');}const ipArr = [];for (let i = 3; i >= 0; i--) {ipArr[i] = (number >>> (8 * (3 - i))) & 255;}return ipArr.join('.');
}/*** 给定的ip,是否在给定的ip起止范围内* @param ip 要判断的ip* @param startIP ip范围起始值* @param endIp ip范围结束值* @returns {boolean} 是否在范围内*/
function inIpAddrRange(ip, startIP, endIp) {const ipNum = ipaddrToNumber(ip);const startIPNum = ipaddrToNumber(startIP);const endIPNum = ipaddrToNumber(endIp);return ipNum >= startIPNum && ipNum <= endIPNum;
}/*** 给定的ip,是否在给定的CIDR ip地址范围内* CIDR是用ip网址+子网掩码的表示法,如 192.168.0.0/16* @param ip 要判断的ip* @param startIP ip网址起始值* @param ipMaskNum 子网掩码* @returns {boolean} 是否在范围内*/
function inIpAddrCIDR(ip, startIP, ipMaskNum) {const ipMask = parseInt(ipMaskNum, 10);if (isNaN(ipMask) || ipMask < 1 || ipMask > 32)throw new Error('子网掩码应在1~32之间');const ipNum = ipaddrToNumber(ip);const startIPNum = ipaddrToNumber(startIP);const endIPNum = startIPNum + (Math.pow(2, 32 - ipMask) - 1);console.log(startIPNum + ':' + endIPNum);return ipNum >= startIPNum && ipNum <= endIPNum;
}
测试代码:
isPrivateIpAddr('192.168.0.1')
结果为 true
ipaddrToNumber('192.168.0.1')
结果为 3232235521
ipaddrToNumber('1.2.3.4')
结果为 16909060
ipaddrToNumber('255.255.255.255')
结果为 4294967295
MySQL的代码
查了一下官方文档,发现MySQL内置了 ip 和 number 互转的函数,文档链接:
SELECT INET_ATON('192.168.0.1') ip1,INET_ATON('1.2.3.4') ip2, INET_ATON('255.255.255.255') ip3, INET_NTOA(16909060) ip4;
上面sql得到的结果如下:
ip1 | ip2 | ip3 | ip4 |
---|---|---|---|
3232235521 | 16909060 | 4294967295 | 1.2.3.4 |
跟上面Javascript代码的结果是一致的。
我们只要把私有地址写入一张表,再把某个ip转数字后,去这张表查询是否有范围内的记录即可。
- 建表:
CREATE TABLE `sysprivateips` (`id` bigint NOT NULL AUTO_INCREMENT COMMENT '主键',`ipStart` varchar(50) COLLATE utf8mb4_general_ci NOT NULL DEFAULT '' COMMENT 'ip段起始值',`ipEnd` varchar(50) COLLATE utf8mb4_general_ci NOT NULL DEFAULT '' COMMENT 'ip段结束值',`ipMask` tinyint NOT NULL DEFAULT '0' COMMENT '掩码,暂时未用',`createTime` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间',`ipStartNum` bigint GENERATED ALWAYS AS (inet_aton(`ipStart`)) VIRTUAL,`ipEndNum` bigint GENERATED ALWAYS AS (inet_aton(`ipEnd`)) VIRTUAL,PRIMARY KEY (`id`),UNIQUE KEY `uqi_ip` (`ipStartNum`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_general_ci ROW_FORMAT=DYNAMIC COMMENT='私有ip列表'
- 插入所有的私有IP数据:
INSERT INTO `sysprivateips`(`ipStart`,`ipEnd`,`ipMask`)VALUES('0.0.0.0', '0.255.255.255', '8');
INSERT INTO `sysprivateips`(`ipStart`,`ipEnd`,`ipMask`)VALUES('10.0.0.0', '10.255.255.255', '8');
INSERT INTO `sysprivateips`(`ipStart`,`ipEnd`,`ipMask`)VALUES('100.64.0.0', '100.127.255.255', '10');
INSERT INTO `sysprivateips`(`ipStart`,`ipEnd`,`ipMask`)VALUES('127.0.0.0', '127.255.255.255', '8');
INSERT INTO `sysprivateips`(`ipStart`,`ipEnd`,`ipMask`)VALUES('169.254.0.0', '169.254.255.255', '16');
INSERT INTO `sysprivateips`(`ipStart`,`ipEnd`,`ipMask`)VALUES('172.16.0.0', '172.31.255.255', '12');
INSERT INTO `sysprivateips`(`ipStart`,`ipEnd`,`ipMask`)VALUES('192.0.0.0', '192.0.0.255', '24');
INSERT INTO `sysprivateips`(`ipStart`,`ipEnd`,`ipMask`)VALUES('192.0.2.0', '192.0.2.255', '24');
INSERT INTO `sysprivateips`(`ipStart`,`ipEnd`,`ipMask`)VALUES('192.88.99.0', '192.88.99.255', '24');
INSERT INTO `sysprivateips`(`ipStart`,`ipEnd`,`ipMask`)VALUES('192.168.0.0', '192.168.255.255', '16');
INSERT INTO `sysprivateips`(`ipStart`,`ipEnd`,`ipMask`)VALUES('198.18.0.0', '198.19.255.255', '15');
INSERT INTO `sysprivateips`(`ipStart`,`ipEnd`,`ipMask`)VALUES('198.51.100.0', '198.51.100.255', '24');
INSERT INTO `sysprivateips`(`ipStart`,`ipEnd`,`ipMask`)VALUES('203.0.113.0', '203.0.113.255', '24');
INSERT INTO `sysprivateips`(`ipStart`,`ipEnd`,`ipMask`)VALUES('224.0.0.0', '239.255.255.255', '4');
INSERT INTO `sysprivateips`(`ipStart`,`ipEnd`,`ipMask`)VALUES('233.252.0.0', '233.252.0.255', '24');
INSERT INTO `sysprivateips`(`ipStart`,`ipEnd`,`ipMask`)VALUES('240.0.0.0', '255.255.255.254', '4');
INSERT INTO `sysprivateips`(`ipStart`,`ipEnd`,`ipMask`)VALUES('255.255.255.255', '255.255.255.254', '32');
- OK,现在可以直接用sql,判断是否私有IP了:
假设要查看 192.168.0.1对应的CIDR地址段,用这个sql:
SELECT CONCAT(ipStart, '/', ipMask) AS CIDR FROM `sysprivateips`WHERE ipStartNum<=INET_ATON('192.168.0.1') AND INET_ATON('192.168.0.1')<=ipEndNum
相关文章:

互联网私有IP地址列表
最近因为业务需要,要判断用户的IP是否私有IP, 以前知道的私有IP,基本上只有如下几个(注意:这不是正确答案): 10.0.0.0/8(10.0.0.0-10.255.255.255)172.16.0.0/12(172.16.0.0-172.31…...

光伏项目管理软件为什么那么多光伏人在用?
在光伏行业迅速发展的今天,光伏项目管理软件已成为众多光伏从业者不可或缺的得力助手。那么,为何这款软件能够受到如此广泛的青睐和应用呢? 一、提高项目管理效率 光伏项目管理软件通过数字化、智能化的手段,对光伏项目的各个环节…...

《AOP实战》— 自定义注解
承接上文(传送门 —>《面试必考》 — AOP-CSDN博客),在被面试官拷打的时候,会被问到一个致命问题:“你了解aop吗?有具体的使用经验吗?” 你:......... 言尽于此,此篇…...

微前端架构下的单页应用实现策略
随着Web应用的复杂性日益增加,传统的多页应用(MPA)模式已经难以满足现代Web开发的需求。单页应用(SPA)以其流畅的用户体验和高效的页面加载速度,逐渐成为Web开发的主流模式。然而,在微前端架构下…...

JWT(JSON Web Token)工作原理及特点
JWT定义 概念:JWT是一种开放标准(RFC 7519),用于在网络上安全传输信息,常用于身份验证。比喻:类似于电子通行证,包含用户身份信息,用于身份验证和享受服务。 JWT组成部分 头部&am…...

【体检】程序人生之健康检查,全身体检与预防疫苗,五大传染病普筛,基因检测等
程序员养生指南之 【体检】程序人生之健康检查,全身体检项目分类,五大传染病普筛,基因检测等 文章目录 一、全身体检与预防疫苗(年检)1、实验室检测:生化全套检查2、医技检查:辅助诊疗科室3、科…...

汇编语言中的指令锁定:解锁高效并发编程
标题:汇编语言中的指令锁定:解锁高效并发编程 在汇编语言的微观世界中,指令锁定(Instruction Locking)是一种确保数据一致性和操作原子性的关键机制。通过使用特定的lock前缀,开发者可以告诉CPU在执行多处…...

《人工智能时代:金融投资决策的潜在系统性风险及防范策略》
在当今数字化飞速发展的时代,人工智能(AI)在金融领域的应用日益广泛,特别是在投资决策方面展现出了巨大的潜力。然而,随着其影响力的不断扩大,我们也必须警惕潜在的系统性风险。 人工智能在金融投资决策中…...

MT7621+MT7915(MT7905)+MT7975 (W7621A6G-SDK)编译固件与升级固件方法
一、搭建开发环境,编译固件。 1、安装在Ubuntu 14.04.5 x86_64系统后,然后安装下面命令行。 $ sudo apt-get install git g make libncurses5-dev subversion libssl-dev gawk libxml-parser-perl unzip wget python xz-utils vim zlibc zlib1g zlib1g…...

[php:\\filter]
写入 #题目 <?php $filename$_GET[filename]; $content$_POST[content]; file_put_contents($filename,<?php exit();.$content); highlight_file(__FILE__); ?> 源码如上,需要再服务器上写入一句话木马 payload如下: #<?php phpinf…...

Linux-环境变量
文章目录 第6章 Linux 环境变量6.1 环境变量简介?6.2 全局变量6.3 局部环境变量6.4 设置用户自定义变量6.4.1 设置局部用户自定义变量6.4.2 设置全局环境变量6.4.3 删除环境变量 6.5 默认的shell环境变量6.6 设置PATH环境变量6.7 定位系统环境变量6.7.1 登录shell6.…...

DISCUZ论坛中 “阅读权限10“这几个字的修改教程以及后台目录路径修改后的管理路径
第一篇:修改“阅读权限10”这几个字 首先找到目录: source\language\lang_message.php 找到这个文件 查找: thread_nopermission 首发地址:玖毅论坛 第二篇:后台管理路径 看到好多人在网上问discuz管理路径怎么…...

springboot 整合spring-boot-starter-data-elasticsearch
依赖 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-elasticsearch</artifactId></dependency> 配置 spring:elasticsearch:rest:uris: "http://localhost:9200" # Elastics…...

Element UI中el-dialog作为子组件如何由父组件控制显示/隐藏~
1、这里介绍的是将el-dialog作为组件封装便于复用,如何通过父组件控制子组件dialog的显示与隐藏。 2、思路:首先el-dialog是通过dialogVisible的值是否为true或false来控制显示与隐藏的。那么我们可以通过父传子props来将true(即showFlag的值࿰…...

【vue讲解:es6导入导出语法、 vue-router简单使用、登录跳转案例、scoped的使用、elementui使用】
1 es6导入导出语法 # 做项目:肯定要写模块--》导入使用# 默认导出和导入 在某个js中 # 命名导出和导入1.1 默认导出和导入 // #########导出语法########### // export default name // 只导出变量 // export default add // 只导出函数// export default {nam…...

#beego的orm一直引入失败#
在导入beego的orm的时候,一直导入失败,orm显示红色,表示导入失败 解决办法: 1:升级go,由1.7升级到1.8 2:执行以下命令 go clean go get github.com/astaxie/beego/orm go mod tidy go mod vendor 3:测试在vendor中可以看到…...

Vue插值:双大括号标签、v-text、v-html、v-bind 指令
创建应用程序实例后,需要通过插值进行数据绑定。数据绑定是 Vue.js 最核心的一个特性。建立数据绑定后,数据和视图会相互关联,当数据发生变化时,视图会自动进行更新。这样就无须手动获取 DOM 的值,使代码更加简洁&…...

实验五之用Processing绘画
1.案例代码如下: import generativedesign.*; import processing.pdf.*; import java.util.Calendar; Tablet tablet; boolean recordPDF false; float x 0, y 0; float stepSize 5.0; PFont font; String letters "Sie hren nicht die folgenden Gesnge…...

Apache CloudStack Official Document 翻译节选(七)
关于 Apache CloudStack 的 最佳实践 (一) Best Practices 部署Apache CloudStack是极具挑战性的,在整个部署过程中需要你做出形形色色的技术性选择。Apache CloudStack的配置条目是相当灵活的,这是因为在组合和配置具体条目时有…...

动态创建 Delphi 按钮的完整指南:基于配置文件的 `TGridPanel` 实现
在 Delphi 开发中,我们经常需要根据不同的配置动态生成 UI 元素。本文将带你通过一个完整的示例,演示如何根据配置文件动态创建按钮,并将它们排列在一个 TGridPanel 中。每个按钮的标题、链接、颜色和大小都将从配置文件中读取。 “C:\myApp\…...

【设计模式】工厂模式和抽象工厂模式
工厂模式 function User(role, pages) {this.role role;this.pages pages; }// new User(admin, [home, user, setting]); // new User(user, [home, user]); // new User(guest, [home]);function UserFactory(role) {switch (role) {case admin:return new User(role, [ho…...

【xilinx】Versal Adaptive SoC DDRMC - NoC QoS 选项卡未出现
在 2024.1 之前的 Vivado 版本中,用户在使用 NoC 验证块设计时可以访问 NoC 对象窗口和 QoS 选项卡。 Vivado 2024.1 中存在一个已知问题,即 NoC 对象窗口和 QoS 选项卡不出现。 要显示 NoC 对象窗口和 QoS 选项卡,请保存块设计,…...

融合创新:EasyCVR视频汇聚平台云计算技术与AI技术共筑雪亮工程智能防线
随着信息技术的飞速发展,视频云计算技术作为云计算领域的一个重要分支,正逐步在公共安全、社会治理等领域展现出其独特的优势。特别是在雪亮工程这一群众性治安防控工程中,视频云计算技术更是发挥了不可替代的作用。本文将从视频云计算技术的…...

keepalived的技术原理及其在负载均衡场景中的应用
keepalived的技术原理及其在负载均衡场景中的应用 深入探讨Keepalived及其在负载均衡场景中的应用1. **Keepalived概述**2. **Keepalived的技术原理**2.1 **VRRP协议**2.2 **健康检查机制**2.3 **脚本管理** 3. **Keepalived与LVS的结合应用**3.1 **LVS优缺点** 4. **Nginx与HA…...

树的重心 by江河湖海
引入 重心是什么? 想象你有一个由线悬挂的秋千,秋千的两端坐着两个人,如果这两个人坐在秋千的重心上,秋千就会保持平衡。在树的结构中,重心就是那个让所有节点到它那里的“距离”(可以理解为线的长度)总和最小的点。 重心为什么最多只有两个? 假设树的重心有两个,…...

MySQL存储过程深入指南
MySQL存储过程深入指南 存储过程是MySQL中一个强大的功能,能够显著提升数据库操作的效率和灵活性。本文将全面介绍存储过程的概念、语法、使用方法及最佳实践,帮助读者熟练掌握存储过程的使用。 1. 什么是存储过程? 存储过程(Stored Procedure)是预先编译并存储在数据库…...

牛客算法小题
目录 牛客.求和编辑 牛客.abb 牛客.合并k个有序链表 牛客.滑雪(暴力->递归->记忆化搜索) 牛客.旋转字符串 牛客.求和 我没想到是dfs,另外我的dfs能力确实也不强,另外难度大的是他的那个输出 import java.util.Scanne…...

小米SU7销量超特斯拉,新车明年上半年发布
小米 SU7,一款国内新能源车品牌纯血新势力旗下首款轿车,上市短短 4 个月卖出超 4 万台,月均销量过万。 该说不说,这放在整个新能源汽车工业史上也足以称得上是一件小刀喇拍屁股,让人开了眼的事儿。 就在本月初&#x…...

基于Java语言的光伏监控系统+光伏发电预测+光伏项目+光伏运维+光伏储能项目
基于Java语言的光伏监控系统光伏发电预测光伏项目光伏运维光伏储能项目 介绍 基于Java语言的光伏监控系统光伏发电系统光伏软件系统光伏监控系统源码光伏发电系统源码 基于Java语言的光伏监控系统光伏发电预测光伏项目光伏运维光伏储能项目 安装教程...

unity json 处理
1. c#对象 -> json public class Item {public int id;public int num;public Item(int id, int num){this.id id;this.num num;} } public class PlayerInfo {public string name;public int atk;public int def;public float moveSpeed;public double roundSpeed;publi…...