1

我需要编译一个可以将公历日期转换为玛雅日期的程序。我还需要使用 01/01/1970 作为参考日期。

以下辅助功能相应工作,没有错误。

# turns dates into tuples


def dmj(date):

"""
>>> dmj('01/01/1970')
(1, 1, 1970)
>>> dmj('00012+00012+02012')
(12, 12, 2012)
"""

tup = ()
for i in date:
    if i.isdigit() == False and i != ' ':
        date = date.replace(i, ' ')

number_str = ''
for i in date:
    number_str += i
    if i == ' ':
        number = int(number_str)
        tup += (number,)
        number_str = ''
tup += (int(number_str),)
return tup

# counts days that have passed since 01/01/1970

def daysPassed(date):
"""
>>> daysPassed('01/01/1970')
0
>>> daysPassed('20-7-1988')
6775
>>> daysPassed('00012+00012+02012')
15686
"""

from datetime import date
tup = dmj(date)
begin = date(1970, 1, 1)
end = date(tup[2], tup[1], tup[0])
passed = abs(end - begin)        
return passed.days

我的想法是首先计算自 1970 年 1 月 1 日这个 pictun(20 baktuns 长)开始以来已经过去了多少天,然后根据给定日期添加从那时起过去的天数。

在玛雅历法中,一天被称为亲属。他们的时期(在一个pictun内)如下:

20 kin = 1 uinal;18 uinal = 1 吨;20 顿 = 1 卡顿;20 katun = 1 baktun

在长计数符号中,1970 年 1 月 1 日的玛雅日期是“12.17.16.7.5”。Baktun 先写,然后是 katuns,等等......玛雅日期从 0 开始。基本上,uinal 的第一个亲属是数字 0,最后一个是 19,总共 20 个。

我首先编译了以下内容:

def mayanDate(date, separation='')

"""
>>> mayanDate('01/01/1970')
'12/17/16/7/5'
>>> mayaDate('20-7-1988', separator='/')
'12/18/15/4/0'
>>> mayaDate('00012+00012+02012', separator='-')
'12-19-19-17-11'
>>> mayaDate('21 12 2012', separator='+')
'13+0+0+0+0'
>>> mayaDate('26.03.2407')
'14.0.0.0.0'
"""
days = daysPassed(date) + 13 * 144000 + 18 * 7200\
 + 17 * 400 + 8 * 20 + 6     
baktun = str((days //((20 **3) * 18)) - 1)
days = days % ((20 **3) * 18)    
katun = str((days //((20 **2) * 18)) - 1)
days = days % ((20 **2) * 18)    
tun = str((days // (20 **2)) - 1)
days = days % (20 **2)    
uinal = str((days // 20) - 1)
days = days % 20 - 1

kin = str(days)
mayanlist = [baktun, katun, tun, uinal, kin]

for i in date:
    if i.isdigit() == False and separator == '':
        separator = i
        break

mayandate = separator.join(mayanlist)
return mayandate

出于某种奇怪的原因,只有 01/01/1970 返回正确的玛雅长符号,尽管从整个 pictun 的开头开始计数(长度为 7,900 年!)。对于所有其他日期,尽管我的第二个辅助函数返回了正确的值(即使是未来几千年),但它似乎在日历中前进得太快了。

我想知道哪里错了。例如mayanDate('20-7-1988')返回'12-18-15-6-0'而不是'12-18-15-4-0'mayanDate('21 12 2012')返回'13 0 1 12 0'而不是'13 0 0 0 0'

4

2 回答 2

1

您在日期“1970 年 1 月 1 日”的负 1 亲属中看到的问题是由于在计算期间从每个日期序数中删除了一个。取值x%20总是会返回 0 到 19 之间的值。从结果中取一个必然会将这个范围转移到 -1 到 18 (含)。

添加到 daysPassed(date) 结果中的数字似乎是 1/1/1970 长格式的转换,每个数字都添加了一个。我假设这样做是为了反驳玛雅历法从零开始计数的事实,但这是不必要的。玛雅日期 0.0.0.0.1.5 计数 25,而不是 151646。但这似乎不是错误的根源,因为从我自己的代码中删除了这个问题,我仍然得到与 20-7 描述的相同的结果-1988 年和 21-12-2012。

当我返回并为命名常量切换代码中的所有幻数时,我终于根除错误(它使代码更容易调试、阅读和维护)。您说一个 tun 中有 18 个 uinal,一个 katun 中有 20 个 tun,但是这些数字在代码中是相反的。

这是我的代码:

def mayanDate(date_str, seperation=','):
days_in_kin     = 1
kin_in_uinal    = 20
uinal_in_tun    = 18
tun_in_katun    = 20
katun_in_baktun = 20

days_in_uinal   = days_in_kin * kin_in_uinal
days_in_tun     = days_in_uinal * uinal_in_tun
days_in_katun   = days_in_tun * tun_in_katun
days_in_baktun  = days_in_katun * katun_in_baktun

days_1970 = 12 * days_in_baktun \
    + 17 * days_in_katun\
    + 16 * days_in_tun\
    + 7 * days_in_uinal\
    + 5 * days_in_kin

total_days = daysPassed(date_str) + days_1970

baktun = total_days // days_in_baktun
total_days -= baktun * days_in_baktun
katun = total_days // days_in_katun
total_days -= katun * days_in_katun
tun = total_days // days_in_tun
total_days -= tun * days_in_tun
uinal = total_days // days_in_uinal
total_days -= uinal * days_in_uinal
kin = total_days // days_in_kin

print seperation.join(map(str, (baktun, katun, tun, uinal, kin)))

(我从总天数中减去了之前的计算,而不是使用模运算符,因为我觉得它更干净。我想这是个人喜好问题。)

于 2013-05-18T18:16:22.177 回答
0

我可能发现了什么。

>>>mayanDate('15/01/1970')
'12/17/16/8/-1'

显然不可能。-1 必须是 19,8 必须是 7。这似乎太早了一个月。仍然不知道为什么 01/01/1970 在这里仍然正确。不知道那个日期有什么特别之处。

于 2013-05-16T14:40:03.357 回答