王小二° 发表于 2023-11-12 11:55:01

救救孩子,今晚就截止了

任务描述
X同学最近非常烦,因为他正面对一大堆日期数据愁眉不展,原因是那些日期书写的极其不规范,不仔细辨别根本认不清是哪一天。
以下每个日期中的三个数,年月日的位置是不确定的,假设其中最大的数表示年份,1-12范围的数表示月份,1-31范围的数表示日期,如果月日的范围都是1-12则按前月后日的顺序识别。
例如:
2/3-123   识别为0123年2月3日,合法;
13+12+45    识别为0045年12月13日,合法;
12=3=13   识别为0013年12月3日,合法;
29,2,2019   识别为2019年2月29日,这一天不存在,不合法;
12@12#12    识别为0012年12月12日,合法;
35.36.37    识别为0037年XX月XX日,XX超出月份和日期范围,不合法;
11*30*30    识别为0030年11月30日,合法;
现在X要编程判断一下那些日期到底是否合法,你们说他能编出来吗。
输入格式:
一行,以X-Y-Z形式表示的一个日期,其中X、Y、Z为不超过4位的正整数,“-”为一个分隔字符,可能为任何符号。

输出格式:
若该日期为合法存在的日期,以“YYYY-MM-DD”的形式输出该日期的规范式,年4位,月日2位,不足前补0。
若该日期为不合法日期,输出:Invalid Date!。

输入样例:
25-30-5
输出样例:
0030-05-25
输入样例:
2019-32-09
输出样例:
Invalid Date!
输入样例:
2019-30-09
输出样例:
2019-09-30

麻烦用C语言,然后顺便讲解一下

jackz007 发表于 2023-11-12 11:55:02

本帖最后由 jackz007 于 2023-11-12 14:29 编辑

#include <stdio.h>

int leapyear(int y)
{
      return ((y % 400 == 0) || ((y % 4 == 0) && (y % 100 != 0))) ? 1 : 0   ;
}

int lastday(int y , int m)
{
      int d[] = {31 , 28 , 31 , 30 , 31 , 30 , 31 , 31 , 30 , 31 , 30 , 31} ;
      return (leapyear(y) && m == 2) ? 29 : d                        ;
}

int main(void)
{
      char c                                             ;
      int a , d , e , f , i , k , m , y               ;
      for(;;) {
                for(i = 0 ; i < 3 ; i ++) e = 0         ; // 每一轮输入-判断开始前,需要先将年月日数据清零
                for(i = k = 0 ; k < 3 ;) {
                        c = getchar()                      ;
                        if(c >= '0' && c <= '9') {
                              e = e * 10 + c - '0' ;
                              i ++                     ;
                        } else if(i) {
                              i = 0                      ;
                              k ++                     ;
                        }
                        if(c == '\n') break                ;
                }
                if(k == 3) {
                        for(a = 0 , i = 1 ; i < 3 ; i ++) if(e > e) a = i   ; // 取得数组 e[] 三个元素中值最大那个元素的索引值 a
                        for(i = 0 , m = 3 ; i < 3 ; i ++) if(i != a) e = e ; // 将数组 e[] 中除最大值外剩余的两个元素复制到 e 和 e 中
                        y = e                                                    ; // 最大值 e 为年,e、e 两个元素为月和日
                        f = 0                                                       ; // 先将日期合法标志初始化为无效
                        if(e >= 1 && e <= 12 && e >= 1 && e <= lastday(y , e)) { // 将 e、e 分别作为月、日判断是否合法
                              m = e                                          ; // 如果判断通过则执行这些代码
                              d = e                                          ;
                              f ++                                                ;
                        } else if(e >= 1 && e <= 12 && e >= 1 && e <= lastday(y , e)) { // 将 e、e 分别作为日、月判断是否合法
                              m = e                                          ; // 如果判断通过则执行这些代码
                              d = e                                          ;
                              f ++                                                ;
                        }
                        if(f) printf("%04d-%d-%d\n" , y , m , d)                  ; // 根据判断标志确定年月日数据是否有效
                        else printf("Invalid Date !\n")                           ;
                } else if(! k) break                                                ;                                                               
      }
}

isdkz 发表于 2023-11-12 11:55:09

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

int isLeapYear(int year) {
    return (year % 4 == 0 && year % 100 != 0) || (year % 400 == 0);
}

int isValidDate(int year, int month, int day) {
    if (month < 1 || month > 12 || day < 1) {
      return 0;
    }
    int maxDay = 31;
    if (month == 4 || month == 6 || month == 9 || month == 11) {
      maxDay = 30;
    } else if (month == 2) {
      maxDay = isLeapYear(year) ? 29 : 28;
    }
    return day <= maxDay;
}

void parseAndValidateDate(char *input) {
    char *token;
    token = strtok(input, "-");
    int nums;
    int i = 0;
    while (token != NULL) {
      nums[ i] = atoi(token);
      token = strtok(NULL, "-");
      i++;
    }
    if (i != 3) {
      printf("Invalid Date!\n");
      return;
    }
    int year, month, day;
    if (nums > 999) {
      year = nums;
      month = nums;
      day = nums;
    } else if (nums > 999) {
      year = nums;
      month = nums;
      day = nums;
    } else {
      year = nums;
      month = nums;
      day = nums;
    }
    if (isValidDate(year, month, day)) {
      printf("%04d-%02d-%02d\n", year, month, day);
    } else {
      printf("Invalid Date!\n");
    }
}

int main() {
    char input;
    scanf("%s", input);
    parseAndValidateDate(input);
    return 0;
}


这段C语言代码实现了对输入日期的解析和验证。首先,使用 strtok 函数将输入的日期字符串按照分隔符 - 进行分割,并将每个部分转换为整数。然后根据最大的数作为年份,依次判断是否为合法日期,如果是则输出规范格式的日期,否则输出"Invalid Date!"。isValidDate函数用于验证日期的合法性,isLeapYear函数用于判断闰年。

王小二° 发表于 2023-11-12 11:58:45

isdkz 发表于 2023-11-12 11:55
这段C语言代码实现了对输入日期的解析和验证。首先,使用 strtok 函数将输入的日期字符串按照分隔符 -...

你好,测试输入:
20-12-432
—— 预期输出 ——0432-12-20
—— 实际输出 ——Invalid Date!

这个测试通过不了

陶远航 发表于 2023-11-12 12:08:03

这是一个有趣的问题,我们可以通过以下步骤来解决:

1. 首先,我们需要解析输入的字符串,将其分解为三个整数。我们可以使用 `sscanf` 函数来实现这一点。
2. 然后,我们需要确定哪个数字代表年份,哪个数字代表月份,哪个数字代表日期。我们可以通过比较三个数字的大小来实现这一点。
3. 最后,我们需要检查日期是否合法。我们可以使用 `tm` 结构和 `mktime` 函数来实现这一点。

以下是一个可能的C语言实现:

#include <stdio.h>
#include <time.h>

int main() {
    int a, b, c;
    char s;
    scanf("%s", s);
    sscanf(s, "%d%*[^0-9]%d%*[^0-9]%d", &a, &b, &c);

    int year, month, day;
    if (a > b && a > c) {
      year = a;
      if (b > c) {
            month = b;
            day = c;
      } else {
            month = c;
            day = b;
      }
    } else if (b > a && b > c) {
      year = b;
      if (a > c) {
            month = a;
            day = c;
      } else {
            month = c;
            day = a;
      }
    } else {
      year = c;
      if (a > b) {
            month = a;
            day = b;
      } else {
            month = b;
            day = a;
      }
    }

    struct tm timeinfo = {0};
    timeinfo.tm_year = year - 1900;
    timeinfo.tm_mon = month - 1;
    timeinfo.tm_mday = day;
    if (mktime(&timeinfo) == -1 || timeinfo.tm_year != year - 1900 || timeinfo.tm_mon != month - 1 || timeinfo.tm_mday != day) {
      printf("Invalid Date!\n");
    } else {
      printf("%04d-%02d-%02d\n", year, month, day);
    }

    return 0;
}

这段代码首先读取输入的字符串,然后使用 `sscanf` 函数将其解析为三个整数。然后,它通过比较这三个数字的大小来确定年份、月份和日期。最后,它使用 `tm` 结构和 `mktime` 函数来检查日期是否合法。如果日期合法,它将以 "YYYY-MM-DD" 的格式打印日期;否则,它将打印 "Invalid Date!"。

zhangjinxuan 发表于 2023-11-12 14:38:08

不可以总司令{:10_256:}

liuhongrun2022 发表于 2023-11-12 15:52:46

zhangjinxuan 发表于 2023-11-12 14:38
不可以总司令

{:10_256:}

yinda_peng 发表于 2023-11-12 17:03:25

额,作业不自己做吗

王小二° 发表于 2023-11-12 18:37:56

陶远航 发表于 2023-11-12 12:08
这是一个有趣的问题,我们可以通过以下步骤来解决:

1. 首先,我们需要解析输入的字符串,将其分解为三 ...

你好,这个有三个不能实现

王小二° 发表于 2023-11-12 18:38:47

isdkz 发表于 2023-11-12 11:55
这段C语言代码实现了对输入日期的解析和验证。首先,使用 strtok 函数将输入的日期字符串按照分隔符 -...

麻烦可以完善一下吗,只差一点了
页: [1]
查看完整版本: 救救孩子,今晚就截止了