子弹 の VISIONS

NEVER back down ~~

C++博客 首页 新随笔 联系 聚合 管理
  112 Posts :: 34 Stories :: 99 Comments :: 0 Trackbacks
#
// created by ztwaker on 2006-8-24
// Used for: count days
// No guarantees offered. Constructive comments to waker615@gmail.com

// 需求:计算两个日期之间相距天数。
// 应用:倒计时等

#include <iostream>
#include <algorithm>
namespace date {

//#define min(a, b) (((a) > (b)) ? (b) : (a))
//#define max(a, b) (((a) > (b)) ? (a) : (b))
	
	class Bad_date {}; // exception throw
	class Bad_year  : public Bad_date {};
	class Bad_month : public Bad_date {};
	class Bad_day   : public Bad_date {};
	
        static const int this_year = 2006;

	inline bool leapyear(int year)
	{
		//if (year > this_year)
		// 	throw Bad_year();
		
		return ((year % 100==0) ? (year % 400==0) : (year % 4==0));
	}
	
	inline int yday(int year)
	{
		//if (year > this_year)
		//	throw Bad_year();
		
		return (365 + leapyear(year));
	}
	
	inline int mday(int month, int year)
	{
		switch(month) {
		case 1: case 3: case 5: 
		case 7: case 8: case 10:
		case 12:
			return 31;
			
		case 4: case 6: case 9: 
		case 11: 
			return 30;
			
		case 2: 
			return 28 + leapyear(year);
			
		default:
			throw Bad_month();
		}
		
		throw Bad_month();
	}
	
	class Date{
		int y, m, d;
		
	public:
		Date(int year, int month, int day)
		{
			// check year/month/day available
			if (0 == year)
				throw Bad_year();

			if (1 > month || 12 < month)
				throw Bad_month();

			if (1 > day || mday(month, year) < day)
				throw Bad_day();
			
			y = year; m = month; d = day;
		}
		Date(const Date& rhs)
		{
			this->y = rhs.year();
			this->m = rhs.month();
			this->d = rhs.day();
		}		
		Date& operator=(const Date& rhs)
		{
			if (rhs == (*this))
				return (*this);
			
			this->y = rhs.year();
			this->m = rhs.month();
			this->d = rhs.day();
			
			return (*this);
		}
		int year () const { return y; }
		int month() const { return m; }
		int day  () const { return d; }
		
		int passday() const
		{
			int dpass = 0;
			for (int i = 1; i != m; ++i)
			{
				dpass += mday(i, y);
			}
			
			return (dpass + d);
		}

		friend std::ostream& operator<<(std::ostream& os, const Date& rhs);
		
		friend bool operator==(const Date& lhs, const Date& rhs);
		friend bool operator!=(const Date& lhs, const Date& rhs);
		friend bool operator< (const Date& lhs, const Date& rhs);
		friend bool operator> (const Date& lhs, const Date& rhs);
		friend bool operator<=(const Date& lhs, const Date& rhs);
		friend bool operator>=(const Date& lhs, const Date& rhs);
		
		friend int  distant(const Date& lhs, const Date& rhs);
	};

	inline std::ostream& operator<<(std::ostream& os, const Date& rhs)
	{
		return (os << rhs.y << "-" << rhs.m << "-" << rhs.d);
	}
	inline bool operator==(const Date& lhs, const Date& rhs)
	{
		return (lhs.y == rhs.y && lhs.m == rhs.m && lhs.d == rhs.d);
	}
	inline bool operator!=(const Date& lhs, const Date& rhs)
	{
		return (!(lhs==rhs));
	}
	inline bool operator<(const Date& lhs, const Date& rhs)
	{
		// compare year
		if (lhs.y < rhs.y)
		{
			return true;
		}
		else if (lhs.y == rhs.y)
		{
			// compare month
			if (lhs.m < rhs.m)
			{
				return true;
			}
			else if (lhs.m == rhs.m)
			{
				// compare day
				if (lhs.d < rhs.d)
					return true;
				else
					return false;
			}
			return false;
		}
		return false;
	}
	inline bool operator>(const Date& lhs, const Date& rhs)
	{
		// compare year
		if (lhs.y > rhs.y)
		{
			return true;
		}
		else if (lhs.y == rhs.y)
		{
			// compare month
			if (lhs.m > rhs.m)
			{
				return true;
			}
			else if (lhs.m == rhs.m)
			{
				// compare day
				if (lhs.d > rhs.d)
					return true;
				else
					return false;
			}
			return false;
		}
		return false;
	}
	inline bool operator<=(const Date& lhs, const Date& rhs)
	{
		return (!(lhs>rhs));
	}
	inline bool operator>=(const Date& lhs, const Date& rhs)
	{
		return (!(lhs<rhs));
	}
	
	int distant(const Date& lhs, const Date& rhs)
	{
		int distance = 0;
		const int lowyear  = std::min(lhs.y, rhs.y);
		const int highyear = std::max(lhs.y, rhs.y);
		for (int yn = lowyear; yn != highyear; ++yn)
		{
			distance += yday(yn);
		}
		
		return (
			distance - std::min(lhs, rhs).passday() 
			+ std::max(lhs, rhs).passday()
			);
	}
} // end of namespace date

void test()
{
	try {
		date::Date today(2006, 1, 29);    // 丙戌年春节
		date::Date deadline(2007, 2, 18); // 丁亥年春节
		std::cout << "From "<< today << " to " << deadline << " is ";
		std::cout << date::distant(today, deadline) << " days left.\n";
	}
	catch (date::Bad_year) {
		std::cout << "Bad year" << std::endl;
	}
	catch (date::Bad_month) {
		std::cout << "Bad month" << std::endl;
	}
	catch (date::Bad_day) {
		std::cout << "Bad day" << std::endl;
	}
	catch (date::Bad_date) {
		std::cout << "Bad date" << std::endl;
	}
}

int main()
{
    test();
    return 0;
}
posted on 2006-08-25 11:08 子弹のVISIONS 阅读(1712) 评论(6)  编辑 收藏 引用

Feedback

# re: [OPPD] 计算两个日期之间相距天数 2006-08-25 11:10 子弹
如果限于此需求,对于operator,仅重载">"也够用了;甚至
copy constructor [Date(const Date& rhs)] 和operator = 也可以不提供。  回复  更多评论
  

# re: [OPPD] 计算两个日期之间相距天数 2006-08-25 19:08 chenger
min和max两个宏用标准库提供的min和max算法代替更好些  回复  更多评论
  

# re: [OPPD] 计算两个日期之间相距天数 2006-08-28 09:06 子弹
@chenger
赞成。

之前使用宏的原因是:VC6对generaic algorithms min and max支持不好。为了方便编译而使用。

// 已经改用STL的算法min和max  回复  更多评论
  

# re: [OPPD] 计算两个日期之间相距天数 2006-08-28 15:43 chenger
听说VC6对C++标准的支持很不好,特别是template。居然连min max这样的简单模板函数都搞不定?
还是用g++或者vs2005得编译器好  回复  更多评论
  

# re: [OPPD] 计算两个日期之间相距天数 2006-08-29 08:58 子弹
@chenger

装了SP6之后对template支持稍好  回复  更多评论
  

# re: [OPPD] 计算两个日期之间相距天数 2006-08-30 16:25
scut师兄?支持一下!~  回复  更多评论
  


只有注册用户登录后才能发表评论。
网站导航: 博客园   IT新闻   BlogJava   博问   Chat2DB   管理