• 暑假总结-集成ip2region实现离线IP地址定位


    本周总结:本周主要对项目进行准备工作:立项、建立文档、建表、项目需求、分工等,在学习的过程中学习了一个功能,来分享一下

    阅读须知

    • 本文是自己总结文档和github源码,绝无抄袭,我先打个水印哈在这里插入图片描述

    • 本文是看了Ruoyi-vue的github开源项目有所总结Ruoyi文档描述

      • 文档里的代码直接复制会报错,缺少一些他的项目里面的类
      • 下面的部分代码是复制文档的,看文档只需要找到ip2region即可
      • 下面的代码是基于文档的一些部分改动一下,可以直接复制运行
      • 记得导入StringUtils的包,一些地方会用得到

    集成ip2region实现离线IP地址定位,为啥是实现离线IP地址定位,因为在线的IP定位会给网络带来资源消耗,离线的话,我们只需要访问内部的一个文件即可

    • 第一步 引入依赖
    <!-- 离线IP地址定位库 -->
    <dependency>
    	<groupId>org.lionsoul</groupId>
    	<artifactId>ip2region</artifactId>
    	<version>1.7.2</version>
    </dependency>
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 第二步 添加工具类 RegionUtil.java 不用理会水印在这里插入图片描述
    import java.io.File;
    import java.io.InputStream;
    import java.lang.reflect.Method;
    import org.apache.commons.io.FileUtils;
    import org.lionsoul.ip2region.DataBlock;
    import org.lionsoul.ip2region.DbConfig;
    import org.lionsoul.ip2region.DbSearcher;
    import org.lionsoul.ip2region.Util;
    import org.slf4j.Logger;
    import org.slf4j.LoggerFactory;
    import org.springframework.core.io.ClassPathResource;
    
    /**
     * 根据ip离线查询地址
     *
     * @author ruoyi
     */
    public class RegionUtil
    {
        private static final Logger log = LoggerFactory.getLogger(RegionUtil.class);
    
        private static final String JAVA_TEMP_DIR = "java.io.tmpdir";
    
        static DbConfig config = null;
        static DbSearcher searcher = null;
    
        /**
         * 初始化IP库
         */
        static
        {
            try
            {
                // 因为jar无法读取文件,复制创建临时文件
                String dbPath = RegionUtil.class.getResource("/ip2region/ip2region.db").getPath();
                File file = new File(dbPath);
                if (!file.exists())
                {
                    String tmpDir = System.getProperties().getProperty(JAVA_TEMP_DIR);
                    dbPath = tmpDir + "ip2region.db";
                    file = new File(dbPath);
                    ClassPathResource cpr = new ClassPathResource("ip2region" + File.separator + "ip2region.db");
                    InputStream resourceAsStream = cpr.getInputStream();
                    if (resourceAsStream != null)
                    {
                        FileUtils.copyInputStreamToFile(resourceAsStream, file);
                    }
                }
                config = new DbConfig();
                searcher = new DbSearcher(config, dbPath);
                log.info("bean [{}]", config);
                log.info("bean [{}]", searcher);
            }
            catch (Exception e)
            {
                log.error("init ip region error:{}", e);
            }
        }
    
        /**
         * 解析IP
         *
         * @param ip
         * @return
         */
        public static String getRegion(String ip)
        {
            try
            {
                // db
                if (searcher == null || StringUtils.isEmpty(ip))
                {
                    log.error("DbSearcher is null");
                    return StringUtils.EMPTY;
                }
                long startTime = System.currentTimeMillis();
                // 查询算法
                int algorithm = DbSearcher.MEMORY_ALGORITYM;
                Method method = null;
                switch (algorithm)
                {
                    case DbSearcher.BTREE_ALGORITHM:
                        method = searcher.getClass().getMethod("btreeSearch", String.class);
                        break;
                    case DbSearcher.BINARY_ALGORITHM:
                        method = searcher.getClass().getMethod("binarySearch", String.class);
                        break;
                    case DbSearcher.MEMORY_ALGORITYM:
                        method = searcher.getClass().getMethod("memorySearch", String.class);
                        break;
                }
    
                DataBlock dataBlock = null;
                if (Util.isIpAddress(ip) == false)
                {
                    log.warn("warning: Invalid ip address");
                }
                dataBlock = (DataBlock) method.invoke(searcher, ip);
                String result = dataBlock.getRegion();
                long endTime = System.currentTimeMillis();
                log.debug("region use time[{}] result[{}]", endTime - startTime, result);
                return result;
    
            }
            catch (Exception e)
            {
                log.error("error:{}", e);
            }
            return StringUtils.EMPTY;
        }
    
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77
    • 78
    • 79
    • 80
    • 81
    • 82
    • 83
    • 84
    • 85
    • 86
    • 87
    • 88
    • 89
    • 90
    • 91
    • 92
    • 93
    • 94
    • 95
    • 96
    • 97
    • 98
    • 99
    • 100
    • 101
    • 102
    • 103
    • 104
    • 105
    • 106
    • 107
    • 108
    • 109
    • 110
    • 111
    • 112
    • 第三步 创建AddressUtils.java,不用理会水印在这里插入图片描述
    
    import org.apache.commons.lang3.StringUtils;
    import org.slf4j.Logger;
    import org.slf4j.LoggerFactory;
    
    /**
     * 获取地址类
     *
     * @author ruoyi
     */
    public class AddressUtils
    {
        private static final Logger log = LoggerFactory.getLogger(AddressUtils.class);
    
        // 未知地址
        public static final String UNKNOWN = "XX XX";
    
        public static String getRealAddressByIP(String ip)
        {
            String address = UNKNOWN;
            // 内网不查询
            if (IpUtils.internalIp(ip))
            {
                return "内网IP";
            }else {
                try
                {
                    String rspStr = RegionUtil.getRegion(ip);
                    if (StringUtils.isEmpty(rspStr))
                    {
                        log.error("获取地理位置异常 {}", ip);
                        return UNKNOWN;
                    }
                    String[] obj = rspStr.split("\\|");
                    String region = obj[2];
                    String city = obj[3];
    
                    return String.format("%s %s", region, city);
                }
                catch (Exception e)
                {
                    log.error("获取地理位置异常 {}", e);
                }
            }
            return address;
        }
    }
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 第四步 需要创建IpUtils,这个里面方法很多,测试只用到了其中一个方法,大家可以自己探索
    
    import org.apache.commons.lang3.StringUtils;
    
    import javax.servlet.http.HttpServletRequest;
    import java.net.InetAddress;
    import java.net.UnknownHostException;
    
    /**
     * 获取IP方法
     * 
     * @author ruoyi
     */
    public class IpUtils
    {
        /**
         * 获取客户端IP
         * 
         * @param request 请求对象
         * @return IP地址
         */
        public static String getIpAddr(HttpServletRequest request)
        {
            if (request == null)
            {
                return "unknown";
            }
            String ip = request.getHeader("x-forwarded-for");
            if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip))
            {
                ip = request.getHeader("Proxy-Client-IP");
            }
            if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip))
            {
                ip = request.getHeader("X-Forwarded-For");
            }
            if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip))
            {
                ip = request.getHeader("WL-Proxy-Client-IP");
            }
            if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip))
            {
                ip = request.getHeader("X-Real-IP");
            }
    
            if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip))
            {
                ip = request.getRemoteAddr();
            }
    
            return "0:0:0:0:0:0:0:1".equals(ip) ? "127.0.0.1" : getMultistageReverseProxyIp(ip);
        }
    
        /**
         * 检查是否为内部IP地址
         * 
         * @param ip IP地址
         * @return 结果
         */
        public static boolean internalIp(String ip)
        {
            byte[] addr = textToNumericFormatV4(ip);
            return internalIp(addr) || "127.0.0.1".equals(ip);
        }
    
        /**
         * 检查是否为内部IP地址
         * 
         * @param addr byte地址
         * @return 结果
         */
        private static boolean internalIp(byte[] addr)
        {
            if (isNull(addr) || addr.length < 2)
            {
                return true;
            }
            final byte b0 = addr[0];
            final byte b1 = addr[1];
            // 10.x.x.x/8
            final byte SECTION_1 = 0x0A;
            // 172.16.x.x/12
            final byte SECTION_2 = (byte) 0xAC;
            final byte SECTION_3 = (byte) 0x10;
            final byte SECTION_4 = (byte) 0x1F;
            // 192.168.x.x/16
            final byte SECTION_5 = (byte) 0xC0;
            final byte SECTION_6 = (byte) 0xA8;
            switch (b0)
            {
                case SECTION_1:
                    return true;
                case SECTION_2:
                    if (b1 >= SECTION_3 && b1 <= SECTION_4)
                    {
                        return true;
                    }
                case SECTION_5:
                    switch (b1)
                    {
                        case SECTION_6:
                            return true;
                    }
                default:
                    return false;
            }
        }
    
        /**
         * 将IPv4地址转换成字节
         * 
         * @param text IPv4地址
         * @return byte 字节
         */
        public static byte[] textToNumericFormatV4(String text)
        {
            if (text.length() == 0)
            {
                return null;
            }
    
            byte[] bytes = new byte[4];
            String[] elements = text.split("\\.", -1);
            try
            {
                long l;
                int i;
                switch (elements.length)
                {
                    case 1:
                        l = Long.parseLong(elements[0]);
                        if ((l < 0L) || (l > 4294967295L))
                        {
                            return null;
                        }
                        bytes[0] = (byte) (int) (l >> 24 & 0xFF);
                        bytes[1] = (byte) (int) ((l & 0xFFFFFF) >> 16 & 0xFF);
                        bytes[2] = (byte) (int) ((l & 0xFFFF) >> 8 & 0xFF);
                        bytes[3] = (byte) (int) (l & 0xFF);
                        break;
                    case 2:
                        l = Integer.parseInt(elements[0]);
                        if ((l < 0L) || (l > 255L))
                        {
                            return null;
                        }
                        bytes[0] = (byte) (int) (l & 0xFF);
                        l = Integer.parseInt(elements[1]);
                        if ((l < 0L) || (l > 16777215L))
                        {
                            return null;
                        }
                        bytes[1] = (byte) (int) (l >> 16 & 0xFF);
                        bytes[2] = (byte) (int) ((l & 0xFFFF) >> 8 & 0xFF);
                        bytes[3] = (byte) (int) (l & 0xFF);
                        break;
                    case 3:
                        for (i = 0; i < 2; ++i)
                        {
                            l = Integer.parseInt(elements[i]);
                            if ((l < 0L) || (l > 255L))
                            {
                                return null;
                            }
                            bytes[i] = (byte) (int) (l & 0xFF);
                        }
                        l = Integer.parseInt(elements[2]);
                        if ((l < 0L) || (l > 65535L))
                        {
                            return null;
                        }
                        bytes[2] = (byte) (int) (l >> 8 & 0xFF);
                        bytes[3] = (byte) (int) (l & 0xFF);
                        break;
                    case 4:
                        for (i = 0; i < 4; ++i)
                        {
                            l = Integer.parseInt(elements[i]);
                            if ((l < 0L) || (l > 255L))
                            {
                                return null;
                            }
                            bytes[i] = (byte) (int) (l & 0xFF);
                        }
                        break;
                    default:
                        return null;
                }
            }
            catch (NumberFormatException e)
            {
                return null;
            }
            return bytes;
        }
    
        /**
         * 获取IP地址
         * 
         * @return 本地IP地址
         */
        public static String getHostIp()
        {
            try
            {
                return InetAddress.getLocalHost().getHostAddress();
            }
            catch (UnknownHostException e)
            {
            }
            return "127.0.0.1";
        }
    
        /**
         * 获取主机名
         * 
         * @return 本地主机名
         */
        public static String getHostName()
        {
            try
            {
                return InetAddress.getLocalHost().getHostName();
            }
            catch (UnknownHostException e)
            {
            }
            return "未知";
        }
    
        /**
         * 从多级反向代理中获得第一个非unknown IP地址
         *
         * @param ip 获得的IP地址
         * @return 第一个非unknown IP地址
         */
        public static String getMultistageReverseProxyIp(String ip)
        {
            // 多级反向代理检测
            if (ip != null && ip.indexOf(",") > 0)
            {
                final String[] ips = ip.trim().split(",");
                for (String subIp : ips)
                {
                    if (false == isUnknown(subIp))
                    {
                        ip = subIp;
                        break;
                    }
                }
            }
            return ip;
        }
    
        /**
         * 检测给定字符串是否为未知,多用于检测HTTP请求相关
         *
         * @param checkString 被检测的字符串
         * @return 是否未知
         */
        public static boolean isUnknown(String checkString)
        {
            return StringUtils.isBlank(checkString) || "unknown".equalsIgnoreCase(checkString);
        }
    
    
        public static boolean isNull(Object object)
        {
            return object == null;
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77
    • 78
    • 79
    • 80
    • 81
    • 82
    • 83
    • 84
    • 85
    • 86
    • 87
    • 88
    • 89
    • 90
    • 91
    • 92
    • 93
    • 94
    • 95
    • 96
    • 97
    • 98
    • 99
    • 100
    • 101
    • 102
    • 103
    • 104
    • 105
    • 106
    • 107
    • 108
    • 109
    • 110
    • 111
    • 112
    • 113
    • 114
    • 115
    • 116
    • 117
    • 118
    • 119
    • 120
    • 121
    • 122
    • 123
    • 124
    • 125
    • 126
    • 127
    • 128
    • 129
    • 130
    • 131
    • 132
    • 133
    • 134
    • 135
    • 136
    • 137
    • 138
    • 139
    • 140
    • 141
    • 142
    • 143
    • 144
    • 145
    • 146
    • 147
    • 148
    • 149
    • 150
    • 151
    • 152
    • 153
    • 154
    • 155
    • 156
    • 157
    • 158
    • 159
    • 160
    • 161
    • 162
    • 163
    • 164
    • 165
    • 166
    • 167
    • 168
    • 169
    • 170
    • 171
    • 172
    • 173
    • 174
    • 175
    • 176
    • 177
    • 178
    • 179
    • 180
    • 181
    • 182
    • 183
    • 184
    • 185
    • 186
    • 187
    • 188
    • 189
    • 190
    • 191
    • 192
    • 193
    • 194
    • 195
    • 196
    • 197
    • 198
    • 199
    • 200
    • 201
    • 202
    • 203
    • 204
    • 205
    • 206
    • 207
    • 208
    • 209
    • 210
    • 211
    • 212
    • 213
    • 214
    • 215
    • 216
    • 217
    • 218
    • 219
    • 220
    • 221
    • 222
    • 223
    • 224
    • 225
    • 226
    • 227
    • 228
    • 229
    • 230
    • 231
    • 232
    • 233
    • 234
    • 235
    • 236
    • 237
    • 238
    • 239
    • 240
    • 241
    • 242
    • 243
    • 244
    • 245
    • 246
    • 247
    • 248
    • 249
    • 250
    • 251
    • 252
    • 253
    • 254
    • 255
    • 256
    • 257
    • 258
    • 259
    • 260
    • 261
    • 262
    • 263
    • 264
    • 265
    • 266
    • 267
    • 268
    • 269
    • 270
    • 第五步 添加离线IP地址库插件,

      • 如果失效了请去文档里面下载他的集成插件堆我只是将这个ip2region单独放了出来
      • 下载集成ip2region离线地址定位.zip
      • 链接:ip2region离线地址定位.zip 提取码:tdkp
      • 解压之后找到这个
        在这里插入图片描述
        在src/main/resources下新建ip2region复制文件ip2region.db到目录下
        在这里插入图片描述
    • 第六步 测试

      • 外网访问
        在这里插入图片描述
      • 内网访问
        在这里插入图片描述
  • 相关阅读:
    Python自学教程1-安装pycharm和执行环境
    Flask框架中Jinja2模板配置静态文件(url_for)
    java key锁 实现对某个key(字符串)加同步锁 带详细注释
    基于未知环境碰撞冲突预测的群机器人多目标搜索研究
    @Transactional 注解 同一个类下的两个方法
    数据链路层
    通信原理学习笔记3-4:数字通信系统性能指标(带宽、信噪比Eb/N0、可靠性与误码率、有效性与频谱利用率)
    JS-BOM-阶乘计算
    一个整合性、功能丰富的.NET网络通信框架
    [小记]shell获取git最近一次提交信息
  • 原文地址:https://blog.csdn.net/qq_51959224/article/details/126329272