传奇三端互通新开服网站百度非企推广开户
news/
2025/9/25 18:58:52/
文章来源:
传奇三端互通新开服网站,百度非企推广开户,旅游网站后台模板下载,建站广告这个算法来自LINUX的源码#xff0c;下面带有大神的解析#xff0c;自己在RTC实验中也使用了#xff0c;不用月份表#xff0c;润平年的处理#xff0c;几行就可得出结果#xff0c;以下是程序和大神的解析Linux源码中的mktime算法解析我们知道#xff0c;从CMOS中读出来…这个算法来自LINUX的源码下面带有大神的解析自己在RTC实验中也使用了不用月份表润平年的处理几行就可得出结果以下是程序和大神的解析Linux源码中的mktime算法解析我们知道从CMOS中读出来的系统时间并不是time_t类型而是类似于struct tm那样年月日时分秒是分开存储的。那么要把它转化为系统便于处理的time_t类型就需要算法进行转换。我们都知道我们的公历还是比较复杂的有大月小月有闰年非闰年处理起来会很麻烦。但是Linux的源代码仅仅用了短短的几行就完成了这个复杂的转换(Gauss算法)实在令人惊奇。话不多说先看源代码include/linux/time.hstatic inline unsigned long mktime (unsigned int year, unsigned int mon,unsigned int day, unsigned int hour,unsigned int min, unsigned int sec){if (0 (int) (mon - 2)){ /**//* 1..12 - 11,12,1..10 */mon 12; /**//* Puts Feb last since it has leap day */year - 1;}return ((((unsigned long) (year/4 - year/100 year/400 367*mon/12 day) year*365 - 719499)*24 hour /**//* now have hours */)*60 min /**//* now have minutes */)*60 sec; /**//* finally seconds */}看上去令人眼花缭乱毫无头绪。下面就让我们对该算法作具体的分析。先不看前面的直接看return那句该式整体上具有这样的结构T ((X * 24 hour) * 60 min) * 60 sec这说明该算法是先算出从1970年1月1日开始的天数X再进而求出具体的时间值T的。因此我们重点看如何求天数X。也就是X year/4 - year/100 year/400 367*mon/12 day year*365 - 719499这一部分。首先可以将上式拆成Y year / 4 - year / 100 year / 400Z 367 * mon / 12W year * 365 dayX Y Z W - 719499Y很简单它计算了从公元元年到所求年份为止所有的闰年数。从W式看出该算法先假设所有年都是正常年(365天)再加上闰年额外的天数(式Y)。到现在为止都算简单关键是Z式和X式中的那个常数719499是怎么回事似乎莫名其妙。还有就是它们和return语句前面的那个if判断有什么关系呢首先要澄清一点常数719499并不是像很多人说的那样是0001年1月1日到1970年1月1日所经历的天数。不信你可以随手写个脚本将得到正确的数字719162天。显然719162和719499是有关系的。我们把注意力放在那个if语句上mon - 2;if (mon 0) ...{mon 12;year--;}很明显它是想把1月和2月当作上一年年底的最后两个月让3月作为一年的第一个月。这样一来我们可以尽量少的被闰年所影响。按照这个假设让我们先不管Z式是怎么来的看看0001年1月1日时Y Z W等于什么mon 1月变成上一年(公元前0001年)的11月year减一后变成了0因此Y 0Z 367 * 11 / 12 336W 1 0 * 365 1Y Z W 337。337这个数正好等于719499 - 719162换句话说它是对上述假设所做的补正于是这些式子就变成了Y year / 4 - year / 100 year / 400Z 367 * mon / 12V Z - 337W year * 365 dayX Y W V - 719162再来看式Z这个式子表面看不出任何名堂367这个数字显然很是奇怪。那让我们穷举一下mon看看这个式子算出的都是些什么值吧mon Z1 302 613 914 1225 1526 1837 2148 2449 27510 30511 33612 367似乎看出了什么再让我们把相邻的两个mon的Z做一下减法看看mon dZ1 302 313 304 315 306 317 318 309 3110 3011 3112 31闻出点味道了吧很象大小月的规则。让我们回想起那个if语句作了什么它把1月2月变成了11月和12月3月变成了1月还原一下看看mon org-mon dZ1 3 302 4 313 5 304 6 315 7 306 8 317 9 318 10 309 11 3110 12 3011 1 3112 2 31怎么本来应该是大月的3月成了30天那好我们想想这个原理假设今天是1月1日那你能说你今年已经过了31天了么显然不是1月还没过我们不能把它算进去。这里同然我们从4月看起如果今天是愚人节那么距离3月1日我们经过了31天。就像前面说的我们假设一年是从3月开始到次年的2月结束。按照这个规则整个式子里有问题的只有3月理论上这里应该是0但是这没关系我们把它减去就行了于是变成Z 367 * mon / 12 - 30V Z - 307回头看看W式year * 365但是按照上面的理论没过完的这一年不应该加进去所以这里把它减去再和V式合并V Z 58W (year - 1) * 365 day我们记得这个算法的一年是从3月开始的因此少算了公元元年的1月和2月的天数31 28 59天(公元元年是正常年)V Z 59 - 1那么最后的这个减1是什么还是上面那个原理今天还没过就不应该把它算进去综上整个算法就明朗了主要难于理解的是那个3月开始的假设以及367 * mon / 12会产生类似大小月的序列。最后把这些式子整理并罗列一下做为本文的结束Y (year - 1) * 365 year / 4 - year / 100 year / 400M 367 * mon / 12 - 30 59D day - 1X Y M D - 719162T ((X * 24 hour) * 60 min) * 60 sec标签12,秒数,经历,31,30,1970,mon,year,367来源 https://blog.csdn.net/wang93IT/article/details/79744711
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/917392.shtml
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!