字符串题目:验证 IP 地址

题目

标题和出处

标题:验证 IP 地址

出处:468. 验证 IP 地址

难度

4 级

题目描述

要求

给你一个字符串 queryIP \texttt{queryIP} queryIP,如果是有效的 IPv4 地址则返回 "IPv4" \texttt{"IPv4"} "IPv4",如果是有效的 IPv6 地址则返回 "IPv6" \texttt{"IPv6"} "IPv6",如果不是上述类型的 IP 地址则返回 "Neither" \texttt{"Neither"} "Neither"

有效的 IPv4 地址是形式为 "x 1 .x 2 .x 3 .x 4 " \texttt{"x}_\texttt{1}\texttt{.x}_\texttt{2}\texttt{.x}_\texttt{3}\texttt{.x}_\texttt{4}\texttt{"} "x1.x2.x3.x4" 的 IP 地址,其中 0 ≤ x i ≤ 255 \texttt{0} \le \texttt{x}_\texttt{i} \le \texttt{255} 0xi255 x i \texttt{x}_\texttt{i} xi 不能包含前导零。例如, "192.168.1.1" \texttt{"192.168.1.1"} "192.168.1.1" "192.168.1.0" \texttt{"192.168.1.0"} "192.168.1.0" 是有效的 IPv4 地址, "192.168.01.1" \texttt{"192.168.01.1"} "192.168.01.1" "192.168.1.00" \texttt{"192.168.1.00"} "192.168.1.00" "192.168@1.1" \texttt{"192.168@1.1"} "192.168@1.1" 是无效的 IPv4 地址。

有效的 IPv6 地址是形式为 "x 1 :x 2 :x 3 :x 4 :x 5 :x 6 :x 7 :x 8 " \texttt{"x}_\texttt{1}\texttt{:x}_\texttt{2}\texttt{:x}_\texttt{3}\texttt{:x}_\texttt{4}\texttt{:x}_\texttt{5}\texttt{:x}_\texttt{6}\texttt{:x}_\texttt{7}\texttt{:x}_\texttt{8}\texttt{"} "x1:x2:x3:x4:x5:x6:x7:x8" 的 IP 地址,其中:

  • 1 ≤ x i .length ≤ 4 \texttt{1} \le \texttt{x}_\texttt{i}\texttt{.length} \le \texttt{4} 1xi.length4
  • x i \texttt{x}_\texttt{i} xi十六进制串,包含数字、小写字母( ‘a’ \texttt{`a'} ‘a’ ‘f’ \texttt{`f'} ‘f’)和大写字母( ‘A’ \texttt{`A'} ‘A’ ‘F’ \texttt{`F'} ‘F’);
  • 允许 x i \texttt{x}_\texttt{i} xi 包含前导零。

例如, "2001:0db8:85a3:0000:0000:8a2e:0370:7334" \texttt{"2001:0db8:85a3:0000:0000:8a2e:0370:7334"} "2001:0db8:85a3:0000:0000:8a2e:0370:7334" "2001:db8:85a3:0:0:8A2E:0370:7334" \texttt{"2001:db8:85a3:0:0:8A2E:0370:7334"} "2001:db8:85a3:0:0:8A2E:0370:7334" 是有效的 IPv6 地址, "2001:0db8:85a3::8A2E:037j:7334" \texttt{"2001:0db8:85a3::8A2E:037j:7334"} "2001:0db8:85a3::8A2E:037j:7334" "02001:0db8:85a3:0000:0000:8a2e:0370:7334" \texttt{"02001:0db8:85a3:0000:0000:8a2e:0370:7334"} "02001:0db8:85a3:0000:0000:8a2e:0370:7334" 是无效的 IPv6 地址。

示例

示例 1:

输入: queryIP   =   "172.16.254.1" \texttt{queryIP = "172.16.254.1"} queryIP = "172.16.254.1"
输出: "IPv4" \texttt{"IPv4"} "IPv4"
解释:有效的 IPv4 地址,返回 "IPv4" \texttt{"IPv4"} "IPv4"

示例 2:

输入: queryIP   =   "2001:0db8:85a3:0:0:8A2E:0370:7334" \texttt{queryIP = "2001:0db8:85a3:0:0:8A2E:0370:7334"} queryIP = "2001:0db8:85a3:0:0:8A2E:0370:7334"
输出: "IPv6" \texttt{"IPv6"} "IPv6"
解释:有效的 IPv6 地址,返回 "IPv6" \texttt{"IPv6"} "IPv6"

示例 3:

输入: queryIP   =   "256.256.256.256" \texttt{queryIP = "256.256.256.256"} queryIP = "256.256.256.256"
输出: "Neither" \texttt{"Neither"} "Neither"
解释:既不是 IPv4 地址,又不是 IPv6 地址。

示例 4:

输入: queryIP   =   "2001:0db8:85a3:0:0:8A2E:0370:7334:" \texttt{queryIP = "2001:0db8:85a3:0:0:8A2E:0370:7334:"} queryIP = "2001:0db8:85a3:0:0:8A2E:0370:7334:"
输出: "Neither" \texttt{"Neither"} "Neither"

示例 5:

输入: queryIP   =   "1e1.4.5.6" \texttt{queryIP = "1e1.4.5.6"} queryIP = "1e1.4.5.6"
输出: "Neither" \texttt{"Neither"} "Neither"

数据范围

  • queryIP \texttt{queryIP} queryIP 仅由英语字母、数字、字符 ‘.’ \texttt{`.'} ‘.’ ‘:’ \texttt{`:'} ‘:’ 组成

前言

有效的 IP 地址有两种,分别是 IPv4 地址和 IPv6 地址。任何一个有效的 IP 地址一定是其中的一种,因此我们可以依次判断给定的 IP 地址是否为有效的 IPv4 地址和 IPv6 地址。

由此可以得到如下代码框架:

class Solution {
    public String validIPAddress(String queryIP) {
        if (isValidIPv4(queryIP)) {
            return "IPv4";
        } else if (isValidIPv6(queryIP)) {
            return "IPv6";
        } else {
            return "Neither";
        }
    }
}

判断给定的 IP 地址是否为有效的 IPv4 地址和 IPv6 地址有多种方法,可以将 IP 地址转成字符串数组然后判断,也可以直接遍历 IP 地址进行判断。

解法一

思路和算法

可以将 IP 地址转成字符串数组然后判断。以下分别说明 IPv4 地址和 IPv6 地址的判断方法。

对于 IPv4 地址的判断方法如下:

  1. 长度为 0 0 0 的 IP 地址不是 IPv4 地址;

  2. 如果 IP 地址以 ‘.’ \text{`.'} ‘.’ 开始或结束,则不是 IPv4 地址;

  3. 如果 IP 地址包含 ‘+’ \text{`+'} ‘+’ ‘–’ \text{`--'} ‘–’,则不是 IPv4 地址,这一步判断是为了在第 7 步避免 ‘+’ \text{`+'} ‘+’ ‘–’ \text{`--'} ‘–’ 对判断造成影响;

  4. 将给定的 IP 地址使用 ‘.’ \text{`.'} ‘.’ 作为分隔转成字符串数组;

  5. 如果字符串数组的长度不为 4 4 4,则不是 IPv4 地址;

  6. 判断字符串数组的每个元素,长度必须在 1 1 1 3 3 3 之间,且长度大于 1 1 1 时第一个字符不能为 0 0 0,如果不符合要求,则不是 IPv4 地址;

  7. 字符串数组的每个元素都应该可以转换成不超过 255 255 255 的整数,如果转换成的整数大于 255 255 255 或者转换失败(包含非数字的字符),则不是 IPv4 地址,可以通过 try-catch \texttt{try-catch} try-catch 块实现。

对于 IPv6 地址的判断方法如下:

  1. 长度为 0 0 0 的 IP 地址不是 IPv6 地址;

  2. 如果 IP 地址以 ‘:’ \text{`:'} ‘:’ 开始或结束,则不是 IPv6 地址;

  3. 将给定的 IP 地址使用 ‘:’ \text{`:'} ‘:’ 作为分隔转成字符串数组;

  4. 如果字符串数组的长度不为 8 8 8,则不是 IPv6 地址;

  5. 判断字符串数组的每个元素,长度必须在 1 1 1 4 4 4 之间,否则不是 IPv6 地址;

  6. 字符串数组的每个元素都只能包含十六进制表示的字符,包括数字、大写字母 ‘A’ \text{`A'} ‘A’ ‘F’ \text{`F'} ‘F’ 和小写字母 ‘a’ \text{`a'} ‘a’ ‘f’ \text{`f'} ‘f’,否则不是 IPv6 地址。

代码

class Solution {
    public String validIPAddress(String queryIP) {
        if (isValidIPv4(queryIP)) {
            return "IPv4";
        } else if (isValidIPv6(queryIP)) {
            return "IPv6";
        } else {
            return "Neither";
        }
    }

    public boolean isValidIPv4(String queryIP) {
        int queryIPLength = queryIP.length();
        if (queryIPLength == 0) {
            return false;
        }
        if (queryIP.charAt(0) == '.' || queryIP.charAt(queryIPLength - 1) == '.') {
            return false;
        }
        if (queryIP.indexOf('+') >= 0 || queryIP.indexOf('-') >= 0) {
            return false;
        }
        String[] array = queryIP.split("\\.");
        if (array.length != 4) {
            return false;
        }
        int length = array.length;
        for (int i = 0; i < length; i++) {
            String numStr = array[i];
            if (numStr.length() == 0 || numStr.length() > 3 || numStr.length() > 1 && numStr.charAt(0) == '0') {
                return false;
            }
            try {
                int num = Integer.parseInt(numStr);
                if (num > 255) {
                    return false;
                }
            } catch (NumberFormatException ex) {
                return false;
            }
        }
        return true;
    }

    public boolean isValidIPv6(String queryIP) {
        int queryIPLength = queryIP.length();
        if (queryIPLength == 0) {
            return false;
        }
        if (queryIP.charAt(0) == ':' || queryIP.charAt(queryIPLength - 1) == ':') {
            return false;
        }
        String[] array = queryIP.split(":");
        if (array.length != 8) {
            return false;
        }
        int length = array.length;
        for (int i = 0; i < length; i++) {
            String numStr = array[i];
            if (numStr.length() == 0 || numStr.length() > 4) {
                return false;
            }
            int curLength = numStr.length();
            for (int j = 0; j < curLength; j++) {
                char c = numStr.charAt(j);
                if (!isHexChar(c)) {
                    return false;
                }
            }
        }
        return true;
    }

    private boolean isHexChar(char c) {
        return Character.isDigit(c) || Character.isUpperCase(c) && c <= 'F' || Character.isLowerCase(c) && c <= 'f';
    }
}

复杂度分析

  • 时间复杂度: O ( n ) O(n) O(n),其中 n n n 是字符串 queryIP \textit{queryIP} queryIP 的长度。判断 IPv4 地址最多需要遍历字符串数组两次,判断 IPv6 地址最多需要遍历字符串数组一次,每次遍历字符串数组需要 O ( n ) O(n) O(n) 的时间。

  • 空间复杂度: O ( n ) O(n) O(n),其中 n n n 是字符串 queryIP \textit{queryIP} queryIP 的长度。需要将字符串 queryIP \textit{queryIP} queryIP 转成字符串数组,数组中的各元素长度之和不超过 n n n

解法二

思路和算法

也可以直接遍历 IP 地址进行判断。在遍历的过程中,维护 IP 地址的片段数,判断每个片段的字符是否合法。

以下分别说明 IPv4 地址和 IPv6 地址的判断方法。

对于 IPv4 地址的判断如下:

  1. 片段之间使用 ‘.’ \text{`.'} ‘.’ 分隔;

  2. 每个片段只包含数字,且每个片段的数字不超过 255 255 255

  3. 当一个片段的长度大于 1 1 1 时,不能有前导零;

  4. 不能有两个相邻的 ‘.’ \text{`.'} ‘.’

  5. 共有 4 4 4 个片段。

对于 IPv6 地址的判断方法如下:

  1. 片段之间使用 ‘:’ \text{`:'} ‘:’ 分隔;

  2. 每个片段的长度必须在 1 1 1 4 4 4 之间;

  3. 每个片段只能包含十六进制表示的字符,包括数字、大写字母 ‘A’ \text{`A'} ‘A’ ‘F’ \text{`F'} ‘F’ 和小写字母 ‘a’ \text{`a'} ‘a’ ‘f’ \text{`f'} ‘f’

  4. 不能有两个相邻的 ‘:’ \text{`:'} ‘:’

  5. 共有 8 8 8 个片段。

具体实现方面,遍历 IP 地址时,如果遇到不合法的字符则直接返回 false \text{false} false,对于合法字符,遇到非分隔符和分隔符时(分隔符指 IPv4 的 ‘.’ \text{`.'} ‘.’ 和 IPv6 的 ‘:’ \text{`:'} ‘:’),分别进行如下操作:

  • 遇到非分隔符时,更新当前片段的状态并判断当前片段是否符合要求;

  • 遇到分隔符时,如果发现相邻两个字符都是分隔符则直接返回 false \text{false} false,其余情况则将片段数量加 1 1 1

遍历结束时需要再次对片段数量加 1 1 1,然后对片段数量和最后一个片段进行判断。

代码

class Solution {
    public String validIPAddress(String queryIP) {
        if (isValidIPv4(queryIP)) {
            return "IPv4";
        } else if (isValidIPv6(queryIP)) {
            return "IPv6";
        } else {
            return "Neither";
        }
    }

    public boolean isValidIPv4(String queryIP) {
        int num = 0;
        int count = 0;
        char prev = '.';
        int length = queryIP.length();
        for (int i = 0; i < length; i++) {
            char curr = queryIP.charAt(i);
            if (Character.isDigit(curr)) {
                if (num == 0 && prev == '0') {
                    return false;
                }
                num = num * 10 + curr - '0';
                if (num > 255) {
                    return false;
                }
            } else if (curr == '.') {
                if (prev == '.') {
                    return false;
                }
                count++;
                num = 0;
            } else {
                return false;
            }
            prev = curr;
        }
        count++;
        return num <= 255 && count == 4 && prev != '.';
    }

    public boolean isValidIPv6(String queryIP) {
        int count = 0;
        int partLength = 0;
        char prev = ':';
        int length = queryIP.length();
        for (int i = 0; i < length; i++) {
            char curr = queryIP.charAt(i);
            if (isHexChar(curr)) {
                partLength++;
                if (partLength > 4) {
                    return false;
                }
            } else if (curr == ':') {
                if (prev == ':') {
                    return false;
                }
                count++;
                partLength = 0;
            } else {
                return false;
            }
            prev = curr;
        }
        count++;
        return count == 8 && prev != ':';
    }

    private boolean isHexChar(char c) {
        return Character.isDigit(c) || Character.isUpperCase(c) && c <= 'F' || Character.isLowerCase(c) && c <= 'f';
    }
}

复杂度分析

  • 时间复杂度: O ( n ) O(n) O(n),其中 n n n 是字符串 queryIP \textit{queryIP} queryIP 的长度。判断 IPv4 地址和判断 IPv6 地址各需要遍历字符串 queryIP \textit{queryIP} queryIP 一次。

  • 空间复杂度: O ( 1 ) O(1) O(1)

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
### 回答1: 以下是一个 Python 实现: ```python ip_bin = input("请输入32位二进制形式的IP地址:") # 判断输入是否合法 if set(ip_bin) - set("01"): print("该字符串不是正确的IP地址") else: # 在不足32位时自动补0 ip_bin = ip_bin.zfill(32) # 将二进制转换为十进制 ip_dec = [str(int(ip_bin[i:i+8], 2)) for i in range(0, 32, 8)] # 输出点分十进制表示的IP地址 print(".".join(ip_dec)) ``` 示例输出: ``` 请输入32位二进制形式的IP地址:11000000101010000000000100000011 192.168.1.131 ``` ### 回答2: 题目要求通过键盘输入一个32位的二进制形式的IP地址,将其转换为点分十进制的表示。如果输入的字符串中含有1和0以外的字符,则输出“该字符串不是正确的IP地址”。 首先,我们需要从键盘获取输入的32位二进制序列,并判断输入的字符串是否符合要求。 然后,我们需要将32位的二进制序列转换为点分十进制形式的IP地址。 具体步骤如下: 1. 从键盘获取输入的32位二进制序列,可以使用input()函数。 2. 判断输入的字符串是否符合要求,即是否只包含0和1,可以使用正则表达式进行匹配。 3. 如果输入的字符串不符合要求,输出“该字符串不是正确的IP地址”,并结束程序。 4. 如果输入的字符串符合要求,则将其分为4个8位的子序列。 5. 对每个子序列进行二进制转十进制的转换,即将每个子序列从二进制转换为十进制。 6. 将每个子序列的十进制表示拼接起来,中间用点号分隔,得到点分十进制形式的IP地址。 7. 输出点分十进制形式的IP地址。 下面是一个示例代码: ```python import re # 从键盘获取输入的32位二进制序列 binary_sequence = input("请输入32位二进制序列:") # 判断输入的字符串是否符合要求 if not re.match(r'^[01]{32}$', binary_sequence): # 输入的字符串不符合要求,输出提示信息 print("该字符串不是正确的IP地址") else: # 将32位的二进制序列分为4个8位的子序列 sub_sequences = [binary_sequence[i:i+8] for i in range(0, 32, 8)] # 将每个子序列从二进制转换为十进制,并拼接起来得到IP地址 ip_address = '.'.join(str(int(sub, 2)) for sub in sub_sequences) # 输出点分十进制形式的IP地址 print("点分十进制IP地址:", ip_address) ``` 注意:以上代码假设输入的二进制序列中只包含0和1,并且长度为32位。 ### 回答3: 题目要求从键盘输入一个32位的二进制形式的IP地址,将其转换为点分十进制的表示。如果输入的字符串中含有1和0以外的字符,则输出“该字符串不是正确的IP地址”。 首先,从键盘接受输入,得到一个字符串形式的二进制IP地址。 接下来,我们需要对输入的字符串进行验证,确保其只包含0和1。我们可以使用正则表达式来进行判断,如果输入的字符串中含有除了0和1之外的字符,则输出“该字符串不是正确的IP地址”。 若输入的字符串通过了验证,我们可以用每8位二进制数来表示一个十进制数,在点分IP地址中使用点分隔。所以我们需将32位的二进制IP地址分成4个8位二进制数。 然后,将每个8位二进制数转换为十进制数。 最后,将四个十进制数使用点分隔,得到最终的点分IP地址格式。 以下是实现该过程的一个示例代码: ```python import re binary_ip = input("请输入一个32位的二进制形式的IP地址:") if not re.match(r'^[01]{32}$', binary_ip): print("该字符串不是正确的IP地址") else: decimal_ip = [] for i in range(0, 32, 8): decimal_ip.append(str(int(binary_ip[i:i+8], 2))) print("点分IP地址为:" + ".".join(decimal_ip)) ``` 注意:这个示例代码假设输入的二进制IP地址有效的,即输入的位数是32位。如果需要对位数进行验证,可以在正则表达式中增加相应的条件。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

伟大的车尔尼

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值