Linux-0.11 kernel_mktime函数分析

kernel_mktime函数在kernel/mktime.c文件中,代码如下:
/*
 *  linux/kernel/mktime.c
 *
 *  (C) 1991  Linus Torvalds
 */

#include <time.h>

/*
 * This isn't the library routine, it is only used in the kernel.
 * as such, we don't care about years<1970 etc, but assume everything
 * is ok. Similarly, TZ etc is happily ignored. We just do everything
 * as easily as possible. Let's find something public for the library
 * routines (although I think minix times is public).
 */
/*
 * PS. I hate whoever though up the year 1970 - couldn't they have gotten
 * a leap-year instead? I also hate Gregorius, pope or no. I'm grumpy.
 */
#define MINUTE 60
#define HOUR (60*MINUTE)
#define DAY (24*HOUR)
#define YEAR (365*DAY)

/* interestingly, we assume leap-years */
static int month[12] = {
	0,
	DAY*(31),
	DAY*(31+29),
	DAY*(31+29+31),
	DAY*(31+29+31+30),
	DAY*(31+29+31+30+31),
	DAY*(31+29+31+30+31+30),
	DAY*(31+29+31+30+31+30+31),
	DAY*(31+29+31+30+31+30+31+31),
	DAY*(31+29+31+30+31+30+31+31+30),
	DAY*(31+29+31+30+31+30+31+31+30+31),
	DAY*(31+29+31+30+31+30+31+31+30+31+30)
};

long kernel_mktime(struct tm * tm)
{
	long res;
	int year;

	year = tm->tm_year - 70;
/* magic offsets (y+1) needed to get leapyears right.*/
	res = YEAR*year + DAY*((year+1)/4);
	res += month[tm->tm_mon];
/* and (y+2) here. If it wasn't a leap-year, we have to adjust */
	if (tm->tm_mon>1 && ((year+2)%4))
		res -= DAY;
	res += DAY*(tm->tm_mday-1);
	res += HOUR*tm->tm_hour;
	res += MINUTE*tm->tm_min;
	res += tm->tm_sec;
	return res;
}
mktime函数用于将struct tm类型的时间转换成time_t类型的一个值,time_t记录的是自1970年1月一日凌晨以来的秒数,而time_t类型实际上就是long int,所以这里的返回值直接写的是long类型。而struct tm记录的是1900年1月1日以来的时间,所以说这里需要注意有两个时间基准点,一个是1970年1月1日凌晨,另一个就是1900年1月1日凌晨,为什么不是一个时间基准点呢,Why?

既然struct tm记录的是1900年以来的时间,在计算year时减去70,那么得到的就是自1970年以来的年份数。

然后根据这个年份数计算出有多少秒,注意这里一年的天数是按365天计算的,所以后面还考虑的闰年这种情况,如果是闰年还需要加上相应的天数,在计算闰年的天数使用使用的是(year+1)/4的方式,那么这里为什么这样写呢?我们知道1972年是闰年,然后1976、1980、1984...这样的规律,那么对应这里的year值就是2、6、10、14...,加1之后就变成了3、7、11、15,对应的年份实际上是1973、1977、1981、1985...,所以说这里计算闰年天数时是从闰年的后1年开始计算的,闰年当年的那一天这里并没有考虑在内,那么在哪里考虑呢,在后面。

然后计算月份经历的天数,注意这里的天数实际上是按闰年的天数来算的,也就是我前面所说的闰年这种情况这里来考虑,也就说如果是1972年,在计算年份那里是不会加上闰年这一天的,那么这里会加上闰年这一天。

但是这里又会带来一个问题,前面计算月份天数是按闰年天数计算的,如果是平年,那么这里会多一天,所以这里也会判断如果月份大于2月并且是平年,那么这里会减去一天。

然后再来计算月份中的天数,注意这里减去了1,即当天并为参与计算(那是肯定的,因为并不是完整的一天),然后加上小时数、分钟数、秒数,得到最终的结果。

从这里我们看出,主要的难点或者问题点在于计算闰年天数,而这里在计算闰年天数时也并没有考虑世纪年这种情况(或许连linus本人也没料到会使用到那一天吧)。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值