设为首页收藏本站

爱吱声

 找回密码
 注册
搜索
查看: 7157|回复: 33
打印 上一主题 下一主题

[科普知识] 程序员的历法

[复制链接]
  • TA的每日心情
    开心
    2025-9-20 01:30
  • 签到天数: 2861 天

    [LV.Master]无

    跳转到指定楼层
    楼主
    发表于 2015-2-1 18:10:26 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
    本帖最后由 heinsect 于 2015-2-1 18:13 编辑 0 E: D8 l; j$ b: f" T9 a! s

    6 c5 _, \( \, }- I, Y程序员计算日期是用儒略日的。8 D. K+ _: i% Y, ~# A* y4 Y
      M+ ~( v7 j! M
    儒略日中的儒略和儒略历中的儒略的关系嘛,只是因为儒略日的发明人的爷爷葱白凯撒,给儿子用了大帝的名字。刚好儒略历也用了大帝的名字。
    $ V" t' O8 `9 L5 l* K' q7 f1 S1 q) ~* I, u% G  r& q1 [
    单用日期的话,儒略日是从某天开始的一个日的整数。两个儒略日的差值就是相差的天数。这样想计算两个日期间的差值,计算星期几就很简单了。至于那个开始日期,儒略日的零点,用起来的时候谁也不关心,我就不讲了。
    + o2 w0 N& w# M9 R! a: s1 Q
    9 s4 {; d+ A# c8 S从格里高利历日期算儒略日(JDN)的公式是这个样子的:
    . g$ m9 Q3 q) m: S$ D7 F$ x; j2 [" }; ~
    先要改一下年月:# u& C0 v4 V2 X8 e" p% ?5 o
    ! L) w; [* r0 S$ E7 f% p) L5 b
    9 Y, S0 h1 q) S8 g7 Z& V! D
    上面这组公式的结果呢,差不多是这个意思:3 T# O/ y  p+ o# N+ [: G  T% y
    三月 m = 0, y=y
    & W9 y! r( f( J/ ^/ O...% `% V; m% `' ~  o( [; L
    十二月 m=9, y=y1 F; r, E* C% l% e6 V  F9 L+ ^; A
    一月 m = 10, y=y-1' d( X' |3 q/ C8 s5 o
    二月 m = 11, y=y-1
    ( i8 G% \) H3 f1 k  k; z- {/ B
    8 F* k8 V% L7 D3 M2 k那个4800,是个计算零点,大概在公元前4801年,是和前面所说的那个零点相关的。
    / ]& ]/ O+ E7 z2 |; s5 S然后计算儒略日的公式长是这个样子的:
    ( x+ z) L& ?8 v, K2 Q2 U( Q; _0 p6 b/ ~& T
    9 ], M; `. B8 `+ r

    ' D" X( r( A4 Q- G& Z6 C) w这个公式中最巧的部分是 floor((153m+2)/5), 做出的效果嘛,看这个表:; ?3 I$ F- j1 z
    Mar–Jul:31 30 31 30 31Aug–Dec:31 30 31 30 31Jan–Feb:31 286 e6 M9 d8 g' o: s1 ]$ ^0 t
    最后面的那个系数,是相对于原点的修正值。原公式算出来的值一般太大,计算中用起来会超过32/64位字长。现在天文计算中一般会选择2000年1月1日为零点,之前有用1900年和1950年的。
    1 `( ]$ V2 e" _1 s4 G) K1 r. K, p. D5 @" K; X7 w
    从儒略日计算星期几,(JDN+1) mod 7 就好了。8 f# r- A: w( y% e( J5 L& y
    ( V! H9 N; w2 K$ d6 C
    这个公式是怎么来的呢? 1582年,教皇格里高利十三(XIII)发现,那一年的春分是3月11日,和儒略历里规定的日期3月21日差了十天。原因嘛,就是回归年的长度是365.2422,儒略历用的365.25。格十三用上了全部的指头,哦,应该是找了很多XX家之后,下令当年10月4日的后一天是10月15日,同时规定在原先四年一闰的基础上,100的整数倍年不是闰年,但400的整数倍年又是闰年。新的历法改名为格里高利历。
    : L; ?4 c9 ]/ h9 ^4 O( O0 ?! B( v
    为了计算转换儒略历和格里高利历,一个法国的教会学者Joseph Justus Scaliger给出了这个公式。“儒略日”中的儒略,是他老爸的名字。! N, J4 \  i0 K& ]

    0 O0 D2 @( ^1 |* i/ c8 H哦,原来的文献中用的是儒略历日期,要算儒略日是这样的:
    9 }! X$ O* Q. {% w+ J% e
    8 Q0 q; K; ^. w5 R: Y
    . E' K/ {; e2 e$ g, O" _  s& [
    ) x. p$ t" [1 V: ~, D从儒略日转格里高利历,也有一组公式,这里有:" N! H' L8 `  b2 B# _, T1 Q5 y
    # A; E( T  h: E! X" |) @3 _7 y: i
    其实这些偏差,在儒略历启用之前是有人知道的。但是,始皇三十五年的某一天,一个罗马士兵在西西里岛上,拔出刀来,朝一个老人身上刺下去。这一刺,西方的科技文明停滞了一千多年。来自群组: 软件人家

    评分

    参与人数 6爱元 +35 学识 +5 收起 理由
    东张西望 + 10 + 5
    懒猫猫 + 8 不明觉厉
    水风 + 3 谢谢分享
    东湖珞珈 + 4 谢谢分享
    马鹿 + 8

    查看全部评分

  • TA的每日心情

    2020-3-6 00:28
  • 签到天数: 564 天

    [LV.9]渡劫

    沙发
    发表于 2015-2-1 19:18:36 | 只看该作者
    不明觉厉,捞分走人

    点评

    也捞一分~  发表于 2015-2-3 12:24

    该用户从未签到

    板凳
    发表于 2015-2-1 20:21:47 | 只看该作者
    程序员为啥不直接用格历?
  • TA的每日心情
    开心
    2025-10-6 05:27
  • 签到天数: 2654 天

    [LV.Master]无

    地板
    发表于 2015-2-2 01:20:49 | 只看该作者
    蛮夷的国家成了黑社会渊薮,大科学家的故乡开始赖账

    点评

    哈哈哈哈~  发表于 2015-2-2 02:31
  • TA的每日心情
    奋斗
    11 小时前
  • 签到天数: 2749 天

    [LV.Master]无

    5#
    发表于 2015-2-2 03:00:26 | 只看该作者
    不明觉厉
    回复

    使用道具 举报

  • TA的每日心情
    奋斗
    2024-3-8 05:45
  • 签到天数: 2441 天

    [LV.Master]无

    6#
    发表于 2015-2-2 07:17:10 | 只看该作者
    N多年前学习BASIC语言的时候,就是用这个公式做核心计算,然后再加上几重循环控制的排版,打印一个当年的日历出来。
  • TA的每日心情
    开心
    2022-4-16 03:01
  • 签到天数: 192 天

    [LV.7]分神

    7#
    发表于 2015-2-2 09:53:15 | 只看该作者
    假装我看懂了,然后评分
  • TA的每日心情
    奋斗
    2020-8-27 18:47
  • 签到天数: 638 天

    [LV.9]渡劫

    8#
    发表于 2015-2-2 11:27:07 | 只看该作者
    能换算干支就更好了。
  • TA的每日心情
    开心
    2023-1-5 00:48
  • 签到天数: 2591 天

    [LV.Master]无

    9#
    发表于 2015-2-2 13:09:09 | 只看该作者
    看来俺一定不是程序员了,因为俺从来就没这么复杂地玩过。日期可以从系统函数或者类库中的方法得到,我最多只需要计算某年是否是闰年就足够了。
  • TA的每日心情
    慵懒
    2025-9-2 03:54
  • 签到天数: 2862 天

    [LV.Master]无

    10#
    发表于 2015-2-2 13:52:01 | 只看该作者
    老兵帅客 发表于 2015-2-2 13:09: A) B5 X+ T  W
    看来俺一定不是程序员了,因为俺从来就没这么复杂地玩过。日期可以从系统函数或者类库中的方法得到,我最多 ...
    & _" G- Q+ r6 @% F4 a% Q
    试试计算下一千年每年复活节是哪一天,我又得昏过去了。。。哈哈哈。。。
  • TA的每日心情
    开心
    2022-12-1 00:01
  • 签到天数: 2488 天

    [LV.Master]无

    11#
    发表于 2015-2-3 01:49:06 | 只看该作者
    老兵帅客 发表于 2015-2-2 13:09% U7 h( T: {  }. P" b$ v
    看来俺一定不是程序员了,因为俺从来就没这么复杂地玩过。日期可以从系统函数或者类库中的方法得到,我最多 ...
    2 l  L) x: k! g
    这个在当年可以用的日期函数还很罕见的时候有用。后来系统本身就提供这些计算了,自然没必要记。
    6 d# T" a9 H! q. b) e8 E5 d. r, j2 a
    我最早见到也是学BASIC的时候。2 z1 @. _5 L; O& t0 \( ]/ c
    - ^! `$ r9 s- e# \% F$ h8 a9 u
  • TA的每日心情
    开心
    2022-12-1 00:01
  • 签到天数: 2488 天

    [LV.Master]无

    12#
    发表于 2015-2-3 01:53:29 | 只看该作者
    本帖最后由 橡树村 于 2015-2-3 01:55 编辑
    0 F  J' W- {4 U) E+ h* I" `8 B% `
    hotmen 发表于 2015-2-2 11:27
      `2 A4 M% p' d. `能换算干支就更好了。
    ) a! }) }6 J: s) q' F! O' D

    0 V& R/ C. B$ b4 B3 f$ z% k计算干支里面的日期不难,时辰是从日期推算的,也不难。
    $ u6 a! \/ r- u8 J3 F月份是按照年来推算的,说起来简单,难点在于一年以及一个月的开始时间的计算。这个很难有通用公式。不过还是比农历要简单,干支记年实际上是阳历,每年开始于立春,然后每间隔一个节气就换一个月,与农历的月份并不相同。这样只要有了节气的准确时间数据库,干支的问题也就解决了。
      x* Z$ ?1 w2 O/ K1 W) j" D3 I* L) K( D9 t' [, j
    农历复杂在于,这个历法经常被改动,要准确把历史上的某一天与西历进行换算,需要把曾经使用过的历法都考虑进去,这个麻烦就大了去了。而且每个月的开始取决于月亮的朔望,这就更要把历史上月亮的运行轨道都考虑进去了。" U2 _2 W5 F2 o% x4 `

    + k! i0 g" V# D* Z6 s% D
  • TA的每日心情
    开心
    2023-1-5 00:48
  • 签到天数: 2591 天

    [LV.Master]无

    13#
    发表于 2015-2-3 02:13:12 | 只看该作者
    橡树村 发表于 2015-2-2 12:495 }) f) a& t4 g  V! h. y& y" l
    这个在当年可以用的日期函数还很罕见的时候有用。后来系统本身就提供这些计算了,自然没必要记。
    / `5 _# f# }" W. E. ]' E* s6 Q1 |1 y; H8 C+ \2 \  R
    我最早 ...
    : t5 ~) `( [; u' y6 L* j4 R
    问题是DOS下面的BASIC已经提供日期函数了啊,程序员何必再用这个?
  • TA的每日心情
    开心
    2022-12-1 00:01
  • 签到天数: 2488 天

    [LV.Master]无

    14#
    发表于 2015-2-3 02:21:35 | 只看该作者
    老兵帅客 发表于 2015-2-3 02:13
    - G8 Z. N1 ?: H1 i1 s( r4 C问题是DOS下面的BASIC已经提供日期函数了啊,程序员何必再用这个?

    1 ?5 \( E3 T$ J$ d8 S; k# M# [不记得当年BASIC有计算两个日期之间有几天的函数。也许有但从来没用过,当年就没有使用BASIC做过这类编程,最多弄个日历啥的。
  • TA的每日心情
    开心
    2023-1-5 00:48
  • 签到天数: 2591 天

    [LV.Master]无

    15#
    发表于 2015-2-3 02:29:31 | 只看该作者
    橡树村 发表于 2015-2-2 13:21
    * ^# g. g, h' O; U% b不记得当年BASIC有计算两个日期之间有几天的函数。也许有但从来没用过,当年就没有使用BASIC做过这类编程 ...

    & S3 m- e8 N$ m4 \: W/ A8 [我当年学PASCAL的时候,一个作业就是编万年历,从你的当前日期开始。因此我们就用PASCAL自带的日期函数找出当前日期,然后自己计算闰年。计算闰年的算法很简单的,远比楼主的简单。
  • TA的每日心情
    开心
    2023-1-5 00:48
  • 签到天数: 2591 天

    [LV.Master]无

    16#
    发表于 2015-2-3 02:29:44 | 只看该作者
    橡树村 发表于 2015-2-2 13:21
    ; ]$ t) }7 Z- C不记得当年BASIC有计算两个日期之间有几天的函数。也许有但从来没用过,当年就没有使用BASIC做过这类编程 ...
    5 S( q) ~6 V9 ?" `7 H+ G- G, r
    我当年学PASCAL的时候,一个作业就是编万年历,从你的当前日期开始。因此我们就用PASCAL自带的日期函数找出当前日期,然后自己计算闰年。计算闰年的算法很简单的,远比楼主的简单。
  • TA的每日心情
    开心
    2022-12-1 00:01
  • 签到天数: 2488 天

    [LV.Master]无

    17#
    发表于 2015-2-3 02:42:02 | 只看该作者
    老兵帅客 发表于 2015-2-3 02:29
    ! d  u1 q* j4 d6 b; @" C我当年学PASCAL的时候,一个作业就是编万年历,从你的当前日期开始。因此我们就用PASCAL自带的日期函数找 ...
    4 F3 h/ `" `1 _8 D" g% w$ H% O9 R
    Turbo Pascal?
    % Z  L- F! V% z$ A1 W% N, p; o: G" @  O  y7 r
    最早PC机带的BASIC函数很少的,和Pascal比不了。
  • TA的每日心情
    开心
    2023-1-5 00:48
  • 签到天数: 2591 天

    [LV.Master]无

    18#
    发表于 2015-2-3 02:44:42 | 只看该作者
    橡树村 发表于 2015-2-2 13:42
    2 Z5 F: I% N$ l1 f9 c; L( tTurbo Pascal?
    6 ~  A" \) I" Z1 g7 ^9 O8 X1 \& R- j2 e5 o1 N
    最早PC机带的BASIC函数很少的,和Pascal比不了。

    $ y; j! U6 t4 O, ~; P  C不,是标准PASCAL,用的是微软的编译器。我上学的时候还没出turbo pascal呢,后来这东西出来了,拿来一试,发现丫不兼容标准PASCAL嘿,于是再也没碰它。
  • TA的每日心情
    开心
    2022-12-1 00:01
  • 签到天数: 2488 天

    [LV.Master]无

    19#
    发表于 2015-2-3 02:52:31 | 只看该作者
    老兵帅客 发表于 2015-2-3 02:44
    7 O8 i% A& W0 ~" ^* B- i5 o不,是标准PASCAL,用的是微软的编译器。我上学的时候还没出turbo pascal呢,后来这东西出来了,拿来一试 ...
    " t$ W( Y2 c! o. m/ j6 L$ ~
    Turbo pascal 是83年的,那时候我还不知道计算机长啥样呢。
    * c! |& o+ `% ?
    + O9 j& P4 h5 S7 f我最早是在Comx35机器上接触的BASIC,84年。

    评分

    参与人数 1爱元 +4 收起 理由
    七月群山 + 4 伙呆了

    查看全部评分

  • TA的每日心情
    奋斗
    2024-3-8 05:45
  • 签到天数: 2441 天

    [LV.Master]无

    20#
    发表于 2015-2-3 04:01:00 | 只看该作者
    橡树村 发表于 2015-2-3 02:52
    ! q% \7 S) K! W  ^5 CTurbo pascal 是83年的,那时候我还不知道计算机长啥样呢。. T0 @5 P2 Z  A8 a4 j: T
    ) w* G3 F- M( A
    我最早是在Comx35机器上接触的BASIC,84年。 ...

    8 J, e& g. d& W4 G村子老资格啊

    手机版|小黑屋|Archiver|网站错误报告|爱吱声   

    GMT+8, 2025-10-19 12:48 , Processed in 0.039645 second(s), 24 queries , Gzip On.

    Powered by Discuz! X3.2

    © 2001-2013 Comsenz Inc.

    快速回复 返回顶部 返回列表