输入要考虑的基本问题
算法竞赛题目一般会有多个测试用例,采用重定向的方式把数据送给程序。
然后观察程序的输出是否和预期的结果一致。
多数情况下,这些测试用例会以文件的形式存在。这就要注意以下的问题:
- 每一行的数据有多项,其分隔符不是完全确定,比如是:一个或多个空格。
- 每一行上的数据有多少项可能不知道。一直持续到本行结束。
- 一共有多少行可能不知道,一直到文件尾。
- 最后一行可能没有回车换行符,直接遇到 EOF (文件结束标记)
- 在不同的操作系统下,换行的方式可能不同。
- 在提供字符串的时候,可能含有空格。
下面我们举一些最常见的例子。
若干行,每行一个整数
测式数据
3
5
-7
9
import java.util.*;
import java.io.*;
public class A
{
public static void main(String[] args){
Scanner scan = new Scanner(System.in);
try{
while(true){
System.out.println("-> " + scan.nextInt());
}
}
catch(Exception e){}
}
}
import java.util.*;
import java.io.*;
public class A
{
public static void main(String[] args){
Scanner scan = new Scanner(System.in);
try{
while(true){
System.out.println("-> " + scan.nextInt());
}
}
catch(Exception e){}
}
}
使用这种方式,最后一行有没有换行符都无关紧要。
甚至是最后有多个空行也不成问题。
若干行,每行两个整数,用一个或多个空格分开
测试数据
10 20
30 40
50 60
import java.util.*;
import java.io.*;
public class B
{
public static void main(String[] args){
Scanner scan = new Scanner(System.in);
try{
while(true){
System.out.println("-> " + scan.nextInt() + "," + scan.nextInt());
}
}
catch(Exception e){}
}
}
import java.util.*;
import java.io.*;
public class B
{
public static void main(String[] args){
Scanner scan = new Scanner(System.in);
try{
while(true){
System.out.println("-> " + scan.nextInt() + "," + scan.nextInt());
}
}
catch(Exception e){}
}
}
先是读入一个整数n,后面紧跟着n行字符串(可能含有空格)
测试数据
3
I am a student
test string
ok
import java.util.*;
public class C
{
public static void main(String[] args){
Scanner scan = new Scanner(System.in);
int n = scan.nextInt();
scan.nextLine(); //这个空读十分关键
for(int i=0; i<n; i++){
System.out.println(scan.nextLine());
}
}
}
import java.util.*;
public class C
{
public static void main(String[] args){
Scanner scan = new Scanner(System.in);
int n = scan.nextInt();
scan.nextLine(); //这个空读十分关键
for(int i=0; i<n; i++){
System.out.println(scan.nextLine());
}
}
}
这里要注意的是:在读入整数以后,不能直接按行读入。而是要先空读一行。
因为,nextInt 会越过空白,读取整数,直到遇到了下一个空白(这里就是回车),
但它不会把遇到的这个分隔符吃掉,而是留在缓冲区中。
所以,此时如果直接按行读入,就会先是空行,然后才能读到需要的内容。
而 nextLine 就不同,它不会把回车符留在缓冲区,同时也不会把回车符返回在结果串中。
这样安排有利于解决跨平台时,换行方式不一致的问题。
先是一个整数 n, 接下来有 n 行, 每行多个整数, 空格分开。要求对第每行求和
测试数据
4
1 2 3
10 20 30 40
333
444
此数据的最后一行没有回车
方法一:
import java.util.*;
public class D
{
public static void main(String[] args){
Scanner scan = new Scanner(System.in);
int n = scan.nextInt();
scan.nextLine(); // 读掉后面的一个回车符
for(int i=0; i<n; i++){
String s = scan.nextLine().trim();
String[] ss = s.split(" +"); // 因为1个或多个空格分开
int sum = 0;
for(int j=0; j<ss.length; j++){
sum += Integer.parseInt(ss[j]);
}
System.out.println(sum);
}
}
}
import java.util.*;
public class D
{
public static void main(String[] args){
Scanner scan = new Scanner(System.in);
int n = scan.nextInt();
scan.nextLine(); // 读掉后面的一个回车符
for(int i=0; i<n; i++){
String s = scan.nextLine().trim();
String[] ss = s.split(" +"); // 因为1个或多个空格分开
int sum = 0;
for(int j=0; j<ss.length; j++){
sum += Integer.parseInt(ss[j]);
}
System.out.println(sum);
}
}
}
这是比较简明的处理方法,每次把整个一行都读进来,再进行分割。
但这样处理可能有一个问题:当一行的数据太大(上百万比如),可能导致读入有问题。
如果能每次只读入一个数据项,读一个处理一个就很理想了。
所以才有方法二:
import java.util.*;
public class D2
{
public static void main(String[] args){
Scanner scan = new Scanner(System.in);
int n = scan.nextInt();
scan.nextLine(); // 读掉后面的一个回车符
scan.useDelimiter(" +"); //默认的数据项分割符是空白和回车换行都可以,这里改为若干空格
for(int i=0; i<n; i++){
int sum = 0;
while(scan.hasNextInt()){
sum += scan.nextInt();
}
if(scan.hasNextLine()){ // 加 if 防止最后一行没有回车符
sum += Integer.parseInt(scan.nextLine().trim());
}
System.out.println(sum);
}
}
}
import java.util.*;
public class D2
{
public static void main(String[] args){
Scanner scan = new Scanner(System.in);
int n = scan.nextInt();
scan.nextLine(); // 读掉后面的一个回车符
scan.useDelimiter(" +"); //默认的数据项分割符是空白和回车换行都可以,这里改为若干空格
for(int i=0; i<n; i++){
int sum = 0;
while(scan.hasNextInt()){
sum += scan.nextInt();
}
if(scan.hasNextLine()){ // 加 if 防止最后一行没有回车符
sum += Integer.parseInt(scan.nextLine().trim());
}
System.out.println(sum);
}
}
}