[comp.lang.c++] C++ Date Class Source

baker@csl.dl.nec.com (Larry Baker) (05/30/91)

Here's a little `date' class I wrote, using some Julian routines I got
out of the Simtel-20 archives.  I make no claims as to the validity
or stability of the two julian date routines - I haven't even looked up
the reference to the algorithm - but they got the dates and days right
in the simple test I ran (also attached).

Note that it doesn't parse date formats, and it doesn't deal with time.
Dealing with time is a relatively simple extention: extend the precision
of the julian integer to include however many units of precision - time
quanta - you need.  Time is actually a lot easier to deal with than dates,
if you are willing to grant that it is a continuous function within the scope
of a day.  This isn't really true, as any astronomer will tell you, but it works
for the rest of us.

There is one potential gotcha in the code, and it has to do with C++'s
handling of "temporary" objects.  On pages 267+, 300+ of
Ellis & Strourstrap, you'll find that the scope of "temporary objects"
per se is implementation dependent.  The use of complicated combinations
of the "+" and "-" operators,  e.g.  a = (d1 - d2) + d3, will generate
temporary values whose existence must be *at least* for the scope
of the expression.  This may not be true for some implementations
(it worked on all the versions of CFRONT I had access to, and Borland's
compiler, though I couldn't test Borland's very extensively).  E&S imply
that the temporary must exist for as long as it is bound to a reference, but
there are places where they seem to contradict themselves.

Note that the "+" and "-" operators each pass a Date object by *value*,
thus causing a call to the copy constructor and the creation of a temporary
value, which is returned by the operator.  This is expensive.  A better way
to do this is described on P. 300 of Ellis & Strourstrap: 

class Date {
	// ...
	friend Date operator + (int, Date&);
	friend Date operator + (Date&, int);
	// etc.
};

But I didn't do that here.

Cheers,

Larry Baker

File follows
////////////////
//
//       class Date         - perform simple date calculations and conversions.
//
//        Copyright (c) 1991, Larry Baker.  All Rights Reserved.  Free reproduction
//        and use of this code for any purpose is hereby granted, under the condition
//        that this copyright notice remain intact in this and all subsequent versions,
//        or any works derived from the computer source code provided herin.
//
////////////////

class Date {
public:
			Date(long jdays);
			Date(int month, int day, int year);
			Date(Date&);
			~Date();

	long		_month;
	long		_day;
	long		_year;

	long		_dow;		/* day of week */
	long		_jdays;		/* julian date day count */
};

Date& operator + (Date, int);		/* add days to a date */
Date& operator + (int, Date);		/* add days to a date */

Date& operator - (Date, int);		/* subtract days from a date */
Date& operator - (int, Date);		/* subtract days from a date */

int   operator - (Date&, Date&);	/* days between dates */

static long 	jday(int mon, int day, int year);
static long * 	jdate(long j);

Date& operator + (Date dt, int i)
{
	long *		ret;

	dt._jdays += i;

	ret = jdate(dt._jdays);
	dt._month = ret[0];
	dt._day = ret[1];
	dt._year = ret[2];
	dt._dow = ret[3];

	return dt;
}

Date& operator + (int i, Date dt)
{
	long *		ret;

	dt._jdays += i;

	ret = jdate(dt._jdays);
	dt._month = ret[0];
	dt._day = ret[1];
	dt._year = ret[2];
	dt._dow = ret[3];

	return dt;
}

Date& operator - (Date dt, int i)
{
	long *		ret;

	dt._jdays -= i;

	ret = jdate(dt._jdays);
	dt._month = ret[0];
	dt._day = ret[1];
	dt._year = ret[2];
	dt._dow = ret[3];

	return dt;
}

Date& operator - (int i, Date dt)
{
	long *		ret;

	dt._jdays -= i;

	ret = jdate(dt._jdays);
	dt._month = ret[0];
	dt._day = ret[1];
	dt._year = ret[2];
	dt._dow = ret[3];

	return dt;
}

int operator - (Date& d1, Date& d2)
{
	return d1._jdays - d2._jdays;
}


Date::Date(int month, int day, int year)
{
	long *		ret;

	_jdays = jday(month, day, year);
	_month = month;
	_day = day;
	_year = year;

	ret = jdate(_jdays);
	_dow = ret[3];
}

Date::Date(long jdays)
{
	long *	ret;

	_jdays = jdays;

	ret = jdate(_jdays);
	_month = ret[0];
	_day = ret[1];
	_year = ret[2];
	_dow = ret[3];
}

Date::Date(Date& dt)
{
	_jdays = dt._jdays;
	_month = dt._month;
	_day = dt._day;
	_year = dt._year;
	_dow = dt._dow;
}

Date::~Date()
{
}

/*
**	PD Programs retrieved from simtel-20 archives.
*/

/*
** Takes a date, and returns a Julian day. A Julian day is the number of
** days since some base date  (in the very distant past).
** Author: Robert G. Tantzen, translator: Nat Howard
** Translated from the algol original in Collected Algorithms of CACM
** (This and jdate are algorithm 199).
**
*/

static long jday(int mon, int day, int year)
{
	long m = mon, d = day, y = year;
	long c, ya, j;

	if(m > 2) m -= 3;
	else {
		m += 9;
		--y;
	}
	c = y/100L;
	ya = y - (100L * c);
	j = (146097L*c)/4L + (1461L*ya)/4L + (153L*m + 2L)/5L + d + 1721119L;
	return(j);
}

/* Julian date converter. Takes a julian date (the number of days since
** some distant epoch or other), and returns an int pointer to static space.
** ip[0] = month;
** ip[1] = day of month;
** ip[2] = year (actual year, like 1977, not 77 unless it was  77 a.d.);
** ip[3] = day of week (0->Sunday to 6->Saturday)
** These are Gregorian.
** Copied from Algorithm 199 in Collected algorithms of the CACM
** Author: Robert G. Tantzen, Translator: Nat Howard
*/

static long * jdate(long j)
{
	static long ret[4];

	long d, m, y;

	ret[3] = (j + 1L)%7L;
	j -= 1721119L;
	y = (4L * j - 1L)/146097L;
	j = 4L * j - 1L - 146097L * y;
	d = j/4L;
	j = (4L * d + 3L)/1461L;
	d = 4L * d + 3L - 1461L * j;
	d = (d + 4L)/4L;
	m = (5L * d - 3L)/153L;
	d = 5L * d - 3 - 153L * m;
	d = (d + 5L) / 5L;
	y = 100L * y + j;
	if(m < 10) 
		m += 3;
	else {
		m -= 9;
		++y;
	}
	ret[0] =  m;
	ret[1] = d;
	ret[2] = y;
	return(ret);
}

#include <stdio.h>

/* print a simple calendar for 1991 */

main()
{
	int		i;
	int		t;
	Date		today(1,1,1991);
	Date		day(1,1,1991);

	printf("%d\n\n", Date(1,1,1992) - Date(1,1,1991));

	for (i = 0; day._year != 1992; i++) {
		if (day._day == 1) {
			printf("\n\n");
			for (t = 0; t <= day._dow - 1; t++)
				printf("    ");
		}

		printf("%3d ", day._day);

		if (day._dow == 6)
			printf("\n");

		day = 1 + day;
	}
}


--
Larry Baker
NEC America C&C Software Laboratories, Irving (near Dallas), TX
baker@csl.dl.nec.com  cs.utexas.edu!necssd!baker