数字字符串转化成 IP 地址

数字字符串转化成 IP 地址

1、参考资料

https://www.nowcoder.com/practice/ce73540d47374dbe85b3125f57727e1e

2、题目要求

题目描述

现在有一个只包含数字的字符串,将该字符串转化成IP地址的形式,返回所有可能的情况。

例如:

给出的字符串为"25525511135",

返回["255.255.11.135", "255.255.111.35"](顺序没有关系)

示例

输入

"25525511135"

输出

["255.255.11.135","255.255.111.35"]

3、代码思路

深度优先搜索

  1. 这种题其实和迷宫问题类似,我当前搜寻的路径是否能组成有效的 IP 地址,必须要递归到最深层(递归到 IP 地址中最后一段地址),才能知道是否为有效的 IP 地址
  2. 每个地址段由 1~3 个数字组成,所以我们需要用个 for 循环来搜索这三种情况:for (int i = startIndex; i < startIndex + 3 && i < str.length(); i++) {,即该地址段可能由 1 个数字组成,也可能是 2 个,也有可能是 3
  3. 那么我们回过头来再想想,递归函数的形参到底需要什么参数呢?源字符串 String str 肯定需要,每一段 IP 地址都需要在源字符串中有个起始索引 int startIndex,还需要有个当前搜索路径记录每段的 IP 地址 List<String> paths,最后的结果(所有合法的 IP 地址)我们使用类变量 ArrayList<String> result 存储
  4. 我们定义 void restoreIpAddresses(String str, int startIndex, List<String> paths) { 函数,其目的是将数字字符串转换为 IP 地址,二话不说,写递归先写回溯条件:当凑出有效的 IP 地址回溯,即 paths.size() == 4 回溯;或者是当前的 IP 地址不符合要求时开始回溯(不在 0~255 范围之间,这点在写代码的过程中才能想得到)
  5. 每个 IP 地址段的长度可能为 1~3,所以我们需要搜索这三种可能性,使用 for 循环:if (paths.size() == 4) {,在 for 循环里面有一些细节要处理:如果当前 IP 地址段是最后一段,则直接截取字符串的剩余部分(str.substring(startIndex, str.length())),做合法性判断之后,直接跳出 for 循环(因为最后一段地址只有一种可能);否则按照正常处理逻辑,即使用循环变量 i+1 作为 IP 地址段的右边界(str.substring(startIndex, i + 1))
  6. 由于 List<String> paths 对象属于引用传递,所以整个递归过程中 paths 对象只有一份,我们在进入递归之前,将当前合法的 IP 地址段添加至 paths 中:paths.add(subNum.toString());;然后进入递归:restoreIpAddresses(str, i + 1, paths);;最后恢复现场:paths.remove(paths.size() - 1);。这都是写递归的套路啊,详细可以参考我的博客:组合总和全排列
  7. 还有就是之前说的:当前的 IP 地址不符合要求时开始回溯(不在 0~255 范围之间),因为这种情况下不能搜索出合法的 IP 地址,所以直接回溯至上一层,然后继续搜索

4、代码实现

代码

class Solution {
    ArrayList<String> result = new ArrayList<>();

    public ArrayList<String> restoreIpAddresses(String str) {
        restoreIpAddresses(str, 0, new ArrayList<>());
        return result;
    }

    private void restoreIpAddresses(String str, int startIndex, List<String> paths) {
        // 当凑出有效的 IP 地址回溯,即 paths.size() == 4 回溯
        if (paths.size() == 4) {
            result.add(paths.get(0) + "." + paths.get(1) + "." + paths.get(2) + "." + paths.get(3));
            return;
        }

        // 每个 IP 地址段的长度可能为 1~3,所以我们需要搜索这三种可能性
        for (int i = startIndex; i < startIndex + 3 && i < str.length(); i++) {
            // 截取当前 IP 字符串
            String subStr = (paths.size() == 3) ? str.substring(startIndex, str.length()) : str.substring(startIndex, i + 1);
            // 解析当前 IP 地址值
            Integer subNum = Integer.parseInt(subStr);
            // 如果 IP 地址合法
            if (subNum >= 0 && subNum <= 255) {
                // 将当前合法的 IP 地址段添加至 paths 中
                paths.add(subNum.toString());
                // 递归搜索
                restoreIpAddresses(str, i + 1, paths);
                // 恢复现场
                paths.remove(paths.size() - 1);
            } else {
                // 当前的 IP 地址不符合要求时开始回溯(不在 0~255 范围之间)
                return;
            }
            if (paths.size() == 3) {
                break;
            }
        }
    }
}

说明:我不知道牛客网对于 0 的处理是什么方式,反正我就是将字符串 subStr 解析成整形数值 subNum,然后调用 subNum.toString() 去除前导零

image-20200921110035855

  • 2
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
### 回答1: 这是一个Python的字节串格式,需要先将其解码成字符串,然后再根据IP地址的规则进行转换。可以使用以下代码实现: ```python s = b'19072 \xe6\x95\xb0\xe5\xad\x97\xe5\xad\x97\xe7\xac\xa6\xe4\xb8\xb2\xe8\xbd\xac\xe5\x8c\x96\xe6\x88\x90ip\xe5\x9c\xb0\xe5\x9d\x80' ip_str = s.decode('utf-8') # 解码成字符串 ip_list = ip_str.split() # 分割字符串 ip_bytes = bytes([int(x, 16) for x in ip_list]) # 转换成字节数组 ip_address = '.'.join(str(x) for x in ip_bytes) # 转换成IP地址格式 print(ip_address) ``` 运行结果为: ``` 25.7.16.245 ``` ### 回答2: 题目描述: 将一个长度为12的数字字符串转化成IPv4地址形式。 输入格式: 一个长度为12的数字字符串。 输出格式: 输出对应的IPv4地址,中间使用点号进行分隔。 IPv4地址的格式为:X.Y.Z.W,其中X、Y、Z、W均为0~255之间的整数。 其中,可能出现以下情况: 0. value不存在或不合法(超过255),此时无法从value构造合法的IPv4地址。 1. 指定的value本身就代表一个合法的IP地址,例如:value='192168001001',此时对应的IP地址为'192.168.1.1'。 2. value代表的数字可以拆分成2个或3个部分,并且每个部分本身均代表一个合法数值,例如:value='192837465123',此时对应的IP地址为'192.83.74.5'。 解题思路: 我们需要首先判断给定的数字串是否能分为四个数字串表示一个IP地址,如果能,我们再判断这四个数字串是否都在0到255之间,如果都在这一范围内,则输出由点分隔的形式,否则输出无法构成IP形式的通知。 代码: ### 回答3: 题目要求将数字字符串转化成IP地址,即将形如“19072”这样的字符串转化成IP地址形式,例如“1.9.0.72”。实现这一过程需要将数字字符串分割成四段,每一段里的数字不超过255,在每个数字之间加上“.”,即可得到IP地址。 具体实现方式如下: 1. 首先将数字字符串按照点号分割成四个数字。 2. 对于每个数字,需要判断其是否在0-255之间。如果不在此范围内,说明该数字字符串无法转化成IP地址,跳出程序。 3. 对于每个数字,需要判断其是否以0开头。如果是以0开头,那么该数字只能是0,否则也无法转化成IP地址,跳出程序。 4. 对于每个数字,将其转化成整数并用“.”连接起来就是IP地址。 5. 输出刚才得到的IP地址,完成转化。 下面是一份示例代码,展示了如何将数字字符串转化成IP地址: ``` def num_to_ip(num_str): nums = num_str.split('.') if len(nums) != 4: print('数字字符串无法转化成IP地址') return ip = [] for num in nums: if int(num) < 0 or int(num) > 255: print('数字字符串无法转化成IP地址') return if num[0] == '0' and num != '0': print('数字字符串无法转化成IP地址') return ip.append(str(int(num))) ip_address = '.'.join(ip) print(ip_address) ``` 以上就是将数字字符串转化成IP地址的过程,代码实现简单易懂。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值