⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 calendarconvert.cpp

📁 公历(阳历)与农历(阴历)相互转换的元代
💻 CPP
字号:
// CalendarConvert.cpp: implementation of the CCalendarConvert class.
//
//////////////////////////////////////////////////////////////////////

#include "stdafx.h"
#include "CalendarConv.h"
#include "CalendarConvert.h"

#ifdef _DEBUG
#undef THIS_FILE
static char THIS_FILE[]=__FILE__;
#define new DEBUG_NEW
#endif

//////////////////////////////////////////////////////////////////////
// Construction/Destruction
//////////////////////////////////////////////////////////////////////

CCalendarConvert::CCalendarConvert()
{

}

CCalendarConvert::~CCalendarConvert()
{

}

//
// 得到表示农历通用表示的字串
//
CString CCalendarConvert::GetLunarString ( IN PSYSTEMTIME pSt, BOOL bEmbolism )
{
	TCHAR szNongli[30], szNongliDay[10],szShuXiang[10];
	/*天干名称*/
	const char *cTianGan[] = {"甲","乙","丙","丁","戊","己","庚","辛","壬","癸"};
	/*地支名称*/
	const char *cDiZhi[] = {"子","丑","寅","卯","辰","巳","午",	"未","申","酉","戌","亥"};
	/*属相名称*/
	const char *cShuXiang[] = {"鼠","牛","虎","兔","龙","蛇","马","羊","猴","鸡","狗","猪"};
	/*农历日期名*/
	const char *cDayName[] = { "*","初一","初二","初三","初四","初五",
		"初六","初七","初八","初九","初十",
		"十一","十二","十三","十四","十五",
		"十六","十七","十八","十九","二十",
		"廿一","廿二","廿三","廿四","廿五",       
		"廿六","廿七","廿八","廿九","三十"};
	/*农历月份名*/
	const char *cMonName[] = {"*","正","二","三","四","五","六", "七","八","九","十","十一","腊"};

	/*--生成农历天干、地支、属相 ==> wNongli--*/
	int nShuXiang = ((pSt->wYear - 4) % 60) % 12;
	if ( nShuXiang < 0 || nShuXiang >= sizeof(cShuXiang)/sizeof(cShuXiang[0]) )
		return "Error Year";
	wsprintf(szShuXiang,"%s",cShuXiang[nShuXiang]);
	int nTianGan = ((pSt->wYear - 4) % 60) % 10;
	if ( nTianGan < 0 || nTianGan >= sizeof(cTianGan)/sizeof(cTianGan[0]) )
		return "Error Year";
	int nDiZhi = ((pSt->wYear - 4) % 60) % 12;
	if ( nDiZhi < 0 || nDiZhi >= sizeof(cDiZhi)/sizeof(cDiZhi[0]) )
		return "Error Year";
	wsprintf(szNongli,"%s(%s%s)年",szShuXiang,cTianGan[nTianGan],cDiZhi[nDiZhi]);
	
	/*--生成农历月、日 ==> wNongliDay--*/
	if ( pSt->wMonth < 0 || pSt->wMonth >= sizeof(cMonName)/sizeof(cMonName[0]) )
		return "Error Month";
	if ( bEmbolism )
	{
		wsprintf(szNongliDay,"闰%s",cMonName[pSt->wMonth]);
	}
	else
	{
		strcpy(szNongliDay,cMonName[pSt->wMonth]);
	}
	
	strcat(szNongliDay,"月");
	if ( pSt->wDay < 0 || pSt->wDay >= sizeof(cDayName)/sizeof(cDayName[0]) )
		return "Error Day";
	strcat(szNongliDay,cDayName[pSt->wDay]);
	return strcat(szNongli,szNongliDay);
}

//
// 公历转农历
//
BOOL CCalendarConvert::Gregorian2Lunar ( IN PSYSTEMTIME pSt, OUT PSYSTEMTIME pDst, OUT BOOL *pEmbolism )
{
	if ( !pSt || !pDst || !pEmbolism ) return FALSE;

	SYSTEMTIME Dst = {0};	
	/*公历每月前面的天数*/
	const int wMonthAdd[12] = {0,31,59,90,120,151,181,212,243,273,304,334};
	/*农历数据*/
	const int wNongliData[100] = {2635,333387,1701,1748,267701,694,2391,133423,1175,396438
		,3402,3749,331177,1453,694,201326,2350,465197,3221,3402
		,400202,2901,1386,267611,605,2349,137515,2709,464533,1738
		,2901,330421,1242,2651,199255,1323,529706,3733,1706,398762
		,2741,1206,267438,2647,1318,204070,3477,461653,1386,2413
		,330077,1197,2637,268877,3365,531109,2900,2922,398042,2395
		,1179,267415,2635,661067,1701,1748,398772,2742,2391,330031
		,1175,1611,200010,3749,527717,1452,2742,332397,2350,3222
		,268949,3402,3493,133973,1386,464219,605,2349,334123,2709
		,2890,267946,2773,592565,1210,2651,395863,1323,2707,265877};
	static int wCurYear,wCurMonth,wCurDay;
	static int nTheDate,nIsEnd,m,k,n,i,nBit;
	/*---取当前公历年、月、日---*/
	wCurYear = pSt->wYear;
	wCurMonth = pSt->wMonth;
	wCurDay = pSt->wDay;
	/*---计算到初始时间1921年2月8日的天数:1921-2-8(正月初一)---*/
	nTheDate = (wCurYear - 1921) * 365 + (wCurYear - 1921) / 4 + wCurDay + wMonthAdd[wCurMonth - 1] - 38;
	if ( nTheDate < 0 ) return FALSE;
	if((!(wCurYear % 4)) && (wCurMonth > 2))
		nTheDate = nTheDate + 1;
	
	/*--计算农历天干、地支、月、日---*/
	nIsEnd = 0;
	m = 0;
	while(nIsEnd != 1)
	{
		if(wNongliData[m] < 4095)
			k = 11;
		else
			k = 12;
		n = k;
		while(n>=0)
		{
			//获取wNongliData(m)的第n个二进制位的值
			nBit = wNongliData[m];
			for(i=1;i<n+1;i++)
				nBit = nBit/2;
			
			nBit = nBit % 2;
			
			if (nTheDate <= (29 + nBit))
			{
				nIsEnd = 1;
				break;
			}
			
			nTheDate = nTheDate - 29 - nBit;
			n = n - 1;
		}
		if(nIsEnd)
			break;
		m = m + 1;
	}
	wCurYear = 1921 + m;
	wCurMonth = k - n + 1;
	wCurDay = nTheDate;
	if ( wCurDay < 0 ) return FALSE;

	if (k == 12)
	{
		if (wCurMonth == wNongliData[m] / 65536 + 1)
			wCurMonth = 1 - wCurMonth;
		else if (wCurMonth > wNongliData[m] / 65536 + 1)
			wCurMonth = wCurMonth - 1;
	}
	
	Dst.wYear = (USHORT)wCurYear;
	if (wCurMonth < 1)
	{
		Dst.wMonth = -wCurMonth;
		*pEmbolism = TRUE;
	}
	else
	{
		Dst.wMonth = wCurMonth;
		*pEmbolism = FALSE;
	}
	Dst.wDay = wCurDay;
	memcpy ( pDst, &Dst, sizeof(SYSTEMTIME) );

	return TRUE;
}

//
// 农历转公历
//
BOOL CCalendarConvert::Lunar2Gregorian ( IN PSYSTEMTIME pSt, IN BOOL bEmbolism, OUT PSYSTEMTIME pDst )
{
	if ( !pSt || !pDst ) return FALSE;
	COleDateTimeSpan Span(1,0,0,0);
	COleDateTime tStart ( *pSt );
	for ( int i=0; i<100; i++ )
	{
		SYSTEMTIME Src = {0};
		SYSTEMTIME Dst = {0};
		BOOL bDstEmbolism = FALSE;
		if ( !tStart.GetAsSystemTime ( Src ) )
			return FALSE;

		if ( !Gregorian2Lunar ( &Src, &Dst, &bDstEmbolism ) )
			return FALSE;
		if ( Dst.wYear==pSt->wYear && Dst.wMonth==pSt->wMonth && Dst.wDay==pSt->wDay && bEmbolism==bDstEmbolism )
		{
			memcpy ( pDst, &Src, sizeof(SYSTEMTIME) );
			return TRUE;
		}
		tStart+=Span;
	}

	return FALSE;
}

COleDateTime CCalendarConvert::ConvertSysTime2OleDateTime(PSYSTEMTIME pSt)
{
	ASSERT ( pSt );
	COleDateTime t ( *pSt );
	return t;
}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -