Spotify拼图问题
我正在尝试解决page中描述的Spotify难题。基本上,如果输入三个用斜杠分隔的整数(例如11/3/4),您应该生成的输出具有2011-03-04格式的最早日期。如果没有可能的日期,它应该返回原始字符串,后跟"是非法的"。
下面我的解决方案的想法借用了我在github中为相同问题找到的一个Python解决方案。当我提交这段Python代码时,它被接受了。由于不熟悉Python,这是我用Java做类似事情的最好尝试,而且不使用任何日历函数,就像在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会给我错误"This方法必须返回Integer[]类型的结果"。它似乎并不满足于将所有的回报都放在if
括号中。有什么办法可以绕过这件事吗?谢谢。
解决方案
一个错误:它接受2100/02/29。2100不是闰年,所以没有2011/02/29。
如果我是您,我会使用SimpleDateFormat
进行解析和验证(提示:lenient
解析)。它更简单、更直观,代码也更容易阅读。(不要重新发明车轮)
上面的一些其他想法。
不必要的作业: 返回日期确定=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;
如果可能,请避免使用静态方法和字段。最后,学习如何编写单元测试是一个好问题。
相关文章