首先理解错了题目意思,做法也很复杂:
import java.util.Scanner;
import java.util.Set;
import java.util.TreeSet;
/**
* 开发一个简单错误记录功能小模块,能够记录出错的代码所在的文件名称和行号。
1、 记录最多8条错误记录,循环记录,对相同的错误记录(净文件名称和行号完全匹配)只记录一条,错误计数增加;
2、 超过16个字符的文件名称,只记录文件的最后有效16个字符;
3、 输入的文件可能带路径,记录文件名称不能带路径。
输入描述:一行或多行字符串。每行包括带路径文件名称,行号,以空格隔开。
输出描述:将所有的记录统计并将结果输出,格式:文件名 代码行数 数目,一个空格隔开,如:
输入例子:E:\V1R2\product\fpgadrive.c 1325
输出例子:fpgadrive.c 1325 1
* @author WGS
*/
public class Main {
private static int errorItems=0;
private static void RecordingErrorLogs(Set<String> set,String[] input) {
//input[0]文件路径:E:\V1R2\product\fpgadrive.c input[1]行号:1325
String fileName="";
String rowNumStr="";
//1 获取文件名,超过长度16的截取
int index1=input[0].lastIndexOf("\\");//最后一个斜杠位置
int lenthOfFile=input[0].length()-index1-1;
if(lenthOfFile>16){
index1=input[0].length()-16;
}
//2 获取行号
rowNumStr=input[1];
String validStr= fileName +","+ rowNumStr;
//3 验证:对相同的错误记录(净文件名称和行号完全匹配)只记录一条,错误计数增加;
if(!set.add(validStr)){
set.add(validStr);
errorItems++;//有重复
}
}
private static String setToString(Set<String> set){
StringBuilder sb=new StringBuilder(256);
for(String s:set){
String[] tempStr=s.split(",");
sb.append(tempStr[0]).append(" ").append(tempStr[1]).append(" ").append(errorItems).append("\n");
}
return sb.toString();
}
public static void main(String[] args) {
Scanner sc=new Scanner(System.in);
Set<String> set=new TreeSet<>();
while(sc.hasNext()){
for(int i=0;i<8;i++){
String[] input=sc.nextLine().split("\\s+");
RecordingErrorLogs(set,input);
}
System.out.println(setToString(set));
}
sc.close();
}
}
参考了别人的做法:
package com.oj.test;
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.Scanner;
public class Main{
public static void main(String[] args) {
Scanner sc=new Scanner(System.in);
Map<String, Integer> map=new LinkedHashMap<String, Integer>();
while(sc.hasNext()){
String str=sc.next();
int linenum=sc.nextInt();
String[] arr=str.split("\\\\"); //根据\切割
String s=arr[arr.length-1];
if(s.length()>16) //截取
s=s.substring(s.length()-16);
String key=s+" "+linenum;
int value=1;
if(map.containsKey(key))
map.put(key, map.get(key)+1);
else {
map.put(key, value);
}
}
int count=0;
for(String string:map.keySet()){
count++;
if(count>(map.keySet().size()-8)) //输出最后八个记录
System.out.println("***"+string+" "+map.get(string));
}
}
}
自己重新做了一遍,整理了一下思路:
package com.oj.exe2;
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Scanner;
import java.util.Set;
import java.util.TreeSet;
/**
* 开发一个简单错误记录功能小模块,能够记录出错的代码所在的文件名称和行号。
1、 记录最多8条错误记录,循环记录,对相同的错误记录(净文件名称和行号完全匹配)只记录一条,错误计数增加;
2、 超过16个字符的文件名称,只记录文件的最后有效16个字符;
3、 输入的文件可能带路径,记录文件名称不能带路径。
输入描述:一行或多行字符串。每行包括带路径文件名称,行号,以空格隔开。
输出描述:将所有的记录统计并将结果输出,格式:文件名 代码行数 数目,一个空格隔开,如:
输入例子:E:\V1R2\product\fpgadrive.c 1325
输出例子:fpgadrive.c 1325 1
* @author WGS
*/
public class Exe19_2 {
private static void RecordingErrorLogs(Map<String, Integer> map,String fileLocation,int lineNum) {
//1 获取文件名,超过长度16的截取
int index1=fileLocation.lastIndexOf("\\");//最后一个斜杠位置
int lenthOfFile=fileLocation.length()-index1-1;
if(lenthOfFile>16){
index1=fileLocation.length()-16;
}
String fileName=fileLocation.substring(index1+1);
//2 将文件路径和行号连接为字符串,作为map的key键值
String inputStr= fileName +" "+ lineNum;
//3 验证:对相同的错误记录(净文件名称和行号完全匹配)只记录一条,错误计数增加;
if(map.containsKey(inputStr)){
map.put(inputStr,map.get(inputStr)+1);
}else{
//没有重复的,次数就设置为1
map.put(inputStr,1);
}
}
public static void main(String[] args) {
Scanner sc=new Scanner(System.in);
//不需要排序,所以使用LinkedHashMap
Map<String, Integer> map=new LinkedHashMap<String, Integer>();
while(sc.hasNext()){
String fileLocation=sc.next();
int lineNum=sc.nextInt();
//输一条信息,记录一条
RecordingErrorLogs(map,fileLocation,lineNum);
}
//遍历map的key键值(含有文件+行号信息),输出最后的八条记录;
int count=0;
for(String s:map.keySet()){
count++;
if(count>(map.keySet().size()-8))//从倒数第八条开始遍历
System.out.println(s+" "+map.get(s));//文件路径行号+个数
}
sc.close();
}
}
结果在eclipse中能够完全通过(使用他给的测试用例),牛客上只通过30%,考虑到代表两个//的意思,可能是输入路径中的“\”被转义成"//"了,所以应该这样:
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Scanner;
/**
* 开发一个简单错误记录功能小模块,能够记录出错的代码所在的文件名称和行号。
1、 记录最多8条错误记录,循环记录,对相同的错误记录(净文件名称和行号完全匹配)只记录一条,错误计数增加;
2、 超过16个字符的文件名称,只记录文件的最后有效16个字符;
3、 输入的文件可能带路径,记录文件名称不能带路径。
输入描述:一行或多行字符串。每行包括带路径文件名称,行号,以空格隔开。
输出描述:将所有的记录统计并将结果输出,格式:文件名 代码行数 数目,一个空格隔开,如:
输入例子:E:\V1R2\product\fpgadrive.c 1325
输出例子:fpgadrive.c 1325 1
* @author WGS
*/
public class Main {
private static void RecordingErrorLogs(Map<String, Integer> map,String fileLocation,int lineNum) {
//input[0]文件路径:E:\V1R2\product\fpgadrive.c input[1]行号:1325
//1 获取文件名,超过长度16的截取
String[] arr=fileLocation.split("\\\\");//分解:>>E: \\ V1R2 \\ product \\ fpgadrive.c
String fileName=arr[arr.length-1];
if(fileName.length()>16){
fileName=fileName.substring(fileName.length()-16);
}
//2 将文件路径和行号连接为字符串,作为map的key键值
String inputStr= fileName +" "+ lineNum;
//3 验证:对相同的错误记录(净文件名称和行号完全匹配)只记录一条,错误计数增加;
if(map.containsKey(inputStr)){
map.put(inputStr,map.get(inputStr)+1);
}else{
//没有重复的,次数就设置为1
map.put(inputStr,1);
}
}
public static void main(String[] args) {
Scanner sc=new Scanner(System.in);
//不需要排序,所以使用LinkedHashMap
Map<String, Integer> map=new LinkedHashMap<String, Integer>();
while(sc.hasNext()){
String fileLocation=sc.next();
int lineNum=sc.nextInt();
//输一条信息,记录一条
RecordingErrorLogs(map,fileLocation,lineNum);
}
//遍历map的key键值(含有文件+行号信息),输出最后的八条记录;
int count=0;
for(String s:map.keySet()){
count++;
if(count>map.size()-8)//从倒数第八条开始遍历
System.out.println(s+" "+map.get(s));//文件路径行号+个数
}
sc.close();
}
}