百度&必应权4, 日IP1w+ 查看详情
自助收录

2023年华为OD机考真题:不含101的数

算法刷题1个月前更新 江南白衣
644 0 0
2023年华为OD机考真题:不含101的数

题目:不含101的数
时间限制:1s 空间限制:256MB 限定语言:不限
题目描述:
小明在学习二进制时,发现了一类不含101的数,也就是:
将数字用二进制表示,不能出现101。
现在给定一个正整数区间[l,r],请问这个区间内包含了多少个不含101的数?
输入描述:
输入的唯一一行包含两个正整数l,r(1<=l<r<=109)。
输出描述:
输出的唯一一行包含一个整数,表示在[l,r]区间内一共有几个不含101的数。
示例1
输入:
1 10
输出:
8
说明:
区间[1,10]内,5的二进制表示为101,10的二进制表示为1010,因此除了5与10不满足条件外,其他数字都满足条件,因此答案为8。
示例2
输入:
10 20
输出:
7
说明:
区间[10,20]内,满足条件的数字有[12,14,15,16,17,18,19],因此答案为7。
解题思路:暴力算法,会导致部分用例超时。
本题非常简单,就是使用Integer.toBinaryString将十进制转化为二进制,然后判断是否存在101。

package com.codernav.demo.hwod.exam;

import java.util.Scanner;

/**
 * @title 不含101的数
 * @Description 小明在学习二进制时,发现了一类不含101的数,也就是:
 * 将数字用二进制表示,不能出现101。
 * 现在给定一个正整数区间[l,r],请问这个区间内包含了多少个不含101的数?
 * @Author 开发者导航
 * @website https://codernav.com
 * @date 2023/5/14
 */
public class Main {
    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);
        int l = sc.nextInt();
        int r = sc.nextInt();
        int res = r - l + 1;
        for (int i = l; i <= r; i++) {
            String binary = Integer.toBinaryString(i);
            if (binary.contains("101")) {
                res--;
            }
        }
        System.out.println(res);
    }
}

算法二(解题思路):以下是满分Java题解
因为0和1也不满足101,所有我们设置dp[0]=1,dp[1]=2
下面是二进制11,也就是3的满二叉树(虽是三层树,但是根节点不计数,所以我们看成2层),因为图中不包含101,所以结果是dp[2]=4

2023年华为OD机考真题:不含101的数
下面是111,也就是7的满二叉树,因为包含一个101,所以结果是dp[3]=7

2023年华为OD机考真题:不含101的数
从上面两张图其实可以看出来,三层树的左子树其实就相当于二层满二叉树,也就是说
dp[3] = dp[2] + 右子树
而右子树的根节点肯定是1,右子树的左子树的根节点肯定是0,而右子树的左子树的右子树的根节点又是1,满足101,如图所示:

2023年华为OD机考真题:不含101的数
这条线肯定是要被抛弃的,那就是需要加上右子树的左子树的左子树,也就是需要加上下两层的左子树,而上一步我们推导出:左子树 = 下一层的满二叉树,转换一下就是需要加上下三层的满二叉树:
dp[3] = dp[3-1] + dp[3-3] + 右子树的右子树
满二叉树除了根节点不一样外,其实其他节点都是一样的,也就是说右子树的右子树等于左子树的右子树,如图:

2023年华为OD机考真题:不含101的数
左子树的右子树 = 下一层的满二叉树 – 下一层的左子树,上面我们求得了左子树 = 下一层的满二叉树,那么左子树的右子树 = 下一层的满二叉树 – 下下层的满二叉树;最终得出结论:
dp[3] = dp[3-1] + dp[3-3] + dp[3-1] – dp[3-2]
= dp[2] + dp[0] + dp[2] - dp[1]
= 4 + 1 + 4 - 2= 7
再给大家展示一下1111,也就是15的满二叉树感受一下:

2023年华为OD机考真题:不含101的数
dp[4] = dp[3] + dp[1] + dp[3] – dp[2] = 7 + 2+ 7 – 4 = 12
最终得出结论:满足二叉树的不包含101的个数公式为:
dp[n] = dp[n-1] + dp[n-3] + dp[n-1] + dp[n-2] (n>2)
dp[0] = 1
dp[1] = 2
dp[2] = 4
以上讨论的是满二叉树的情况,那么非满二叉树的情况呢?
首先判断是否存在右子树,因为如果存在右子树,那么左子树就是满二叉树,直接套用公式:
如110,也就是6的情况:

2023年华为OD机考真题:不含101的数
首先让6 & 2^3=8进行与运算,110&1000=0,说明在4层没有右子节点;
继续跟2^2 = 4进行与运算,110&100 = 100 != 0,说明有右子节点,左子树为2层满二叉树res + dp[2] = 4,记录根节点值为pre=1;
6-4 & 2^1 = 10&10=10,说明有右子树,因为pre=1,说明左子树的右子树满足101,所以只要加上res + dp[1-1] = res + dp[0] = 5,更新 preOfPre = pre = 1,pre = 1;
2-2 & 2^0 = 0 & 0 == 0,说明没有右子节点,且已经到底了,需要再加1,最终结果
4+1+1 = 6(包含0)

package com.codernav.demo.hwod.exam;

import java.util.Scanner;

/**
 * @title 不含101的数
 * @Description 小明在学习二进制时,发现了一类不含101的数,也就是:
 * 将数字用二进制表示,不能出现101。
 * 现在给定一个正整数区间[l,r],请问这个区间内包含了多少个不含101的数?
 * @Author 开发者导航
 * @website https://codernav.com
 * @date 2023/5/14
 */
public class Main {
    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);
        int l = sc.nextInt();
        int r = sc.nextInt();
        //因为右边界也包含在内,所以右侧需要-1
        int res = handle(r) - handle(l - 1);
        System.out.println(res);
    }
    public static int handle(int num) {
        //10^9=1,000,000,000;2^30=1,073,741,824;数组长度设定为30
        int[] dp = new int[31];
        dp[0] = 1;  //0
        dp[1] = 2;  //0,1
        dp[2] = 4;  //00,01,10,11
        for (int i = 3; i < 31; i++) {
            dp[i] = dp[i - 1] + dp[i - 3] + dp[i - 1] - dp[i - 2];
        }
        // preOfPre 记录上上一层的根节点值
        // pre 记录上一层的根节点值
        // res 记录最终路径数
        int preOfPre = 0, pre = 0, count = 0;
        for (int i = 29; i >= 0; i--) {
            //位运算,用来判断是否包含右子树
            int val = 1 << i;
            //判断当前子树是否有右子树
            if ((num & val) != 0) {
                // 有右子树
                num -= val;
                if (pre == 1) {
                    //上节点为1则说明左子树(满二叉树)的右子树不符合要求
                    //只能将左子树的左子树dp[i-1](满二叉树)加到结果中
                    count += i == 0 ? 1 : dp[i - 1];
                } else {
                    count += dp[i]; // 先将左子树dp[i](满二叉树)加到结果中
                }

                //出现了“101”的情况说明下面都可以忽略了
                if (preOfPre == 1 && pre == 0) {
                    break;
                }
                //更新上上节点值,标记当前根节点为 1
                preOfPre = pre;
                pre = 1;
            } else {
                // 无右子树,此时不能保证左子树是否为满二叉树,所以要在下一层再继续判断
                //更新上上节点值,标记当前根节点为 0
                preOfPre = pre;
                pre = 0;
            }

            //不能忘记最后一层的最后一个数
            if (i == 0) {
                count += 1;
            }
        }
        return count;
    }
}
© 版权声明
开发者导航

相关文章

开发者导航

暂无评论

暂无评论...