【题目描述】
已知大写字母 A 的 ASCII 码为 65,请问大写字母 L 的 ASCII 码是多少?
【答案】:76
【题解】简单的acsll计算,按照字母表,A为第1个,L为第12个,故L的ASCll码为65+11=76.
【题目描述】
小蓝有很多数字卡片,每张卡片上都是数字0 到9。
小蓝准备用这些卡片来拼一些数,他想从1 开始拼出正整数,每拼一个,就保存起来,卡片就不能用来拼其它数了。
小蓝想知道自己能从1 拼到多少。
例如,当小蓝有30 张卡片,其中0 到9 各3 张,则小蓝可以拼出1 到10,但是拼11 时卡片1 已经只有一张了,不够拼出11。
现在小蓝手里有0 到9 的卡片各2021 张,共20210 张,请问小蓝可以从1拼到多少?
提示:建议使用计算机编程解决问题。
【答案】:3181
【题解】:0-9的卡片每个数字有2021张,从1开始遍历,每遍历到一个数字,这个卡片数字的个数就-1,如果某一个数字的卡片没有了,就遍历结束了。
import java.util.Arrays;
public class Main {
static int a[] = new int[10];//0-9
public static void main(String[] args) {
Arrays.fill(a, 2021);//每个数初始值为2021
int ans = 0;
for(int i = 1; i <= 20210; i++) {
int t = i;
while(t > 0) {
int s = t % 10;
t /= 10;
a[s]--;
//注意:这里不是==0,上面a[s]--,减了一次才为0,说明这次没减之前为1,还有最后一个数,
//如果为0,-1就会小于0,说明这个数2021张牌都用完啦
if(a[s] < 0) {
ans = i-1; //当前遍历的这个数肯定没有遍历完,故遍历到这个数的前一个
break;
}
}
if(ans != 0) break;
}
System.out.println(ans);
}
}
【题目描述】
在平面直角坐标系中,两点可以确定一条直线。
如果有多点在一条直线上,那么这些点中任意两点确定的直线是同一条。
给定平面上2 × 3 个整点{(x, y)|0 ≤ x < 2, 0 ≤ y < 3, x ∈ Z, y ∈ Z},
即横坐标是0 到1 (包含0 和1) 之间的整数、纵坐标是0 到2 (包含0 和2) 之间的整数的点。
这些点一共确定了11 条不同的直线。
给定平面上20 × 21 个整点{(x, y)|0 ≤ x < 20, 0 ≤ y < 21, x ∈ Z, y ∈ Z},
即横坐标是0 到19 (包含0 和19) 之间的整数、纵坐标是0 到20 (包含0 和20) 之间的整数的点。
请问这些点一共确定了多少条不同的直线。
【答案】:40257
【题解】这题的做法是先保存所有直线的斜率k和截距b,然后去重就可以得到结果。
其中最重要的是找到每一条直线的k和b(y=kx+b),然后要去重,因为每一条直线上可能有多个节点,遍历的时候会重复,然后在记录k和b的时候最好不要使用double,因为精度不好控制,我使用的是String储存。当直线为x=b的时候没有斜率,在题中一共有20条,在最后加上即可。
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
public class Main{
static ListsList = new ArrayList<>();
static int gcd(int a,int b) {
return b != 0 ? gcd(b, a % b) : a;
}
public static void main(String[] args) {
for(int x1 = 0; x1 < 20; x1++) {
for(int y1 = 0; y1 < 21; y1++) {
for(int x2 = 0; x2 < 20; x2++) {
for(int y2 = 0; y2 < 21; y2++) {
if(x1 != x2) {
String line = "";//存储k和b
int y = y2 - y1;
int x = x2 - x1;
int g = gcd(y,x);//求两个数的最小公倍数,记录最简分数(唯一)
line = y / g + "/" + x / g;//k = (y2 - y1) / (x2 - x1);
int b = y1 * x - x1 * y;//b =( y * (x2 - x1) - x * (y2 - y1)) / (x2 - x1)
int g1 = gcd(b, x);
line = line +"&"+ b / g1 + "/" + x / g1;
sList.add(line);
}
}
}
}
}
long ans = 0L;
Mapmap = new HashMap<>();
for(int i = 0; i < sList.size(); i++) {
if(!map.containsKey(sList.get(i))) {
ans++;
// System.out.println(sList.get(i));
map.put(sList.get(i), true);
}
}
System.out.println(ans+20);
}
}
【题目描述】
小蓝有一个超大的仓库,可以摆放很多货物。
现在,小蓝有n 箱货物要摆放在仓库,每箱货物都是规则的正方体。
小蓝规定了长、宽、高三个互相垂直的方向,每箱货物的边都必须严格平行于长、宽、高。
小蓝希望所有的货物最终摆成一个大的立方体。即在长、宽、高的方向上分别堆L、W、H 的货物,满足n = L × W × H。
给定n,请问有多少种堆放货物的方案满足要求。
例如,当n = 4 时,有以下6 种方案:1×1×4、1×2×2、1×4×1、2×1×2、2×2×1、4×1×1。
请问,当n = 2021041820210418 (注意有16 位数字)时,总共有多少种
方案?
提示:建议使用计算机编程解决问题。
【答案】:2430
【题解】这一题是问我们有n箱货物可以有多少种摆放方法,通过案例可以得知L,W,H,都是n的质因子(L* W*H=n),因此我们可以记录n的所以质因子,然后通过双重循环可以确定两个未知数,另外已知n,可以求出另一个未知数(H = n / (L * w)),题中要求算出一共多少种,所以只要满足条件(n % (L * W)) == 0即可。
import java.util.LinkedList;
import java.util.List;
public class Main {
static Listlist = new LinkedList<>();
public static void main(String[] args) {
long n = 2021041820210418L;
for(long i = 1; i <= n / i; i++) {//算出n的所有质因数
if(n % i == 0) {
list.add(i);
if(i != n / i) {
list.add(n / i);
}
}
}
long ans = 0L;
for(long a : list) {//双重循环找答案
for(long b : list) {
if(n % (a * b) == 0) ans++;
}
}
System.out.println(ans);
}
}
【题目描述】
小蓝学习了最短路径之后特别高兴,他定义了一个特别的图,希望找到图中的最短路径。
小蓝的图由2021 个结点组成,依次编号1 至2021。
对于两个不同的结点a, b,如果a 和b 的差的绝对值大于21,则两个结点之间没有边相连;
如果a 和b 的差的绝对值小于等于21,则两个点之间有一条长度为a 和b 的最小公倍数的无向边相连。
例如:结点1 和结点23 之间没有边相连;结点3 和结点24 之间有一条无向边,长度为24;
结点15 和结点25 之间有一条无向边,长度为75。
请计算,结点1 和结点2021 之间的最短路径长度是多少。
提示:建议使用计算机编程解决问题。
【答案】:10266837
【题解】这个题是最短路径的板子题,因为a和b 的绝对值不能大于21,说明a最多需要和42个点求最小公倍数来建无向边。求最短路径我用的是Dijkstra。
import java.util.Arrays;
public class Main {
static int N = 2050,n = 2021,INF = 0x3f3f3f3f;
static int g[][] = new int[N][N];
static int dist[] = new int[N];
static boolean st[] = new boolean[N];
static int gcd(int a,int b) {
return b != 0 ? gcd(b, a % b) : a;
}
static int dijkstra() {
Arrays.fill(dist, INF);
dist[1] = 0;
for(int i = 1; i < n; i++) {
int t = -1;
for(int j = 1; j <= n; j++) {
if(!st[j] && (t== -1 || dist[t] > dist[j] + g[t][j])) {
t = j;
}
}
st[t] = true;
for(int j = 1; j <= n; j++) {
dist[j] = Math.min(dist[j], dist[t] + g[t][j]);
}
}
return dist[n];
}
public static void main(String[] args) {
for(int i = 1; i < N; i++) {
Arrays.fill(g[i], INF);
}
for(int i = 1; i <= n; i++) {
for(int j = Math.max(1, i - 21); j <= Math.min(2021, i + 21); j++) {
int t = gcd(i, j);
g[i][j] = g[j][i] = (i * j) / t;
}
}
System.out.println(dijkstra());
}
}
【题目描述】
小蓝要和朋友合作开发一个时间显示的网站。
在服务器上,朋友已经获取了当前的时间,用一个整数表示。
值为从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次方的正整数。
【题解】题目是给出毫秒数,然后只需要显示时、分、秒,说明答案是在24小时之内。首先可以将输入的时间模除以一天的时间,将时间控制在24小时之内。然后分别算出时、分、秒即可。
一天:24 x 60 x 60 x 1000 (小时 x 分钟 x 秒 x 毫秒)= 86,400,000(毫秒)
以实例2为例子 1618708103123 % 86,400,000 = 4,103,123;
从毫秒转入秒:4,103,123 / 1000 = 4103(秒)
时: 4103 /(60 * 60) = 1(时);
剩余秒数:4103 %(60 * 60)= 503;
分:503 / 60 = 8(分);
剩余秒数:503 % 60 = 23(秒)
答案:01:08:23
java的补前导0可以使用printf输出。
import java.util.Scanner;
public class Main {
public static void main(String[] args) {
Scanner sc = new Scanner(System.in);
long n = sc.nextLong();
n = n % (24 * 60 * 60 * 1000);
n /= 1000;
int h = (int) (n / 3600) ;
int m = (int) ((n % 3600) / 60);
int s = (int) ((n % 3600) % 60);
System.out.printf("%02d:%02d:%02d\n",h,m,s);
sc.close();
}
}
【题目描述】
你有一架天平。现在你要设计一套砝码,使得利用这些砝码可以称出任意小于等于N的正整数重量。
那么这套砝码最少需要包含多少个砝码?
注意砝码可以放在天平两边。
【输入格式】
一行包含一个正整数N。
【输出格式】
一行输出一个表示答案。
【输入样例】
7
【输出样例】
3
【样例说明】
33 个砝码重量是 1、4、6,可以称出 1 至 7的所有重量。
1 = 1;
2 = 6 − 4(天平一边放 66,另一边放 44);
3 = 4 − 1;
4 = 4;
5 = 6 − 1;
6 = 6;
7 = 1 + 6;
少于 3 个砝码不可能称出 1 至 7 的所有重量。
【测评用例规模与约定】
对于所有评测用例,1 ≤ N ≤ 1000000000。
【题解】可以一步一步推过来,一个砝码的时候先加一个1,如果第二个砝码是2,那么最远可以拼到3(1,2,1+2),而如果第二个砝码是3的话,最远可以拼接到4(1,3-1,3,3+1)
减法代表放在天平两边,加法代表放在天平的同一边。如果第二个砝码是4的话,不能拼到数字2(1,2=?,4-1=3,4,4+1=5),所以要拼到更大的数需要第三个砝码,以此类推,我发现当第三个砝码为9时达到最大数字13(1,3-1,3,3+1,9-3-1,9-3,9+1-3,9-1,9,9+1,9+3-1,9+3,9+3+1);因此得到规律:
选择的砝码 最大的数量 1 1 3 4 9 13 ... ...
import java.util.Scanner;
public class Main {
public static void main(String[] args) {
Scanner sc = new Scanner(System.in);
int n = sc.nextInt();
int ans = 0;
while(n > 0) {
n = n - (int)Math.pow(3, ans);
ans++;
}
System.out.println(ans);
sc.close();
}
}
【题目描述】
下面的图形是著名的杨辉三角形:
如果我们按从上到下、从左到右的顺序把所有数排成一列,可以得到如下数列:
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。
【题解】我直接打暴力骗分。。。
还有两个大题不会写,听大佬说只要前面7个题都对,最后三个打暴力骗点分就可以拿到省一了,,,呀呀呀,大家都太强了,我好菜......