当前位置: 首页 > article >正文

C 语言实现计算一年中指定日期是第几天 题】

引言

 在编程的世界里,处理日期和时间相关的问题是非常常见的。比如在日历应用、任务管理系统、数据分析等场景中,经常需要计算某个日期在一年中是第几天。本文将详细介绍如何使用 C 语言来实现这一功能,通过分析代码的结构、逻辑以及可能存在的问题和改进方法,帮助大家更好地理解和掌握相关知识。

 

代码整体功能概述

 给定的 C 语言代码旨在实现一个简单的功能:从用户那里获取输入的年份、月份和日期,然后计算并输出该日期是对应年份中的第几天。代码通过定义变量存储年份、月份、日期,使用数组存储每个月的天数,并通过循环和条件判断来累加天数,最终得到结果。

代码详细解析

 1. 头文件和主函数声明

#include <stdio.h>

 

int main() 

{

 

代码开头包含了  <stdio.h>  头文件,这个头文件提供了标准输入输出函数的声明,比如我们后面要用到的  scanf  和  printf  函数。 main  函数是 C 程序的入口点,程序从这里开始执行。

 2. 变量定义

    int year=0;

    int month=0;

    int day=0;

    int sum=0;

 这里定义了四个整型变量。 year  用于存储输入的年份, month  存储月份, day  存储日期, sum  用于累加从 1 月到输入月份之前所有月份的天数。

 3. 输入获取

    scanf("%d %d %d", &year, &month, &day);

 

 scanf  函数用于从标准输入(通常是键盘)读取用户输入的内容。 %d %d %d  是格式控制字符串,表示要读取三个整数,分别对应年份、月份和日期。 &year 、 &month  和  &day  是变量的地址, scanf  函数会将读取到的值存储到这些变量对应的内存位置中。

 4. 存储每月天数的数组

    int arr[13]={0,31,28,31,30,31,30,31,31,30,31,30,31};

 定义了一个包含 13 个元素的整型数组  arr ,用于存储每个月的天数。数组的下标从 0 开始,但下标为 0 的元素没有实际意义,下标 1 - 12 分别对应 1 - 12 月。初始值按照平年的月份天数设置,即 1、3、5、7、8、10、12 月有 31 天,4、6、9、11 月有 30 天,2 月有 28 天。

 5. 结果变量和循环累加

    int resault=0;

    for(int i=1;i<month;i++) 

    {

        if(year%4==0&&year%100!=0 || year%400==0)

        {

            if(i==2)

            {

                arr[i]=29;

            }

            sum = sum +arr[i];

        }

        else

        {

            sum = sum +arr[i];

        }

    }

 

 resault  变量用于存储最终的结果,即该日期是一年中的第几天。 for  循环从 1 开始,到  month - 1  结束,用于累加输入月份之前所有月份的天数。在循环内部,首先通过条件  if(year%4==0&&year%100!=0 || year%400==0)  判断当前年份是否为闰年。如果是闰年,并且当前循环到的月份是 2 月( i == 2 ),则将  arr[2]  的值改为 29,即 2 月有 29 天。然后将当前月份的天数累加到  sum  中。如果不是闰年,则直接按照数组中存储的平年天数进行累加。

 

6. 计算最终结果并输出

    resault=sum+day;

    printf("%d",resault);

 将累加得到的之前月份的天数  sum  加上输入的日期  day ,得到最终的结果并存储在  resault  中。最后使用  printf  函数将结果输出到标准输出(通常是控制台)。

 

7. 主函数结束

    return 0;

}

 return 0  表示程序正常结束,返回值 0 通常用于向操作系统表明程序执行成功。

 

代码存在的问题

 1. 输入验证缺失

 代码中没有对用户输入的年份、月份和日期进行有效性检查。例如,年份可能是负数,月份可能不在 1 - 12 的范围内,日期可能超出了对应月份应有的天数(比如 2 月输入 30 天等)。如果用户输入了无效数据,程序可能会产生错误的结果甚至崩溃。

 2. 闰年判断逻辑的位置问题

 当前闰年判断逻辑在累加月份天数的循环内部,虽然功能上可以实现,但从代码结构和可读性角度来看,不够清晰。可以将闰年判断逻辑封装成一个单独的函数,这样代码的模块化程度更高,也更易于维护和理解。

 3. 数组使用的小瑕疵

数组  arr  的下标 0 没有实际意义,这种设计可能会让代码阅读者产生困惑,并且在一定程度上浪费了内存空间。可以考虑从下标 0 开始对应 1 月,重新设计数组的使用方式。

 

改进后的代码

#include <stdio.h>

 

// 判断是否为闰年

int isLeapYear(int year) {

    return (year % 4 == 0 && year % 100 != 0) || (year % 400 == 0);

}

 

// 获取每个月的天数

int getDaysInMonth(int year, int month) {

    int days[] = {31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};

    if (month == 2 && isLeapYear(year)) {

        return 29;

    }

    return days[month - 1];

}

 

int main() {

    int year, month, day;

    printf("请输入年份 月份 日期: ");

    while (scanf("%d %d %d", &year, &month, &day) != 3 || year < 0 || month < 1 || month > 12 || day < 1 || day > getDaysInMonth(year, month)) {

        printf("输入无效,请重新输入年份 月份 日期: ");

        while (getchar() != '\n'); // 清空输入缓冲区

    }

 

    int sum = 0;

    for (int i = 1; i < month; i++) {

        sum += getDaysInMonth(year, i);

    }

    sum += day;

    printf("这是该年的第 %d 天\n", sum);

    return 0;

}

 改进点说明

 - 输入验证:在  main  函数中,使用  while  循环对用户输入进行验证。如果输入的格式不正确( scanf  的返回值不等于 3,因为  scanf  成功读取三个整数时返回 3),或者年份、月份、日期不在合理范围内,就提示用户重新输入,并使用  getchar()  函数清空输入缓冲区,避免无效输入一直滞留在缓冲区影响后续输入。

 - 函数封装:将闰年判断逻辑封装成  isLeapYear  函数,将获取每个月天数的逻辑封装成  getDaysInMonth  函数。这样代码结构更加清晰,可读性和可维护性大大提高。

 - 数组优化:在  getDaysInMonth  函数中,数组  days  从下标 0 开始对应 1 月,通过  month - 1  来正确索引数组,避免了之前下标 0 无实际意义的问题。

 

总结

 通过对计算一年中指定日期是第几天的 C 语言代码的分析,我们深入了解了代码的实现逻辑、存在的问题以及如何进行改进。在编写程序时,不仅要关注功能的实现,还要注重代码的健壮性(如输入验证)、可读性(如函数封装和合理的变量命名)和可维护性。希望本文能帮助大家更好地理解 C 语言在日期计算方面的应用,并且在今后的编程中写出更加优质的代码。

       以上就是关于该 C 语言代码的全面解析和相关知识介绍,希望对你有所帮助。如果你在学习过程中还有其他疑问,欢迎随时交流探讨。


http://www.kler.cn/a/527215.html

相关文章:

  • FFmpeg(7.1版本)编译:Ubuntu18.04交叉编译到ARM
  • 最近最少使用算法(LRU最近最少使用)缓存替换算法
  • 【算法】动态规划专题① ——线性DP python
  • SSM开发(三) spring与mybatis整合(含完整运行demo源码)
  • 【Block总结】DynamicFilter,动态滤波器降低计算复杂度,替换传统的MHSA|即插即用
  • 【股票数据API接口41】如何获取股票指最新分时MA数据之Python、Java等多种主流语言实例代码演示通过股票数据接口获取数据
  • 【Linux】软硬链接
  • 英语语法 第一天
  • 【算法应用】基于鲸鱼优化算法求解OTSU多阈值图像分割问题
  • python 之 zip 和 * 解包操作
  • 微店的Flutter混合开发组件化与工程化架构
  • SQL NOW() 函数详解
  • Day52:range()函数
  • 精准化糖尿病知识问答(LLM+机器学习预测模型)
  • ELK模块封装starter
  • 数据结构初探: 顺序表
  • Mysql的主从复制及扩展功能
  • 代发考试战报:1月22号 1月23号 CCDE考试通过
  • 深入解析JUnit中的@ClassRule注解
  • 代码随想录算法训练营第十五天| 二叉树3
  • Python-操作列表
  • 38【2进制与ascall码】
  • 今日头条公域流量引流新径:开源 AI 智能名片 2 + 1 链动模式 S2B2C 商城小程序融合之道
  • 【C++语言】卡码网语言基础课系列----2. A+B问题II
  • 【漫话机器学习系列】072.异常处理(Handling Outliers)
  • 算法题(53):对称二叉树