Java的next() nextLine()的区别

2019-04-14 19:52发布

    做CCF20150903模板自动生成系统的时候遇到一个坑,刚开始以为是程序逻辑问题,改来改去,直到后面发现原来是nextInt()的坑,这里特别总结一下next(),nextLine(),nextInt(),nextFloat(),nextDouble(),希望对大家有所帮助~     很多书上会介绍next(),nextLine()不会像下面这篇博主这么详细:参考下博主的成果     这里引用新浪博客http://blog.sina.com.cn/s/blog_781d21190100terb.html,写的比较详细     next()一定要读取到有效字符后才可以结束输入,对输入有效字符之前遇到的空格键、Tab键或Enter键等结束符,next()方法会自动将其去掉,只有在输入有效字符之后,next()方法才将其后输入的空格键、Tab键或Enter键等视为分隔符或结束符。简单地说,next()查找并返回来自此扫描器的下一个完整标记。完整标记的前后是与分隔模式匹配的输入信息,所以next方法不能得到带空格的字符串。     而nextLine()方法的结束符只是Enter键,即nextLine()方法返回的是Enter键之前的所有字符,它是可以得到带空格的字符串的。 鉴于以上两种方法的只要区别,同学们一定要注意next()方法和nextLine(0方法的连用,下面举个例子来说明: import java.util.Scanner;
public class NextTest {  
 public static void main(String[] args) {
  // TODO Auto-generated method stub
  String s1,s2;
  Scanner sc=new Scanner(System.in);
  System.out.print("请输入第一个字符串:");
  s1=sc.nextLine();
  System.out.print("请输入第二个字符串:");
  s2=sc.next();
  System.out.println("输入的字符串是:"+s1+"  "+s2);
 }
} 运行结果: 请输入第一个字符串:home
请输入第二个字符串:work
输入的字符串是:home  work 但如果把程序改一下, s1=sc.next(); s2=sc.nextLine(); 运行结果是: 请输入第一个字符串:home
请输入第二个字符串:输入的字符串是:home 

可以看到,nextLine()自动读取了被next()去掉的Enter作为他的结束符,所以没办法给s2从键盘输入值。经过验证,我发现其他的next的方法,如double nextDouble()  , float nextFloat() , int nextInt() 等与nextLine()连用时都存在这个问题,解决的办法是:在每一个 next()、nextDouble()  、 nextFloat()、nextInt() 等语句之后加一个nextLine()语句,将被next()去掉的Enter结束符过滤掉。例如上面的程序改写为: import java.util.Scanner;
public class NextTest {
 
 public static void main(String[] args) {
  // TODO Auto-generated method stub
  String s1,s2;
  Scanner sc=new Scanner(System.in);
  System.out.print("请输入第一个字符串:");
  s1=sc.next();
  sc.nextLine();
  System.out.print("请输入第二个字符串:");
  s2=sc.nextLine();
  System.out.println("输入的字符串是:"+s1+"  "+s2);
 }
} 运行结果是: 请输入第一个字符串:home
请输入第二个字符串:work
输入的字符串是:home  work 然后结合模板生成系统这道题目来叙述一下: 成成最近在搭建一个网站,其中一些页面的部分内容来自数据库中不同的数据记录,但是页面的基本结构是相同的。例如,对于展示用户信息的页面,当用户为 Tom 时,网页的源代码是


  而当用户为 Jerry 时,网页的源代码是


  这样的例子在包含动态内容的网站中还有很多。为了简化生成网页的工作,成成觉得他需要引入一套模板生成系统。
  模板是包含特殊标记的文本。成成用到的模板只包含一种特殊标记,格式为 {{ VAR }},其中 VAR 是一个变量。该标记在模板生成时会被变量 VAR 的值所替代。例如,如果变量 name = "Tom",则 {{ name }} 会生成 Tom。具体的规则如下:
  ·变量名由大小写字母、数字和下划线 (_) 构成,且第一个字符不是数字,长度不超过 16 个字符。
  ·变量名是大小写敏感的,Name 和 name 是两个不同的变量。
  ·变量的值是字符串。
  ·如果标记中的变量没有定义,则生成空串,相当于把标记从模板中删除。
  ·模板不递归生成。也就是说,如果变量的值中包含形如 {{ VAR }} 的内容,不再做进一步的替换。 输入格式   输入的第一行包含两个整数 mn,分别表示模板的行数和模板生成时给出的变量个数。
  接下来 m 行,每行是一个字符串,表示模板。
  接下来 n 行,每行表示一个变量和它的值,中间用一个空格分隔。值是字符串,用双引号 (") 括起来,内容可包含除双引号以外的任意可打印 ASCII 字符(ASCII 码范围 32, 33, 35-126)。 输出格式   输出包含若干行,表示模板生成的结果。 样例输入 11 2



User {{ name }}


{{ name }}


Email: {{ email }}


Address: {{ address }}




name "David Beckham"
email "david@beckham.com" 样例输出


User David Beckham


David Beckham


Email: david@beckham.com


Address:



评测用例规模与约定   0 ≤ m ≤ 100
  0 ≤ n ≤ 100
  输入的模板每行长度不超过 80 个字符(不包含换行符)。
  输入保证模板中所有以 {{ 开始的子串都是合法的标记,开始是两个左大括号和一个空格,然后是变量名,结尾是一个空格和两个右大括号。
  输入中所有变量的值字符串长度不超过 100 个字符(不包括双引号)。
  保证输入的所有变量的名字各不相同。 //90分 还有一种测试案例没通过,估计跟template[i] = template[i].replaceAll("\{{2}\s.*\s\}{2}", "");有关系 import java.util.*;
public class test {
public static void main(String[] args) {
Scanner input = new Scanner(System.in);
int m = input.nextInt();
int n = input.nextInt();
input.nextLine();//一定要过滤掉空格符号
String[] template = new String[m];
String[] key = new String[n];
String[] value = new String[n];
String[] newValue = new String[n];
for (int i = 0; i < m; i++)
template[i] = input.nextLine();
for (int i = 0; i < n; i++) {
key[i] = input.next();
value[i] = input.nextLine().trim();
}
for(int i=0;inewValue[i] = value[i].substring(1, value[i].length()-1);
}
for (int i = 0; i < m; i++) {
if (template[i].contains("{{ ") && template[i].contains(" }}")) {
for (int j = 0; j < n; j++) {
if (template[i].contains(key[j])) {
template[i] = template[i].replace("{{ " + key[j] + " }}", newValue[j]);
}
}
template[i] = template[i].replaceAll("\{{2}\s.*\s\}{2}", "");
}
}
for (int i = 0; i < m; i++)
System.out.println(template[i]);
}
}
PS:这种方法很基础,CSDN有大佬用映射表写的答案,100分