CalendarEx.cpp 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309
  1. #include "stdafx.h"
  2. #include "CalendarEx.h"
  3. static const int BEGIN_YEAR = 1901;
  4. static const int NUMBER_YEAR = 199;
  5. //使用比特位记录每年的情况
  6. //0~4 共5bit 春节号数;
  7. //5~6 共2bit 春节月份
  8. //7~19 共13bit 13个月的大小月情况(如果无闰月,最后位无效),大月为1,小月为0
  9. //20~23 共4bit 记录闰月的月份,如果没有闰月为0
  10. //0x955ABF 2014年, 农9月有闰月, 公历1月31号为春节;
  11. // 1001 0101010110101 01 11111
  12. // 闰月月份 7-19.13个月 春月 春节日期;
  13. ////1001 0101010110101 01 11111
  14. //0X252ABF
  15. //1001010010101010111111
  16. static const unsigned int LUNAR_YEARS[199] =
  17. {
  18. 0x04AE53, 0x0A5748, 0x5526BD, 0x0D2650, 0x0D9544, 0x46AAB9, 0x056A4D, 0x09AD42, 0x24AEB6, 0x04AE4A, //1901-1910
  19. 0x6A4DBE, 0x0A4D52, 0x0D2546, 0x5D52BA, 0x0B544E, 0x0D6A43, 0x296D37, 0x095B4B, 0x749BC1, 0x049754, //1911-1920
  20. 0x0A4B48, 0x5B25BC, 0x06A550, 0x06D445, 0x4ADAB8, 0x02B64D, 0x095742, 0x2497B7, 0x04974A, 0x664B3E, //1921-1930
  21. 0x0D4A51, 0x0EA546, 0x56D4BA, 0x05AD4E, 0x02B644, 0x393738, 0x092E4B, 0x7C96BF, 0x0C9553, 0x0D4A48, //1931-1940
  22. 0x6DA53B, 0x0B554F, 0x056A45, 0x4AADB9, 0x025D4D, 0x092D42, 0x2C95B6, 0x0A954A, 0x7B4ABD, 0x06CA51, //1941-1950
  23. 0x0B5546, 0x555ABB, 0x04DA4E, 0x0A5B43, 0x352BB8, 0x052B4C, 0x8A953F, 0x0E9552, 0x06AA48, 0x7AD53C, //1951-1960
  24. 0x0AB54F, 0x04B645, 0x4A5739, 0x0A574D, 0x052642, 0x3E9335, 0x0D9549, 0x75AABE, 0x056A51, 0x096D46, //1961-1970
  25. 0x54AEBB, 0x04AD4F, 0x0A4D43, 0x4D26B7, 0x0D254B, 0x8D52BF, 0x0B5452, 0x0B6A47, 0x696D3C, 0x095B50, //1971-1980
  26. 0x049B45, 0x4A4BB9, 0x0A4B4D, 0xAB25C2, 0x06A554, 0x06D449, 0x6ADA3D, 0x0AB651, 0x093746, 0x5497BB, //1981-1990
  27. 0x04974F, 0x064B44, 0x36A537, 0x0EA54A, 0x86B2BF, 0x05AC53, 0x0AB647, 0x5936BC, 0x092E50, 0x0C9645, //1991-2000
  28. 0x4D4AB8, 0x0D4A4C, 0x0DA541, 0x25AAB6, 0x056A49, 0x7AADBD, 0x025D52, 0x092D47, 0x5C95BA, 0x0A954E, //2001-2010
  29. 0x0B4A43, 0x4B5537, 0x0AD54A, 0x955ABF, 0x04BA53, 0x0A5B48, 0x652BBC, 0x052B50, 0x0A9345, 0x474AB9, //2011-2020
  30. 0x06AA4C, 0x0AD541, 0x24DAB6, 0x04B64A, 0x69573D, 0x0A4E51, 0x0D2646, 0x5E933A, 0x0D534D, 0x05AA43, //2021-2030
  31. 0x36B537, 0x096D4B, 0xB4AEBF, 0x04AD53, 0x0A4D48, 0x6D25BC, 0x0D254F, 0x0D5244, 0x5DAA38, 0x0B5A4C, //2031-2040
  32. 0x056D41, 0x24ADB6, 0x049B4A, 0x7A4BBE, 0x0A4B51, 0x0AA546, 0x5B52BA, 0x06D24E, 0x0ADA42, 0x355B37, //2041-2050
  33. 0x09374B, 0x8497C1, 0x049753, 0x064B48, 0x66A53C, 0x0EA54F, 0x06B244, 0x4AB638, 0x0AAE4C, 0x092E42, //2051-2060
  34. 0x3C9735, 0x0C9649, 0x7D4ABD, 0x0D4A51, 0x0DA545, 0x55AABA, 0x056A4E, 0x0A6D43, 0x452EB7, 0x052D4B, //2061-2070
  35. 0x8A95BF, 0x0A9553, 0x0B4A47, 0x6B553B, 0x0AD54F, 0x055A45, 0x4A5D38, 0x0A5B4C, 0x052B42, 0x3A93B6, //2071-2080
  36. 0x069349, 0x7729BD, 0x06AA51, 0x0AD546, 0x54DABA, 0x04B64E, 0x0A5743, 0x452738, 0x0D264A, 0x8E933E, //2081-2090
  37. 0x0D5252, 0x0DAA47, 0x66B53B, 0x056D4F, 0x04AE45, 0x4A4EB9, 0x0A4D4C, 0x0D1541, 0x2D92B5 //2091-2099
  38. };
  39. CCalendarEx::CCalendarEx()
  40. {
  41. }
  42. CCalendarEx::~CCalendarEx()
  43. {
  44. }
  45. bool CCalendarEx::IsValidLunanrDate(int lunar_year, int lunar_month, int lunar_day)
  46. {
  47. //越界检查,如果越界,返回无效日期;
  48. if (lunar_year <= BEGIN_YEAR || lunar_year > BEGIN_YEAR + NUMBER_YEAR - 1)
  49. return FALSE;
  50. if (lunar_month < 1 || lunar_month > 12)
  51. return FALSE;
  52. // 1.农历年是否有闰月;
  53. int nYearIndex = lunar_year - BEGIN_YEAR;
  54. int leap_month = (LUNAR_YEARS[nYearIndex] >> 20) & 0xF;
  55. // 2.月份大小月天数;
  56. int month_day = 0;
  57. if ((LUNAR_YEARS[nYearIndex] >> (7 + 13 - lunar_month)) & 0x1)
  58. month_day = 30;
  59. else
  60. month_day = 29;
  61. if (lunar_day < 0 || lunar_day > month_day)
  62. return FALSE;
  63. return TRUE;
  64. }
  65. //计算这个公历日期是一年中的第几天
  66. int CCalendarEx::DayOfSolarYear(int year, int month, int day)
  67. {
  68. //为了提高效率,记录每月一日是一年中的第几天;
  69. static const int NORMAL_YDAY[12] = { 0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334 };
  70. //闰年的情况;
  71. static const int LEAP_YDAY[12] = { 0, 31, 60, 91, 121, 152, 182, 213, 244, 274, 305, 335 };
  72. const int *t_year_yday_ = NORMAL_YDAY;
  73. //判断是否是公历闰年;
  74. if (year % 4 == 0)
  75. {
  76. if (year % 100 != 0)
  77. t_year_yday_ = LEAP_YDAY;
  78. if (year % 400 == 0)
  79. t_year_yday_ = LEAP_YDAY;
  80. }
  81. return t_year_yday_[month - 1] + day;
  82. }
  83. Date CCalendarEx::SolarToLuanrDate(int solar_year, int solar_month, int solar_day)
  84. {
  85. Date luanr_date;
  86. luanr_date.year = solar_year;
  87. luanr_date.month = 0;
  88. luanr_date.day = 0;
  89. luanr_date.leap = false;
  90. //越界检查,如果越界,返回无效日期;
  91. if (solar_year <= BEGIN_YEAR || solar_year > BEGIN_YEAR + NUMBER_YEAR - 1)
  92. return luanr_date;
  93. int year_index = solar_year - BEGIN_YEAR;
  94. //计算春节的公历日期;
  95. int spring_ny_month = (LUNAR_YEARS[year_index] & 0x60) >> 5;
  96. int spring_ny_day = (LUNAR_YEARS[year_index] & 0x1F);
  97. //计算今天是公历年的第几天;
  98. int today_solar_yd = DayOfSolarYear(solar_year, solar_month, solar_day);
  99. //计算春节是公历年的第几天;
  100. int spring_ny_yd = DayOfSolarYear(solar_year, spring_ny_month, spring_ny_day);
  101. //计算今天是农历年的第几天 = 农历今天在的公历第几天 - 今年农历春节在公历的第几天;
  102. int today_luanr_yd = today_solar_yd - spring_ny_yd + 1;
  103. //如果今天在春节的前面,重新计算today_luanr_yd;
  104. if (today_luanr_yd <= 0)
  105. {
  106. //农历年比当前公历年小1;
  107. year_index--;
  108. luanr_date.year--;
  109. //越界,返回无效日期;
  110. if (year_index < 0)
  111. return luanr_date;
  112. // 去年农历春节月;
  113. spring_ny_month = (LUNAR_YEARS[year_index] & 0x60) >> 5;
  114. // 去年农历春节日;
  115. spring_ny_day = (LUNAR_YEARS[year_index] & 0x1F);
  116. // 去年公历年共多少天;
  117. int year_total_day = DayOfSolarYear(luanr_date.year, 12, 31);
  118. spring_ny_yd = DayOfSolarYear(solar_year, spring_ny_month, spring_ny_day);
  119. // 计算今天是农历年的第几天;
  120. today_luanr_yd = today_solar_yd + year_total_day - spring_ny_yd + 1;
  121. }
  122. int luanr_month = 1;
  123. //计算月份和日期;
  124. for (; luanr_month <= 13; luanr_month++)
  125. {
  126. int month_day = 29;
  127. if ((LUNAR_YEARS[year_index] >> (7 + 13 - luanr_month)) & 0x1)
  128. month_day = 30;
  129. if (today_luanr_yd <= month_day)
  130. break;
  131. else
  132. today_luanr_yd -= month_day;
  133. }
  134. luanr_date.day = today_luanr_yd;
  135. //处理闰月;
  136. int leap_month = (LUNAR_YEARS[year_index] >> 20) & 0xf;
  137. if (leap_month > 0 && leap_month < luanr_month)
  138. {
  139. luanr_month--;
  140. //如果当前月为闰月,设置闰月标志;
  141. if (luanr_month == leap_month)
  142. luanr_date.leap = true;
  143. }
  144. assert(leap_month <= 12);
  145. luanr_date.month = luanr_month;
  146. return luanr_date;
  147. }
  148. CString CCalendarEx::GetLunarDate(int solar_year, int solar_month, int solar_day)
  149. {
  150. Date luanr_date;
  151. luanr_date.year = solar_year;
  152. luanr_date.month = 0;
  153. luanr_date.day = 0;
  154. luanr_date.leap = false;
  155. //越界检查,如果越界,返回无效日期;
  156. if (solar_year <= BEGIN_YEAR || solar_year > BEGIN_YEAR + NUMBER_YEAR - 1)
  157. return _T("1900-01-01");
  158. int year_index = solar_year - BEGIN_YEAR;
  159. //计算春节的公历日期;
  160. int spring_ny_month = (LUNAR_YEARS[year_index] & 0x60) >> 5;
  161. int spring_ny_day = (LUNAR_YEARS[year_index] & 0x1F);
  162. //计算今天是公历年的第几天;
  163. int today_solar_yd = DayOfSolarYear(solar_year, solar_month, solar_day);
  164. //计算春节是公历年的第几天;
  165. int spring_ny_yd = DayOfSolarYear(solar_year, spring_ny_month, spring_ny_day);
  166. //计算今天是农历年的第几天 = 农历今天在的公历第几天 - 今年农历春节在公历的第几天;
  167. int today_luanr_yd = today_solar_yd - spring_ny_yd + 1;
  168. //如果今天在春节的前面,重新计算today_luanr_yd;
  169. if (today_luanr_yd <= 0)
  170. {
  171. //农历年比当前公历年小1;
  172. year_index--;
  173. luanr_date.year--;
  174. //越界,返回无效日期;
  175. if (year_index < 0)
  176. return _T("1900-01-01");
  177. // 去年农历春节月;
  178. spring_ny_month = (LUNAR_YEARS[year_index] & 0x60) >> 5;
  179. // 去年农历春节日;
  180. spring_ny_day = (LUNAR_YEARS[year_index] & 0x1F);
  181. // 去年公历年共多少天;
  182. int year_total_day = DayOfSolarYear(luanr_date.year, 12, 31);
  183. spring_ny_yd = DayOfSolarYear(solar_year, spring_ny_month, spring_ny_day);
  184. // 计算今天是农历年的第几天;
  185. today_luanr_yd = today_solar_yd + year_total_day - spring_ny_yd + 1;
  186. }
  187. int luanr_month = 1;
  188. //计算月份和日期;
  189. for (; luanr_month <= 13; luanr_month++)
  190. {
  191. int month_day = 29;
  192. if ((LUNAR_YEARS[year_index] >> (7 + 13 - luanr_month)) & 0x1)
  193. month_day = 30;
  194. if (today_luanr_yd <= month_day)
  195. break;
  196. else
  197. today_luanr_yd -= month_day;
  198. }
  199. luanr_date.day = today_luanr_yd;
  200. //处理闰月;
  201. int leap_month = (LUNAR_YEARS[year_index] >> 20) & 0xf;
  202. if (leap_month > 0 && leap_month < luanr_month)
  203. {
  204. luanr_month--;
  205. //如果当前月为闰月,设置闰月标志;
  206. if (luanr_month == leap_month)
  207. luanr_date.leap = true;
  208. }
  209. assert(leap_month <= 12);
  210. luanr_date.month = luanr_month;
  211. CString strSolarDate;
  212. strSolarDate.Format(_T("%04d-%02d-%02d"), luanr_date.year, luanr_date.month, luanr_date.day);
  213. return strSolarDate;
  214. }
  215. CString CCalendarEx::LuanrToSolarDate(int lunar_year, int lunar_month, int lunar_day, BOOL bLeap/*=FALSE*/)
  216. {
  217. //越界检查,如果越界,返回无效日期;
  218. if (lunar_year <= BEGIN_YEAR || lunar_year > BEGIN_YEAR + NUMBER_YEAR - 1)
  219. return _T("");
  220. // 1.判断该农历年是否有闰月;
  221. int nYearIndex = lunar_year - BEGIN_YEAR;
  222. int leap_month = (LUNAR_YEARS[nYearIndex] >> 20) & 0xF;
  223. CString strSolarDate = _T("1900-01-01");
  224. COleDateTime Solar;
  225. Solar.SetDateTime(lunar_year, lunar_month, lunar_day, 0, 0, 0);
  226. COleDateTimeSpan Span(1, 0, 0, 0);
  227. Date Luanr_date;
  228. for (int i = 0; i < 120; i++)
  229. {
  230. Luanr_date = SolarToLuanrDate(Solar.GetYear(), Solar.GetMonth(), Solar.GetDay());
  231. if (Luanr_date.year == lunar_year && Luanr_date.month == lunar_month && Luanr_date.day == lunar_day)
  232. {
  233. if (bLeap && leap_month == lunar_month)
  234. {
  235. bLeap = FALSE;
  236. Solar += Span;
  237. continue;
  238. }
  239. else
  240. {
  241. strSolarDate.Format(_T("%4d-%02d-%02d"), Solar.GetYear(), Solar.GetMonth(), Solar.GetDay());
  242. break;
  243. }
  244. }
  245. Solar += Span;
  246. }
  247. // 如果没有找,那么就没有这个农历日期;2012-06-30号,这个是不存在的农历日期.
  248. // 不存在,则返回1900-01-01这个初值:设置strSolarDate = _T("1900-01-01");
  249. return strSolarDate;
  250. }