数学家高斯很小的时候就天分过人。一次老师指定的算数题目是:1+2+...+100。
高斯立即做出答案:5050!
这次你的任务是类似的。但并非是把一个个的数字加起来,而是对该数字的每一个数位作累加。
这样从1加到100的“和”是:901
从10加到15是:21,也就是:1+0+1+1+1+2+1+3+1+4+1+5,这个口算都可以出结果的。
按这样的“加法”,从1加到1000是多少呢? 请通过浏览器提交该结果。
当然,我们并不期望你能像高斯一样,发现数字背后深奥的秘密,只要请计算机帮忙,一切都easy!
注意:你需要提交的是一个整数,不要填写任何多余的内容(比如:说明性文字)
答案:13501
import java.util.*;
public class Main {
public static int f(int num)
{
String str=String.valueOf(num);
int res=0;
for(int i=0;i<str.length();i++)
{
res+=str.charAt(i)-'0';
}
return res;
}
public static void main(String[] args) {
int count=0;
for(int i=0;i<=1000;i++)
{
count+=f(i);
}
System.out.println(count);
}
}
2.标题:数字划分
w星球的长老交给小明一个任务:
1,2,3...16 这16个数字分为两组。
要求:
这两组数字的和相同,
并且,两组数字的平方和也相同,
并且,两组数字的立方和也相同。
请你利用计算机的强大搜索能力解决这个问题。
并提交1所在的那个分组的所有数字。
这些数字要从小到大排列,两个数字间用一个空格分开。
即类似:1 4 5 8 ... 这样的答案。
注意,只提交这一组数字,不要填写任何多余的内容。
----------------------------------------
笨笨有话说:
只要一个组的成员确定了,另一个组的成员也就确定了。枚举一个组的成员就可以了。
凭直觉,两个组的成员数目不会差太多吧。
歪歪有话说:
既然求 1 所在的那个组,那只要枚举剩余的成员就可以了。
貌似都是8个成员的可能性很大啊。
思路:根据题目的暗示数组长度为8,写个排列即可。
答案:[1, 4, 6, 7, 10, 11, 13, 16]
import java.util.*;
public class Main {
static ArrayList<Integer> tmpArr = new ArrayList<>();
public static boolean test(ArrayList<Integer> a)
{
for(int k=1;k<=3;k++) {
int num1 = 0;
int num2 = 0;
for (int i = 1; i <= 16; i++) {
if(a.contains(i))
{
num1+=Math.pow(i,k);
}
else
{
num2+=Math.pow(i,k);
}
}
if (num1 != num2) {
return false;
}
}
return true;
}
public static void combine(int index,int k,int []arr) {
if(k == 1){
for (int i = index; i < arr.length; i++) {
tmpArr.add(arr[i]);
if(test(tmpArr))
System.out.print(tmpArr.toString()+" ");
tmpArr.remove((Object)arr[i]);
}
}else if(k > 1){
for (int i = index; i <= arr.length - k; i++) {
tmpArr.add(arr[i]); //tmpArr都是临时性存储一下
combine(i + 1,k - 1, arr); //索引右移,内部循环,自然排除已经选择的元素
tmpArr.remove((Object)arr[i]); //tmpArr因为是临时存储的,上一个组合找出后就该释放空间,存储下一个元素继续拼接组合了
}
}else{
return ;
}
}
public static void main(String[] args) {
int a[]={1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16};
combine(0,8,a);
}
}
3.标题:树形显示
对于分类结构可以用树形来形象地表示。比如:文件系统就是典型的例子。
树中的结点具有父子关系。我们在显示的时候,把子项向右缩进(用空格,不是tab),并添加必要的连接线,以使其层次关系更醒目。
下面的代码就是为了这个目的的,请仔细阅读源码,并填写划线部分缺少的代码。
- import java.util.*;
- class MyTree
- {
- private Map<String, List<String>> map_ch = new HashMap<String, List<String>>();
- private Map<String,String> map_pa = new HashMap<String,String>();
- public void add(String parent, String child)
- {
- map_pa.put(child, parent);
- List<String> lst = map_ch.get(parent);
- if(lst==null){
- lst = new ArrayList<String>();
- map_ch.put(parent, lst);
- }
- lst.add(child);
- }
- public String get_parent(String me){
- return map_pa.get(me);
- }
- public List<String> get_child(String me){
- return map_ch.get(me);
- }
- private String space(int n)
- {
- String s = "";
- for(int i=0; i<n; i++) s += ' ';
- return s;
- }
- private boolean last_child(String x){
- String pa = map_pa.get(x);
- if(pa==null) return true;
- List<String> lst = map_ch.get(pa);
- return lst.get(lst.size()-1).equals(x);
- }
- public void show(String x){
- String s = "+--" + x;
- String pa = x;
- while(true){
- pa = map_pa.get(pa);
- if(pa==null) break;
- s = ___________________________________ ; // 填空
- }
- System.out.println(s);
- }
- public void dfs(String x){
- show(x);
- List<String> lst = map_ch.get(x);
- if(lst==null) return;
- for(String it: lst){
- dfs(it);
- }
- }
- }
- public class TreeView
- {
- public static void main(String[] args)
- {
- MyTree tree = new MyTree();
- tree.add("root", "dog");
- tree.add("root", "cat");
- tree.add("root", "duck");
- tree.add("dog", "AAdog");
- tree.add("dog", "BBdog");
- tree.add("dog", "CCdog");
- tree.add("AAdog", "AAdog01");
- tree.add("AAdog", "AAdog02");
- tree.add("cat", "XXcat");
- tree.add("cat", "YYcat");
- tree.add("XXcat","XXcat-oo");
- tree.add("XXcat","XXcat-qq");
- tree.add("XXcat-qq", "XXcat-qq-hahah");
- tree.add("duck", "TTduck");
- tree.add("TTduck", "TTduck-001");
- tree.add("TTduck", "TTduck-002");
- tree.add("TTduck", "TTduck-003");
- tree.add("YYcat","YYcat.hello");
- tree.add("YYcat","YYcat.yes");
- tree.add("YYcat","YYcat.me");
- tree.dfs("root");
- }
- }
+--root
+--dog
| +--AAdog
| | +--AAdog01
| | +--AAdog02
| +--BBdog
| +--CCdog
+--cat
| +--XXcat
| | +--XXcat-oo
| | +--XXcat-qq
| | +--XXcat-qq-hahah
| +--YYcat
| +--YYcat.hello
| +--YYcat.yes
| +--YYcat.me
+--duck
+--TTduck
+--TTduck-001
+--TTduck-002
+--TTduck-003
如有平字体对齐问题,可以参见图【p1.png】
注意,只填写划线部分缺少的代码,不要抄写已有的代码或符号。
答案:
- s=last_child(pa)? " " + s: "| " + s;
思路:一看show函数就想到是问怎么打印,先试下s="",发现逐行打印,那么问题就很简单了,只要在根节点前面加“| ",子节点加" "就行了。
我们知道,整数做除法时,有时得到有限小数,有时得到无限循环小数。
如果我们把有限小数的末尾加上无限多个0,它们就有了统一的形式。
本题的任务是:在上面的约定下,求整数除法小数点后的第n位开始的3位数。
输入:
一行三个整数:a b n,用空格分开。a是被除数,b是除数,n是所求的小数后位置(0<a,b,n<1000000000)
输出:
一行3位数字,表示:a除以b,小数后第n位开始的3位数字。
比如:
输入:
1 8 1
程序应该输出:
125
再比如:
输入:
1 8 3
程序应该输出:
500
再比如:
输入:
282866 999000 6
程序应该输出:
914
资源约定:
峰值内存消耗 < 256M
CPU消耗 < 1000ms
请严格按要求输出,不要画蛇添足地打印类似:“请您输入...” 的多余内容。
所有代码放在同一个源文件中,调试通过后,拷贝提交该源码。
注意:不要使用package语句。不要使用jdk1.7及以上版本的特性。
注意:主类的名字必须是:Main,否则按无效代码处理。
思路:难点在于判断循环小数段
import java.util.*;
public class Main {
public static void main(String[] args) {
Scanner in=new Scanner(System.in);
double a,b,n;
double sa,sn,count;
a=in.nextInt();
b=in.nextInt();
n=in.nextInt();
sn = n;
sa = a % b; // 此时的sa*10对b取余后得到小数点后第一位
count = 0;
while(sn>0)
{
if(sa == b) //取余之后会等于零
break;
if(sa < b)
{
sa = sa * 10;
}
else
{
//除法法则,逐步运算
sa = sa % b;
sa = sa * 10;
if(sa==0)
break; //后面都是零则直接跳出循环
}
count++;
if(sa % b == a % b) // 减掉循环的数
{
sn = n % count;
}
sn--;
}
if(sa==0)
{
System.out.print("000");
}
else
{
int i = 3;
while(i>0)
{
System.out.print((int)(sa / b));//逐步输出n后三位的每一位
sa = sa % b;
sa = sa * 10;
i--;
}
}
}
}
5.标题:分考场
n个人参加某项特殊考试。
为了公平,要求任何两个认识的人不能分在同一个考场。
求是少需要分几个考场才能满足条件。
输入格式:
第一行,一个整数n(1<n<100),表示参加考试的人数。
第二行,一个整数m,表示接下来有m行数据
以下m行每行的格式为:两个整数a,b,用空格分开 (1<=a,b<=n) 表示第a个人与第b个人认识。
输出格式:
一行一个整数,表示最少分几个考场。
例如:
输入:
5
8
1 2
1 3
1 4
2 3
2 4
2 5
3 4
4 5
程序应该输出:
4
再比如:
输入:
5
10
1 2
1 3
1 4
1 5
2 3
2 4
2 5
3 4
3 5
4 5
则程序应该输出:
5
资源约定:
峰值内存消耗 < 256M
CPU消耗 < 1000ms
请严格按要求输出,不要画蛇添足地打印类似:“请您输入...” 的多余内容。
所有代码放在同一个源文件中,调试通过后,拷贝提交该源码。
注意:不要使用package语句。不要使用jdk1.7及以上版本的特性。
注意:主类的名字必须是:Main,否则按无效代码处理。
思路:如果某个人与教室里任意人认识,则寻找下一个教室,如果不存在能进入的教室则新开一个教室。
import java.util.*;
public class Main {
static ArrayList<ArrayList<Integer>> list = null;
static HashSet<Integer>[] map;
public static void main(String[] args) {
Scanner input = new Scanner(System.in);
int n = input.nextInt();
int m = input.nextInt();
map = new HashSet[n + 1];
for (int i = 0; i < m; i++) {
int key = input.nextInt();
int value = input.nextInt();
if (map[key] == null) {
map[key] = new HashSet<Integer>();
}
if (map[value] == null) {
map[value] = new HashSet<Integer>();
}
map[key].add(value);
map[value].add(key);
}
list = new ArrayList<ArrayList<Integer>>();
for (int i = 1; i <= n; i++) {
if (list.size() == 0) {
ArrayList<Integer> child = new ArrayList<Integer>();
child.add(i);
list.add(child);
} else {
boolean find = false;
for (int j = 0; j < list.size(); j++) {
if (isBo(j, i)) {
list.get(j).add(i);
find = true;
break;
}
}
if (find == false) {// 没找到哦
ArrayList<Integer> child = new ArrayList<Integer>();
child.add(i);
list.add(child);
}
}
}
System.out.println(list.size());
}
public static boolean isBo(int school, int num) {
ArrayList<Integer> child = list.get(school);
HashSet<Integer> set = map[num];
if (set == null)
return true;
for (int i = 0; i < child.size(); i++) {
Integer stu = child.get(i);
if (set.contains(stu)) {
return false;
}
}
return true;
}
}
6.标题:合根植物
w星球的一个种植园,被分成 m * n 个小格子(东西方向m行,南北方向n列)。每个格子里种了一株合根植物。
这种植物有个特点,它的根可能会沿着南北或东西方向伸展,从而与另一个格子的植物合成为一体。
如果我们告诉你哪些小格子间出现了连根现象,你能说出这个园中一共有多少株合根植物吗?
输入格式:
第一行,两个整数m,n,用空格分开,表示格子的行数、列数(1<m,n<1000)。
接下来一行,一个整数k,表示下面还有k行数据(0<k<100000)
接下来k行,第行两个整数a,b,表示编号为a的小格子和编号为b的小格子合根了。
格子的编号一行一行,从上到下,从左到右编号。
比如:5 * 4 的小格子,编号:
1 2 3 4
5 6 7 8
9 10 11 12
13 14 15 16
17 18 19 20
样例输入:
5 4
16
2 3
1 5
5 9
4 8
7 8
9 10
10 11
11 12
10 14
12 16
14 18
17 18
15 19
19 20
9 13
13 17
样例输出:
5
其合根情况参考图[p1.png]
资源约定:
峰值内存消耗 < 256M
CPU消耗 < 2000ms
请严格按要求输出,不要画蛇添足地打印类似:“请您输入...” 的多余内容。
所有代码放在同一个源文件中,调试通过后,拷贝提交该源码。
注意:不要使用package语句。不要使用jdk1.7及以上版本的特性。
注意:主类的名字必须是:Main,否则按无效代码处理。
思路:并查集
import java.util.HashSet;
import java.util.Scanner;
import java.util.Set;
class Main{
static int []head;
public static void main(String[] args) {
//并查集
Scanner s=new Scanner(System.in);
int m=s.nextInt();
int n=s.nextInt();
int k=s.nextInt();
head=new int [m*n];
for(int i=0;i<m*n;i++)
head[i]=i;
int a,b;
for(int i=0;i<k;i++)
{
a=s.nextInt()-1;
b=s.nextInt()-1;
u(a,b);
}
Set<Integer> set=new HashSet<Integer>();
for(int i=0;i<m*n;i++)
{
set.add(f(i));
}
System.out.println(set.size());
}
private static void u(int a, int b) {
if(f(a)==f(b))
return;
head[f(a)]=f(b);
}
private static int f(int i)
{
if(head[i]==i)
return i;
return head[i]=f(head[i]);
}
}