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

news2025/1/6 4:26:55

目录

【案例1】

【题目描述】

 【思路解析】

【代码实现】

【案例2】

【题目描述】

 【思路解析】

【代码实现】

【案例3】

【题目描述】

【思路解析】

【案例4】

【题目描述】

【思路解析】

 【代码实现】

【案例5】

【题目描述】

【子序列概念】

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

【代码实现1】

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

【代码实现2】

【案例6】

【题目描述】

 【思路解析】

【代码实现】


【案例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】

【题目描述】

 【思路解析】

我们已知先序数组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】

【题目描述】

【思路解析】

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

 【代码实现】

/**
 * @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】

【题目描述】

 【思路解析】

如果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);
    }

}

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/812214.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

CTF-MISC:BUUCTF练习汇总(持续更新)

CTF-MISC&#xff1a;BUUCTF练习汇总 1、金三胖2、你竟然赶我走3、二维码4、大白 1、金三胖 解题思路&#xff1a; flag隐藏在gif图片帧中&#xff0c;使用在线GIF拆分工具即可 在线GIF图片帧拆分工具&#xff1a;https://uutool.cn/gif2img/ flag为&#xff1a;flag{he11oho…

应用层协议——https

文章目录 1. HTTPS 是什么2. 什么是"加密"3. 常见的加密方式4. 数据摘要 && 数字签名5. HTTPS 的工作过程探究5.1 方案1 - 只使用对称加密5.2 方案2 - 只使用非对称加密5.3 方案3 - 双方都使用非对称加密5.4 方案4 - 非对称加密 对称加密5.5 中间人攻击5.6 …

【MTI 6.S081 Lab】locks

【MTI 6.S081 Lab】locks Memory allocator (moderate)实验任务Hint解决方案 Buffer cache (hard)实验任务Hint解决方案数据结构设计初始化数据结构getrelse 本实验前去看《操作系统导论》第29章基于锁的并发数据结构&#xff0c;将会是很有帮助的。 在这个实验室中&#xff0…

懒得改变原始对象?JavaScript代理模式教你怎样一步步偷懒!

前言 系列首发gong zhong hao[『非同质前端札记』] &#xff0c;若不想错过更多精彩内容&#xff0c;请“星标”一下&#xff0c;敬请关注gong zhong hao最新消息。 懒得改变原始对象&#xff1f;JavaScript代理模式教你怎样一步步偷懒&#xff01; 何为代理模式 例如&#x…

倍增与ST算法

倍增与ST算法 倍增倍增原理倍增法的局限例题 &#xff1a;国旗计划 (洛谷 P4155)例题题解带注释的代码 ST算法ST算法原理ST算法步骤ST算法应用场合例题 &#xff1a;【模板】ST表 (洛谷 P3865) 倍增 倍增原理 倍增法的局限 例题 &#xff1a;国旗计划 (洛谷 P4155) 例题题解 带…

华为OD机试真题 Java 实现【报文回路】【2023 B卷 100分】,俗称“礼尚往来”

目录 专栏导读一、题目描述二、输入描述三、输出描述四、解题思路1、报文回路2、异常情况&#xff1a;3、解题思路 五、Java算法源码六、效果展示1、输入2、输出 华为OD机试 2023B卷题库疯狂收录中&#xff0c;刷题点这里 专栏导读 本专栏收录于《华为OD机试&#xff08;JAVA&…

《JavaSE-第二十章》之线程的创建与Thread类

文章目录 什么是进程&#xff1f;什么是线程&#xff1f;为什么需要线程&#xff1f; 基本的线程机制创建线程1.实现 Runnable 接口2.继承 Thread 类3.其他变形 Thread常见构造方法1. Thread()2. Thread(Runnable target)3. Thread(String name)4. Thread(Runnable target, Str…

epoll复用

cli #include <stdlib.h> #include <stdio.h> #include <string.h> #include <unistd.h> #include <arpa/inet.h> #include <sys/socket.h>// 服务器ip #define IP "192.168.250.100" // 服务器端口 #define PORT 8888int main…

c++11 标准模板(STL)(std::basic_ifstream)(一)

定义于头文件 <fstream> template< class CharT, class Traits std::char_traits<CharT> > class basic_ifstream : public std::basic_istream<CharT, Traits> 类模板 basic_ifstream 实现文件流上的高层输入操作。它将 std::basic_istream…

Flink - souce算子

水善利万物而不争&#xff0c;处众人之所恶&#xff0c;故几于道&#x1f4a6; 目录 1. 从Java的集合中读取数据 2. 从本地文件中读取数据 3. 从HDFS中读取数据 4. 从Socket中读取数据 5. 从Kafka中读取数据 6. 自定义Source 官方文档 - Flink1.13 1. 从Java的集合中读取数据 …

二叉树(C语言)

文章目录 1.树1.1概念1.2相关定义1.3 表示&#xff08;左孩子右兄弟&#xff09; 2.二叉树2.1概念2.2特殊的二叉树1. 满二叉树&#xff1a;2. 完全二叉树&#xff1a; 2.3二叉树的性质2.4练习 3.二叉树的存储结构1. 顺序存储2. 链式存储 4.完全二叉树的代码实现4.1堆的介绍1.堆…

ssm德宏贸易项目java人资企业办公jsp源代码mysql

本项目为前几天收费帮学妹做的一个项目&#xff0c;Java EE JSP项目&#xff0c;在工作环境中基本使用不到&#xff0c;但是很多学校把这个当作编程入门的项目来做&#xff0c;故分享出本项目供初学者参考。 一、项目描述 ssm德宏贸易项目 系统有1权限&#xff1a;管理员 二…

接口自动化测试平台

下载了大神的EasyTest项目demo修改了下<https://testerhome.com/topics/12648 原地址>。也有看另一位大神的HttpRunnerManager<https://github.com/HttpRunner/HttpRunnerManager 原地址>&#xff0c;由于水平有限&#xff0c;感觉有点复杂~~~ 【整整200集】超超超…

查询结果元数据-MetaData对象、数据库工具类的封装、通过反射实现数据查询的封装

六、查询结果元数据-MetaData对象 七、数据库工具类的封装 1、PropertieUtil类 2、DbUtil类 3、DBHepler类 查询&#xff1a; 4、TestDb测试类&#xff1a; 更新&#xff1a; 1&#xff09;插入&#xff1a; 2&#xff09;修改&#xff1a; 3&#xff09;删除&#xff1a; 查…

2024考研408-计算机网络 第二章-物理层学习笔记

文章目录 前言一、通信基础1.1、物理层基本概念1.1.1、认识物理层1.1.2、认识物理层的四种接口特性 1.2、数据通信基础知识1.2.1、典型的数据通信模型及相关术语1.2.2、数据通信相关术语1.2.3、设计数据通信系统要考虑的三个问题&#xff1a;问题1&#xff1a;采用单工通信/半双…

通讯录的实现(超详细)——C语言(进阶)

目录 一、创建联系人信息&#xff08;结构体&#xff09; 二、创建通讯录&#xff08;结构体&#xff09; 三、define定义常量 四、打印通讯录菜单 五、枚举菜单选项 六、初始化通讯录 七、实现通讯的的功能 7.1 增加加联系人 7.2 显示所有联系人的信息 ​7.3 单独查…

【自动化运维】Ansible常见模块的运用

目录 一、Ansible简介二、Ansible安装部署2.1环境准备 三、ansible 命令行模块3.1&#xff0e;command 模块3.2&#xff0e;shell 模块3.3&#xff0e;cron 模块3.4&#xff0e;user 模块3.5&#xff0e;group 模块3.6&#xff0e;copy 模块3.7&#xff0e;file 模块8&#xff…

C++之观察者模式(发布-订阅)

目录 模式简介 介绍 优点 缺点 代码实现 场景说明 实现代码 运行结果 模式简介 观察者模式&#xff08;Observer Pattern&#xff09;&#xff0c;也叫我们熟知的发布-订阅模式。 它是一种行为型模式。 介绍 观察者模式主要关注的是对象的一对多的关系&#xff0c; …

4-3 Working with time series

本文所用数据下载 Data from a Washington, D.C., bike-sharing system reporting the hourly count of rental bikes in 2011–2012 in the Capital Bikeshare system, along with weather and seasonal information. Our goal will be to take a flat, 2D dataset and trans…

搭建网站 --- 快速WordPress个人博客并内网穿透发布到互联网

文章目录 快速WordPress个人博客并内网穿透发布到互联网 快速WordPress个人博客并内网穿透发布到互联网 我们能够通过cpolar完整的搭建起一个属于自己的网站&#xff0c;并且通过cpolar建立的数据隧道&#xff0c;从而让我们存放在本地电脑上的网站&#xff0c;能够为公众互联…