左神算法之中级提升班(8)

目录

【案例1】

【题目描述】

 【思路解析】

【代码实现】

【案例2】

【题目描述】

 【思路解析】

【代码实现】

【案例3】

【题目描述】

【思路解析】

【案例4】

【题目描述】

【思路解析】

 【代码实现】

【案例5】

【题目描述】

【子序列概念】

【思路解析1 经典方法 时间复杂度为O(N^2)】

【代码实现1】

【思路解析2 优化技巧之构建单调性  时间复杂度为O(N*logN)】

【代码实现2】

【案例6】

【题目描述】

 【思路解析】

【代码实现】


【案例1】

【题目描述】

左神算法之中级提升班(8)_第1张图片

 【思路解析】

一个简单的贪心策略。

对于字符串str,任意一个位置上只能为字符'.'或者'X'。则对于任意i位置有两种情况。

(1)i位置上为字符'X',此位置不需要放路灯

(2)i位置上为字符'.',此位置放灯则需要根据i+1位置来判断,如果i+1位置是‘X',则i位置放灯,如果i+1位置是’.',则i+1位置放灯。

【代码实现】

import java.util.Scanner;

/**
 * @ProjectName: study3
 * @FileName: Ex1
 * @author:HWJ
 * @Data: 2023/7/30 9:27
 */
public class Ex1 {
    public static void main(String[] args) {
        Scanner input = new Scanner(System.in);
        int t = input.nextInt();
        for (int i = 0; i < t; i++) {
            int n = input.nextInt();
            String s = input.nextLine();
            System.out.println(minLight(s, n));
        }

    }

    public static int minLight(String s, int length) {
        if (s == null || length == 0) {
            return 0;
        }
        char[] str = s.toCharArray();
        int i = 0;
        int count = 0;
        while (i < length) {
            if (str[i] == 'X') {
                i++;
            } else {
                count++;
                // 加入条件(i + 1) < length 来防止越界
                if ((i + 1) < length && str[i + 1] == 'X') {
                    i += 2;
                } else if ((i + 1) < length && str[i + 1] == '.') {
                    i += 3;
                }else { // 如果不是以上两种情况 就说明 i+1 == length,说明已经遍历完成,直接退出循环
                    break;
                }
            }
        }
        return count;
    }
}

【案例2】

【题目描述】

左神算法之中级提升班(8)_第2张图片

 【思路解析】

我们已知先序数组int[ ] pre,中序数组int[ ] in,我们需要得到后序数组int[ ] pos,先序数组和中序数组和后序数组长度相同,然后先序数组对应二叉树的顺序为头左右,中序数组对应二叉树的顺序为左头右,后序数组对应二叉树的顺序为左右头,所以我们可以通过先序数组的第一个位置确定整棵树的头,然后找出它在中序数组的位置,此位置左边就是整棵树最大的左子树,右边就是整棵树最大的右子树,然后在先序数组中头位置的右边同样多的数也是左子树,然后剩下的数为右子树,这样又可以继续递归确定子树的位置然后将其填入后序数组中。填入时,头位置填入在当前填充后序数组位置中最后一个,然后填入右子树,然后填入左子树,填入顺序为从右至左。

【代码实现】

代码中在中序数组中寻找头的位置使用的是遍历的方式,我们可以对中序数组中每个位置上的值做一个哈希表来对应,这样下次寻找头的时候直接访问哈希表即可,创建哈希表只需要做一次遍历,所以这是一个优化搜索的方法。

import java.util.Arrays;

/**
 * @ProjectName: study3
 * @FileName: Ex2
 * @author:HWJ
 * @Data: 2023/7/30 10:16
 */
public class Ex2 {
    public static void main(String[] args) {
        int[] pre = {1,2,4,5,3,6,7};
        int[] in = {4,2,5,1,6,3,7};
        int[] pos = getPosArray(pre, in);
        System.out.println(Arrays.toString(pos));

    }

    public static int[] getPosArray(int[] pre,int[] in){
        int N = pre.length;
        int[] pos = new int[N];
        process(pre, 0, N - 1, in, 0, N - 1, pos, 0, N - 1);
        return pos;
    }

    public static void process(int[] pre, int prei, int prej,
                               int[] in, int ini, int inj,
                               int[] pos, int posi, int posj){
        if (prei > prej){
            return;
        }
        pos[posj] = pre[prei];
        int find = ini;
        for (; find < inj; find++) {
            if (in[find] == pre[prei]){
                break;
            }
        }
        process(pre, prei + 1 + find - ini, prej, in, find + 1, inj, pos, posj - (find - ini), posj - 1);
        process(pre, prei + 1, prei + find - ini, in, ini, find - 1, pos, posi, posi - 1 + (find - ini));
    }
}

【案例3】

【题目描述】

将一个数字用中文表示出来,表示规则需要满足中国人的读数习惯。

经考查 15中国人习惯读十五,215习惯读二百一十五(个别省份读两百一十五),1017习惯读一千零十七,1215习惯读一千二百一十五,所以十位前读不读1,取决于有没有百位。

【思路解析】

 纯coding问题,但是有一个策略,可以先完成1-10的发言,再完成1-99,在完成1-999,后面依次累加,完成全部。

【案例4】

【题目描述】

左神算法之中级提升班(8)_第3张图片

【思路解析】

如果使用遍历的话完全能得到个数,但时间复杂度为O(N),所以我们需要利用完全二叉树的特性来进行优化,时间复杂度可以达到O((logN)^2)。

左神算法之中级提升班(8)_第4张图片

 【代码实现】

/**
 * @ProjectName: study3
 * @FileName: Ex4
 * @author:HWJ
 * @Data: 2023/7/30 11:17
 */
public class Ex4 {
    public static void main(String[] args) {

    }

    public static class Node {
        public int value;
        public Node left;
        public Node right;

        public Node(int value) {
            this.value = value;
        }
    }

    // 请保证以head为头的树是一颗完全二叉树。
    public static int nodeNum(Node head) {
        if (head == null) {
            return 0;
        }
        return bs(head, 1, mostLeftLevel(head, 1));
    }

    // level 表示node节点所在层数, h表示整棵树的深度
    public static int bs(Node node, int level, int h) {
        if (level == h) {
            return 1;
        }
        if (mostLeftLevel(node.right, level + 1) == h) {
            // 如果mostLeftLevel(node.right, level + 1) == h,即右子树的最左节点到了h层说明左子树一定是一个满二叉树。
            // 否则说明右子树是一个满二叉树。
            return (1 << (h - level)) + bs(node.right, level + 1, h);
        } else {
            return (1 << (h - level - 1)) + bs(node.left, level + 1, h);
        }

    }

    // 求出此节点在这棵树上的最左节点的层数
    // L表示node在的层数
    // 以node为节点的整棵树应该满足完全二叉树的定义
    public static int mostLeftLevel(Node node, int L) {
        while (node.left != null) {
            L++;
            node = node.left;
        }
        return L;
    }
}

【案例5】

【题目描述】

【子序列概念】

子序列是指一个序列中任意选取若干个元素(可以是相邻的元素,也可以是不相邻的元素)组成的序列。这些元素的原先相对顺序保持不变。例如,对于序列 [1, 2, 3, 4],其子序列可以是 [1, 2, 3]、[1, 4]、[2, 4]、[1, 3, 4] 等等。在算法和数据结构中,子序列广泛用于字符串匹配、最长子序列等问题的求解。

【思路解析1 经典方法 时间复杂度为O(N^2)】

生成一个与数组arr等长的数组dp,dp[i]的含义为以arr[i[为结尾的子序列最长长度为多少。这样对于任意位置i,它前面的位置已经形成了它的最长递增子序列,然后我们以i位置的数字结尾,我们只需要找到0 --- i-1位置上小于i位置数字的最优解作为倒数第二个数字即可(如果找不到这样的第二个数字,这dp[i]填充1),这样就可以完成对dp数组的填充。然后找出dp数组中最大的那个,我们便得到了最长递增子序列的长度。

【代码实现1】

/**
 * @ProjectName: study3
 * @FileName: Ex5
 * @author:HWJ
 * @Data: 2023/7/30 12:58
 */
public class Ex5 {
    public static void main(String[] args) {
        int[] arr = {6, 1, 5, 2, 7, 3, 4};
        System.out.println(getMaxSub(arr));
    }

    public static int getMaxSub(int[] arr) {
        int N = arr.length;
        int[] dp = new int[N];
        for (int i = 0; i < N; i++) {
            dp[i] = 1; //初始化为1,不管后面是否能找到,都能得到正确的dp结果
            for (int j = i - 1; j >= 0; j--) {
                if (arr[i] > arr[j]) { // 找到0 --- i-1位置上小于i位置数字的最优解作为倒数第二个数字
                    dp[i] = Math.max(dp[i], dp[j] + 1);
                }
            }
        }
        int max = dp[0];
        for (int i = 1; i < N; i++) {
            max = Math.max(max, dp[i]);
        }
        return max;
    }
}

【思路解析2 优化技巧之构建单调性  时间复杂度为O(N*logN)】

构建一个等长的ends数组,刚开始里面数据全是无效区域,然后遍历arr数组每个数字,然后在ends数字的有效区域中找到大于这个数字的最左位置,然后更新这个位置上的数字,如果找不到这个数字,则扩充有效区域,然后将这个数字更新到有效区域上,因为这个数组满足单调性,所以在ends数组遍历时可以使用二分法,使遍历的时间复杂度为O(logN)。然后确定了这个数字的在ends上的所在位置后如果该位置假设为i位置,则以它为结尾的最长递增子序列的长度为i+1。

但是如果我们只需要得到最长递增子序列的长度,而不需要知道以arr每个数字结尾的最长递增子序列的长度,我们便可以不再构造dp数组,而是将ends的有效区域长度作为最优解返回。

即代码2的返回修改为 return right + 1;  

【代码实现2】

/**
 * @ProjectName: study3
 * @FileName: Ex6
 * @author:HWJ
 * @Data: 2023/7/30 13:15
 */
public class Ex6 {
    public static void main(String[] args) {
        int[] arr = {6, 1, 5, 2, 7, 3, 4};
        System.out.println(getMaxSub(arr));
    }

    public static int getMaxSub(int[] arr) {
        int N = arr.length;
        int[] dp = new int[N];
        int[] ends = new int[N];
        int left = 0;
        int right = 0;
        ends[0] = arr[0];
        dp[0] = 1;
        for (int i = 1; i < N; i++) {
            int index = getIndex(ends, left, right, arr[i]);
            if (index == -1){
                ends[++right] = arr[i];
                dp[i] = right + 1;
            }else {
                ends[index] = arr[i];
                dp[i] = index + 1;
            }
        }
        int max = Integer.MIN_VALUE;
        for (int i = 0; i < N; i++) {
            max = Math.max(dp[i], max);
        }
        return max;
    }

    // 这里的left和right与上面的left和right同义,代表ends数组的有效区域,范围为[left,right]
    // 返回ends数组中,大于num的最左数字的索引
    // 如果返回的索引为-1,则没有找到比num大的数字,则需要扩充ends数组的有效区域,然后更新数组,否则直接根据索引进行更新
    public static int getIndex(int[] ends, int left, int right, int num){
        int index = -1;
        while(left <= right){
            int mid = (left + right) / 2;
            if (ends[mid] > num){
                right = mid - 1;
                index = mid;
            }else {
                left = mid + 1;
            }
        }
        return index;
    }
}

【案例6】

【题目描述】

左神算法之中级提升班(8)_第5张图片

 【思路解析】

如果n=13,则这个数字为12345678910111213,然后他能不能被3整除可以转化为(1+2+3+4+5+6+7+8+9+1+0+1+1+1+2+1+3)能不能被3整除,然后10能不能被3整除又可以转化为1+0能不能被3整除,所以我们可以将(1+2+3+4+5+6+7+8+9+1+0+1+1+1+2+1+3)这里的数字作同语替换改为(1+2+3+4+5+6+7+8+9+10+11+12+13),这就可以转为 ((n * (n+1))/2)%3,然后防止((n * (n+1))/2)过大,所以使用long来存放

【代码实现】

import java.util.Scanner;

/**
 * @ProjectName: study3
 * @FileName: Ex7
 * @author:HWJ
 * @Data: 2023/7/30 14:00
 */
public class Ex7 {
    public static void main(String[] args) {
        Scanner input = new Scanner(System.in);
        int l =input.nextInt();
        int r = input.nextInt();
        int count = 0;
        for (int i = l; i <= r; i++) {
            long s = ((long) (i + 1) * i) / 2;
            if (s % 3 == 0){
                count++;
            }
        }
        System.out.println(count);
    }

}

 

你可能感兴趣的:(算法,java,数据结构)