2020-06-06 01:53:38 +08:00
|
|
|
/**
|
|
|
|
* \file
|
|
|
|
* \brief [Problem 19](https://projecteuler.net/problem=19) solution
|
2020-06-07 02:51:49 +08:00
|
|
|
* \author [Krishna Vedala](https://github.com/kvedala)
|
2020-06-06 01:53:38 +08:00
|
|
|
*/
|
2020-04-02 00:10:59 +08:00
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
/**
|
2020-07-05 03:05:30 +08:00
|
|
|
* Function to get the number of days in a month.
|
|
|
|
* \param month month identified by an integer -\n
|
|
|
|
* > 0 = Jan and 11 = December
|
|
|
|
* \returns number of days in given month
|
|
|
|
* \note For February, adjust for leap year outside the function.
|
2020-06-29 03:18:52 +08:00
|
|
|
*/
|
2020-04-02 00:10:59 +08:00
|
|
|
char get_month_days(short month)
|
|
|
|
{
|
|
|
|
if (month == 1) /* February has 28 days. Adjust leap year in the loop */
|
|
|
|
return 28;
|
|
|
|
else if (month <= 6) /* odd months till July have 30 days - Jan = 0 (even)*/
|
|
|
|
{
|
|
|
|
if (month & 0x01)
|
|
|
|
return 30;
|
|
|
|
else
|
|
|
|
return 31;
|
2020-05-30 04:23:24 +08:00
|
|
|
}
|
2020-07-05 03:05:30 +08:00
|
|
|
|
|
|
|
// else if (month >= 7) /* odd months after July have 31 days*/
|
|
|
|
|
|
|
|
if (month & 0x01)
|
|
|
|
return 31;
|
|
|
|
|
|
|
|
return 30;
|
2020-04-02 00:10:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-07-05 03:05:30 +08:00
|
|
|
* @brief Check if input year is a leap year.
|
|
|
|
* \param year year to check
|
|
|
|
* \return 1 if input year is a leap year
|
|
|
|
* \return 0 if input year is not a leap year
|
2020-06-29 03:18:52 +08:00
|
|
|
*/
|
2020-04-02 00:10:59 +08:00
|
|
|
char is_leap_year(short year)
|
|
|
|
{
|
|
|
|
if ((year % 400 == 0) || ((year % 4 == 0) && (year % 100 != 0)))
|
|
|
|
return 1;
|
2020-05-30 04:23:24 +08:00
|
|
|
|
2020-04-02 00:10:59 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
2020-07-05 03:05:30 +08:00
|
|
|
/** Function to convert integer month to string
|
|
|
|
* \param day integer identifier of day (0 = Sunday and 7 = Saturday
|
|
|
|
* \return pointer to string representation)
|
|
|
|
*/
|
2020-05-30 04:23:24 +08:00
|
|
|
const char *day_string(int day)
|
2020-04-02 00:10:59 +08:00
|
|
|
{
|
2020-05-30 04:23:24 +08:00
|
|
|
switch (day)
|
2020-04-02 00:10:59 +08:00
|
|
|
{
|
2020-05-30 04:23:24 +08:00
|
|
|
case 0:
|
|
|
|
return "Sunday";
|
|
|
|
case 1:
|
|
|
|
return "Monday";
|
|
|
|
case 2:
|
|
|
|
return "Tuesday";
|
|
|
|
case 3:
|
|
|
|
return "Wednesday";
|
|
|
|
case 4:
|
|
|
|
return "Thursday";
|
|
|
|
case 5:
|
|
|
|
return "Friday";
|
|
|
|
case 6:
|
|
|
|
return "Saturday";
|
|
|
|
default:
|
2020-07-05 03:05:30 +08:00
|
|
|
return "Shouldn't see this!";
|
2020-04-02 00:10:59 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2020-06-06 01:53:38 +08:00
|
|
|
/** Main function */
|
2020-04-02 00:10:59 +08:00
|
|
|
int main(int argc, char **argv)
|
|
|
|
{
|
|
|
|
int count_sundays = 0;
|
|
|
|
const short start_year = 1901;
|
|
|
|
const short end_year = 2000;
|
|
|
|
|
2020-06-29 03:18:52 +08:00
|
|
|
/*
|
2020-05-30 04:23:24 +08:00
|
|
|
* Let us identify days i.e., Sunday thru Saturday with integers - 0 thru 6
|
2020-06-29 03:18:52 +08:00
|
|
|
* respectively Jan 1 1901 was a Tuesday
|
|
|
|
*/
|
2020-04-02 00:10:59 +08:00
|
|
|
char start_day = 2;
|
|
|
|
|
|
|
|
for (int year = start_year; year <= end_year; year++)
|
|
|
|
{
|
|
|
|
char is_leap = is_leap_year(year);
|
2020-05-30 04:23:24 +08:00
|
|
|
for (char month = 0; month < 12; month++)
|
2020-04-02 00:10:59 +08:00
|
|
|
{
|
2020-06-29 03:18:52 +08:00
|
|
|
/*
|
2020-05-30 04:23:24 +08:00
|
|
|
* These two for-loops count the start of day for the next month.
|
2020-06-29 03:18:52 +08:00
|
|
|
* Hence, we have to skip the last December count
|
|
|
|
*/
|
2020-04-02 00:10:59 +08:00
|
|
|
if (year == end_year && month == 11)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
int days = get_month_days(month);
|
2020-05-30 04:23:24 +08:00
|
|
|
|
|
|
|
if (is_leap && month == 1) /* for a leap year february, add a day */
|
|
|
|
days++;
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
2020-04-02 00:10:59 +08:00
|
|
|
if (year == end_year)
|
|
|
|
{
|
2020-05-30 04:23:24 +08:00
|
|
|
printf("Year: %d\t Month: %d\t Days: %d\t First of day: %s\n",
|
|
|
|
year, month, days, day_string(start_day));
|
2020-04-02 00:10:59 +08:00
|
|
|
}
|
2020-05-30 04:23:24 +08:00
|
|
|
#endif
|
2020-04-02 00:10:59 +08:00
|
|
|
|
2020-06-29 03:18:52 +08:00
|
|
|
/* Main Algorithm:
|
2020-05-30 04:23:24 +08:00
|
|
|
* every week has 7 days hence, the start of next day would be
|
2020-06-29 03:18:52 +08:00
|
|
|
* modulo 7 add to this, the current start date and ensure the
|
|
|
|
* result is still modulo 7!
|
|
|
|
*/
|
2020-04-02 00:10:59 +08:00
|
|
|
start_day = ((days % 7) + start_day) % 7;
|
|
|
|
|
|
|
|
/* If start-day is a Sunday, increment counter */
|
|
|
|
if (start_day == 0)
|
2020-05-30 04:23:24 +08:00
|
|
|
count_sundays++;
|
2020-04-02 00:10:59 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-28 23:25:37 +08:00
|
|
|
printf(
|
|
|
|
"Total number of Sundays that happened on the 1st of a month in the "
|
|
|
|
"last century: %d\n",
|
|
|
|
count_sundays);
|
2020-04-02 00:10:59 +08:00
|
|
|
|
|
|
|
return 0;
|
2020-06-06 01:53:38 +08:00
|
|
|
}
|