第十二届蓝桥杯省赛JavaC组【第一场】真题——详细答案对照(完整版)

举报
红目香薰 发表于 2022/03/24 16:23:11 2022/03/24
【摘要】 ​ 目录A :ASCB: 空间试题 C: 卡片D: 相乘E: 路径F: 时间显示G: 最少砝码H: 杨辉三角形试题 I: 左孩子右兄弟试题 J: 双向排序A :ASC本题总分:5 分【问题描述】已知大写字母 A 的 ASCII 码为 65,请问大写字母 L 的 ASCII 码是多少?76代码实现public static void main(String[] args) { c...

 

目录


A :ASC

B: 空间

试题 C: 卡片

D: 相乘

E: 路径

F: 时间显示

G: 最少砝码

H: 杨辉三角形

试题 I: 左孩子右兄弟

试题 J: 双向排序



A :ASC

本题总分:5 分

【问题描述】已知大写字母 A 的 ASCII 码为 65,请问大写字母 L 的 ASCII 码是多少?
76
代码实现

public static void main(String[] args) {
        char a='L';
        System.out.println((int)a);

    }

B: 空间

本题总分:5 分

【问题描述】

小蓝准备用 256MB 的内存空间开一个数组,数组的每个元素都是 32 位二进制整数,如果不考虑程序占用的空间和维护内存需要的辅助空间,请问256MB 的空间可以存储多少个 32 位二进制整数?

代码实现

 //8Bit(比特)等于1B(字节)
	//1024B(字节)等于1KB
	//1024KB等于1MB
	//1MB=1024kb
	public static void main(String[] args) {
		//1kb=1024b
		//1b=8byte
		//1位即是1byte
		long Mb=256;
		long Kb=1024;
		long b=1024;
		long Byte=8;
		System.out.println(Mb*Kb*b*Byte/32);
	}

试题 C: 卡片

本题总分:10 分

【问题描述】

小蓝有很多数字卡片,每张卡片上都是数字 0 到 9。
小蓝准备用这些卡片来拼一些数,他想从 1 开始拼出正整数,每拼一个,就保存起来,卡片就不能用来拼其它数了。
小蓝想知道自己能从 1 拼到多少。
例如,当小蓝有 30 张卡片,其中 0 到 9 各 3 张,则小蓝可以拼出 1 到 10,
但是拼 11 时卡片 1 已经只有一张了,不够拼出 11。
现在小蓝手里有 0 到 9 的卡片各 2021 张,共 20210 张,请问小蓝可以从 1
拼到多少?
提示:建议使用计算机编程解决问题。

代码实现

public static void main(String[] args) {
    int sum=2021;
    int a1=0;
    for (int i = 1; i <= 20210; i++) {
        a1=i;
        while(a1>0) {
            if(a1%10==1&&sum>0) {
                sum--;    
            }
            a1/=10;
        }
        if(sum==0) {
            System.out.println(i);
            break;
        }
    }
}

D: 相乘

本题总分:10 分

【问题描述】

小蓝发现,他将 1 至 1000000007 之间的不同的数与 2021 相乘后再求除以1000000007 的余数,会得到不同的数。
小蓝想知道,能不能在 1 至 1000000007 之间找到一个数,与 2021 相乘后再除以 1000000007 后的余数为 99999999。如果存在,请在答案中提交这个数;如果不存在,请在答案中提交 0。
448955022


代码实现

public static void main(String[] args) {
	int a1=1000000007;
	int sum=0;
	for (int i = 1; i <= a1; i++) {
		if((i*2021)%a1==99999999) {
			sum=i;
		}
	}
	System.out.println(sum);

}

E: 路径

本题总分:15 分

【问题描述】

小蓝学习了最短路径之后特别高兴,他定义了一个特别的图,希望找到图
中的最短路径。
小蓝的图由 2021 个结点组成,依次编号 1 至 2021。
对于两个不同的结点 a, b,如果 a 和 b 的差的绝对值大于 21,则两个结点之间没有边相连;如果 a 和 b 的差的绝对值小于等于 21,则两个点之间有一条长度为 a 和 b 的最小公倍数的无向边相连。
例如:结点 1 和结点 23 之间没有边相连;结点 3 和结点 24 之间有一条无向边,长度为 24;结点 15 和结点 25 之间有一条无向边,长度为 75。
请计算,结点 1 和结点 2021 之间的最短路径长度是多少。
提示:建议使用计算机编程解决问题。
代码实现

public static void main(String[] args) {
        int[] r = new int[2022];
        for(int i = 2; i < 2022; i ++){
            r[i] = findMinRoute(i, r);
        }
        System.out.println(r[2021]);
    }
    static int findMinRoute(int pos, int[] r){
        int minRoute = Integer.MAX_VALUE;
        for(int i = pos - 1; pos - i <=21 && i > 0; i--){
            int len = lcm(pos, i);
            minRoute = Math.min(minRoute, len + r[i]);
        }
        return minRoute;
    }
    static int gcd(int a, int b) {//最大公约数
        return b == 0 ? a : gcd(b, a % b);
    }

    static int lcm(int a, int b) {//最小公倍数
        return a * b / gcd(a, b);
    }

F: 时间显示

时间限制: 1.0s 内存限制: 512.0MB 本题总分:15 分

【问题描述】

小蓝要和朋友合作开发一个时间显示的网站。在服务器上,朋友已经获取了当前的时间,用一个整数表示,值为从 1970 年 1 月 1 日 00:00:00 到当前时刻经过的毫秒数。
现在,小蓝要在客户端显示出这个时间。小蓝不用显示出年月日,只需要显示出时分秒即可,毫秒也不用显示,直接舍去即可。
给定一个用整数表示的时间,请将这个时间对应的时分秒输出。

【输入格式】

输入一行包含一个整数,表示时间。

【输出格式】

输出时分秒表示的当前时间,格式形如 HH:MM:SS,其中 HH 表示时,值为 0 到 23,MM 表示分,值为 0 到 59,SS 表示秒,值为 0 到 59。时、分、秒 不足两位时补前导 0。

【样例输入 1】

46800999

【样例输出 1】

13:00:00

【样例输入 2】

1618708103123

【样例输出 2】

01:08:23

【评测用例规模与约定】

对于所有评测用例,给定的时间为不超过 10^18 的正整数。
代码实现

public static void main(String[] args) throws ParseException {
    Scanner sc=new Scanner(System.in);
    long ms=sc.nextLong();
    long m,s,h;
    ms=ms/1000;
    s=ms%60;
    ms/=60;
    m=ms%60;
    ms/=60;
    h=ms;
    if(h>24) {
        h=h%24;
    }
    DateFormat dt=new SimpleDateFormat("HH:mm:ss");
    String aa=h+":"+m+":"+s;
    System.out.println(dt.format(dt.parse(aa)));

}

G: 最少砝码

时间限制: 1.0s 内存限制: 512.0MB 本题总分:20 分

【问题描述】

你有一架天平。现在你要设计一套砝码,使得利用这些砝码可以称出任意小于等于 N 的正整数重量。
那么这套砝码最少需要包含多少个砝码?
注意砝码可以放在天平两边。

【输入格式】

输入包含一个正整数 N。

【输出格式】

输出一个整数代表答案。

【样例输入】

7

【样例输出】

3

【样例说明】

3 个砝码重量是 1、4、6,可以称出 1 至 7 的所有重量。
1 = 1;
2 = 6 − 4 (天平一边放 6,另一边放 4);
3 = 4 − 1;
4 = 4;

5 = 6 − 1;
6 = 6;
7 = 1 + 6;
少于 3 个砝码不可能称出 1 至 7 的所有重量。

【评测用例规模与约定】

对于所有评测用例,1 ≤ N ≤ 1000000000。
代码实现

public static void main(String[] args) {
    Scanner sc = new Scanner(System.in);
    int n = sc.nextInt();
    sc.close();
    int weight = 1;
    int count = 1;
    int total = 1;
    while (total < n) {
        count++;
        weight *= 3;
        total += weight;
    }
    System.out.println(count);
}

H: 杨辉三角形

时间限制: 5.0s 内存限制: 512.0MB 本题总分:20 分

【问题描述】

(图8)图形是著名的杨辉三角形:
如果我们按从上到下、从左到右的顺序把所有数排成一列,可以得到如下
数列:
1, 1, 1, 1, 2, 1, 1, 3, 3, 1, 1, 4, 6, 4, 1, …
给定一个正整数 N,请你输出数列中第一次出现 N 是在第几个数?

【输入格式】

输入一个整数 N。

【输出格式】

输出一个整数代表答案。

【样例输入】

6

【样例输出】

13

【评测用例规模与约定】

对于 20% 的评测用例,1 ≤ N ≤ 10;
对于所有评测用例,1 ≤ N ≤ 1000000000。


代码实现

public static void main(String[] args) {
    Scanner sc=new Scanner(System.in);
    int n=sc.nextInt();
    int[][] arr=new int[n][n];
    arr[0][0]=1;
    int sum=0;
    for (int i = 1; i < arr.length; i++) {
        for (int j = 1; j < arr[0].length; j++) {
            arr[i][j]=arr[i-1][j]+arr[i-1][j-1];
            if(arr[i][j]!=0) {
                sum++;
            }
            if(arr[i][j]==n) {
                System.out.println(sum);
                return;
            }
        }
    }
}

试题 I: 左孩子右兄弟

时间限制: 2.0s 内存限制: 512.0MB 本题总分:25 分

【问题描述】

对于一棵多叉树,我们可以通过 “左孩子右兄弟” 表示法,将其转化成一棵二叉树。
如果我们认为每个结点的子结点是无序的,那么得到的二叉树可能不唯一。
换句话说,每个结点可以选任意子结点作为左孩子,并按任意顺序连接右兄弟。
给定一棵包含 N 个结点的多叉树,结点从 1 至 N 编号,其中 1 号结点是根,每个结点的父结点的编号比自己的编号小。请你计算其通过 “左孩子右兄弟” 表示法转化成的二叉树,高度最高是多少。注:只有根结点这一个结点的树高度为 0 。
例如如下的多叉树:
(图9-1)
可能有以下 3 种 (这里只列出 3 种,并不是全部) 不同的 “左孩子右兄弟”
表示:(图9-2)
其中最后一种高度最高,为 4。

【输入格式】

输入的第一行包含一个整数 N。
以下 N −1 行,每行包含一个整数,依次表示 2 至 N 号结点的父结点编号。

【输出格式】

输出一个整数表示答案。

【样例输入】

5
1
1
1
2

【样例输出】

4

【评测用例规模与约定】

对于 30% 的评测用例,1 ≤ N ≤ 20;
对于所有评测用例,1 ≤ N ≤ 100000。


代码实现

static Map<Integer,List<Integer>> map=new HashMap<>();
    public static void main(String[] args) {
        Scanner sc=new Scanner(System.in);
        int n=sc.nextInt();
        for (int i = 2; i <= n; i++) {
            int a=sc.nextInt();
            //判断一个键在集合里面是否存在,如果不存在就将这个a添加到集合中
            map.putIfAbsent(a, new ArrayList<>());
            map.get(a).add(i);    
        }
        int max=dsf(1);
        System.out.println(max);
    }
    public static int dsf(int x) {
        int sum=0,max=0;
        if(!map.containsKey(x)) {
            return 0;
        }else {
            List<Integer> aa=map.get(x);
            sum=aa.size();
            for (Integer item:aa) {
                max=Math.max(dsf(item), max);
            }
            return sum+max;
        }
    }

试题 J: 双向排序

时间限制: 5.0s 内存限制: 512.0MB 本题总分:25 分

【问题描述】

给定序列 (a1, a2, · · · , an) = (1, 2, · · · , n),即 ai = i。
小蓝将对这个序列进行 m 次操作,每次可能是将 a1, a2, · · · , aqi 降序排列,或者将 aqi, aqi+1, · · · , an 升序排列。
请求出操作完成后的序列。

【输入格式】

输入的第一行包含两个整数 n, m,分别表示序列的长度和操作次数。
接下来 m 行描述对序列的操作,其中第 i 行包含两个整数 pi, qi 表示操作类型和参数。当 pi = 0 时,表示将 a1, a2, · · · , aqi 降序排列;当 pi = 1 时,表示将 aqi, aqi+1, · · · , an 升序排列。

【输出格式】

输出一行,包含 n 个整数,相邻的整数之间使用一个空格分隔,表示操作
完成后的序列。

【样例输入】

3 3
0 3
1 2
0 2

【样例输出】

3 1

【样例说明】

原数列为 (1, 2, 3)。
第 1 步后为 (3, 2, 1)。
第 2 步后为 (3, 1, 2)。
第 3 步后为 (3, 1, 2)。与第 2 步操作后相同,因为前两个数已经是降序了。

【评测用例规模与约定】

对于 30% 的评测用例,n, m ≤ 1000;
对于 60% 的评测用例,n, m ≤ 5000;
对于所有评测用例,1 ≤ n, m ≤ 100000,0 ≤ ai ≤ 1,1 ≤ bi ≤ n
代码实现

public static void main(String[] args) {
    Scanner sc=new Scanner(System.in);
    int n=sc.nextInt();
    int k=sc.nextInt();
    int[] arr=new int[n+1];
    String[] str=new String[k];
    for (int i = 1; i <= n; i++) {
        arr[i]=i;
    }
    sc.nextLine();
    for (int i = 0; i < str.length; i++) {
        str[i]=sc.nextLine();
    }
    for (int i = 0; i < k; i++) {
        int s1=Integer.parseInt(String.valueOf((char)str[i].charAt(0)));
        int s2=Integer.parseInt(String.valueOf((char)str[i].charAt(2)));
        px(s1,s2,arr);
    }
    System.out.println(Arrays.toString(arr));

}
public static void px(int x,int y,int[] arr) {
    if(x==0) {
        for (int i = 1; i <= y; i++) {
            for (int j = i+1; j <=y; j++) {
                if(arr[i]<arr[j]) {
                    int a=arr[j];
                    arr[j]=arr[i];
                    arr[i]=a;
                }
            }
        }
    }else {
        for (int i = y; i < arr.length; i++) {
            for (int j = i+1; j <arr.length; j++) {
                if(arr[i]>arr[j]) {
                    int a=arr[j];
                    arr[j]=arr[i];
                    arr[i]=a;
                }
            }
        }
    }
}

【版权声明】本文为华为云社区用户原创内容,转载时必须标注文章的来源(华为云社区)、文章链接、文章作者等基本信息, 否则作者和本社区有权追究责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。