我正在尝试解决此page中描述的“最佳之前”Spotify拼图。基本上,对于由斜杠分隔的三个整数的输入(例如11/3/4),您应该以2011-03-04格式生成具有最早可能日期的输出。如果没有可能的日期,则应返回原始字符串,后跟“非法”。
下面我的解决方案的想法是从我在github找到同样问题的Python解决方案中借来的。当我提交这个Python代码时,它被接受了。不熟悉Python,这是我尝试用Java创建类似的东西,而不使用任何Calendar函数,这可以在stackoverflow上发布的this解决方案中看到。
但是,当我提交我的解决方案时,我会收到“错误答案”作为回复。尽我所能,我找不到这个代码的任何错误。我觉得我已经尝试了所有可能的输入组合,并且我的所有输出都正确输出。任何人都知道我可能会缺少什么?
由于我对编程总体上比较陌生,所以如果你愿意,也可以随意提供有关如何改进代码的建议。我相信它可能看起来很无聊。谢谢!
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.Arrays;
public class DateProggy3 {
static int var1, var2, var3;
static int slashPosition1, slashPosition2;
static String yearString, monthString, dayString;
public static void main(String[] args) throws IOException {
String dateInput = readDate();
splitInputToInts(dateInput);
Integer[] dateArray = {var1, var2, var3};
Arrays.sort(dateArray);
Integer bestDate[] = getBestDate(dateArray, dateInput);
convertDate(bestDate);
printDate(bestDate);
}
public static String readDate() throws IOException {
BufferedReader stdin = new BufferedReader
(new InputStreamReader(System.in));
String dateInput;
dateInput = stdin.readLine();
return dateInput;
}
public static void splitInputToInts(String dateInput) {
try {
slashPosition1 = dateInput.indexOf('/');
slashPosition2 = dateInput.lastIndexOf('/');
var1 = Integer.parseInt(dateInput.substring(0, slashPosition1));
var2 = Integer.parseInt(dateInput.substring(slashPosition1+1, slashPosition2));
var3 = Integer.parseInt(dateInput.substring(slashPosition2+1, dateInput.length()));
}catch (StringIndexOutOfBoundsException e){
illegal(dateInput);
}catch (NumberFormatException e){
illegal(dateInput);
}
}
public static void illegal(String dateInput){
System.out.println(dateInput + " is illegal");
System.exit(0);
}
public static Integer[] getBestDate(Integer[] dateArray, String dateInput){
var1 = dateArray[0];
var2 = dateArray[1];
var3 = dateArray[2];
if (testDate(var1, var2, var3)){
Integer[] bestDate = {var1, var2, var3};
return bestDate;
}
else if (testDate(var1, var3, var2)){
Integer[] bestDate = {var1, var3, var2};
return bestDate;
}
else if (testDate(var2, var1, var3)){
Integer[] bestDate = {var2, var1, var3};
return bestDate;
}
else if (testDate(var2, var3, var1)){
Integer[] bestDate = {var2, var3, var1};
return bestDate;
}
else if (testDate(var3, var1, var2)){
Integer[] bestDate = {var3, var1, var2};
return bestDate;
}
else if (testDate(var3, var2, var1)){
Integer[] bestDate = {var3, var2, var1};
return bestDate;
}else{
illegal(dateInput);
}
Integer[] bestDate = {var1, var2, var3};
return bestDate;
}
public static boolean testDate(int year, int month, int day){
boolean leapYear = false;
boolean dateOK;
if (year > 100 && year < 2000){
return dateOK = false;
}
if (year < 1000){
year+=2000;
}
if (year < 0){
return dateOK = false;
}
if (year % 4 == 0) {
if (year % 100 == 0 && year % 400 != 0) {
leapYear = false;
}
leapYear = true;
}else{
leapYear = false;
}
if (month > 12 || month < 1){
return dateOK = false;
}
switch (month){
case 1:
case 3:
case 5:
case 7:
case 8:
case 10:
case 12:
if (day > 31 || day < 1){
return dateOK = false;
}
break;
case 4:
case 6:
case 9:
case 11:
if (day > 30 || day < 1){
return dateOK = false;
}
break;
case 2:
int maxDay;
if (leapYear){
maxDay = 29;
}else{
maxDay = 28;
}
if (day > maxDay || day < 1){
return dateOK = false;
}
}
return dateOK = true;
}
public static void convertDate(Integer[] dateArray){
if (dateArray[0] < 1000){
dateArray[0]+=2000;
}
yearString = String.valueOf(dateArray[0]);
if (dateArray[1] < 10){
monthString = "0" + dateArray[1];
}else{
monthString = String.valueOf(dateArray[1]);
}
if (dateArray[2] < 10){
dayString = "0" + dateArray[2];
}else{
dayString = String.valueOf(dateArray[2]);
}
}
public static void printDate(Integer[] dateArray){
System.out.println(yearString + "-" + monthString +"-" + dayString);
}
}
我是那个提出这个问题的人,但是因为我注册了stackoverflow并丢失了我原来的cookie或其他东西,所以似乎无法评论并正常回复答案。
无论如何,感谢palacsint的回答。我修复了闰年问题,现在我的答案终于被接受了!
关于getBestDate()
方法中最后两行的问题。我把那些放在那里只是因为Eclipse IDE否则会给我错误“此方法必须返回Integer []类型的结果”。在if
括号中获得所有回报似乎并不满意。有没有办法解决?谢谢。
答案 0 :(得分:1)
一个错误:它接受2100/02/29。 2100年不闰年,因此没有2011/02/29。
如果我是你,我会使用SimpleDateFormat
进行解析和验证(提示:lenient
解析)。它更简单,更直观,代码更容易阅读。 (不要重新发明轮子)
上面的一些其他想法。
不必要的作业: return dateOK = false;
返回false
:
return false;
(在您的情况下,dataOK
变量是不必要的。)
public static void illegal(String dateInput){
System.out.println(dateInput + " is illegal");
System.exit(0);
}
抛出异常而不是System.exit()
。
在getBestDate()
方法中,最后两行永远不会运行。它们是死代码(因为illegal()
调用System.exit()
):
}else{
illegal(dateInput);
}
Integer[] bestDate = {var1, var2, var3};
return bestDate;
如果可以避免静态方法和字段。最后,学习如何编写单元测试是一个很好的问题。
答案 1 :(得分:0)
您的'年'处理存在问题。谜题文字说
鉴于可能含糊不清的日期“A / B / C”,其中A,B,C是整数 在0到2999之间,输出最早的合法日期 2000年1月1日和2999年12月31日(包括在内)将它们用作日,月和日 一年(但不一定按顺序)。
但是,当我在你的程序中输入1000到2000之间的年份时,它会逐字报告,即使它们不是有效的输出。