Python实现公历(阳历)转农历(阴历)的方法示例


Posted in Python onAugust 22, 2017

本文实例讲述了Python实现公历(阳历)转农历(阴历)的方法。分享给大家供大家参考,具体如下:

两个要点:

1、公历转农历用了查表法(第126行)
2、节气用了天文法?(第176行)

运行图 (背景是hao123万年历)

Python实现公历(阳历)转农历(阴历)的方法示例

Python实现公历(阳历)转农历(阴历)的方法示例

源代码:

# lunar.py
# 2015/02/27 罗兵
import datetime
class Lunar(object):
  #******************************************************************************
  # 下面为阴历计算所需的数据,为节省存储空间,所以采用下面比较变态的存储方法.
  #******************************************************************************
  #数组g_lunar_month_day存入阴历1901年到2050年每年中的月天数信息,
  #阴历每月只能是29或30天,一年用12(或13)个二进制位表示,对应位为1表30天,否则为29天
  g_lunar_month_day = [
    0x4ae0, 0xa570, 0x5268, 0xd260, 0xd950, 0x6aa8, 0x56a0, 0x9ad0, 0x4ae8, 0x4ae0,  #1910
    0xa4d8, 0xa4d0, 0xd250, 0xd548, 0xb550, 0x56a0, 0x96d0, 0x95b0, 0x49b8, 0x49b0,  #1920
    0xa4b0, 0xb258, 0x6a50, 0x6d40, 0xada8, 0x2b60, 0x9570, 0x4978, 0x4970, 0x64b0,  #1930
    0xd4a0, 0xea50, 0x6d48, 0x5ad0, 0x2b60, 0x9370, 0x92e0, 0xc968, 0xc950, 0xd4a0,  #1940
    0xda50, 0xb550, 0x56a0, 0xaad8, 0x25d0, 0x92d0, 0xc958, 0xa950, 0xb4a8, 0x6ca0,  #1950
    0xb550, 0x55a8, 0x4da0, 0xa5b0, 0x52b8, 0x52b0, 0xa950, 0xe950, 0x6aa0, 0xad50,  #1960
    0xab50, 0x4b60, 0xa570, 0xa570, 0x5260, 0xe930, 0xd950, 0x5aa8, 0x56a0, 0x96d0,  #1970
    0x4ae8, 0x4ad0, 0xa4d0, 0xd268, 0xd250, 0xd528, 0xb540, 0xb6a0, 0x96d0, 0x95b0,  #1980
    0x49b0, 0xa4b8, 0xa4b0, 0xb258, 0x6a50, 0x6d40, 0xada0, 0xab60, 0x9370, 0x4978,  #1990
    0x4970, 0x64b0, 0x6a50, 0xea50, 0x6b28, 0x5ac0, 0xab60, 0x9368, 0x92e0, 0xc960,  #2000
    0xd4a8, 0xd4a0, 0xda50, 0x5aa8, 0x56a0, 0xaad8, 0x25d0, 0x92d0, 0xc958, 0xa950,  #2010
    0xb4a0, 0xb550, 0xb550, 0x55a8, 0x4ba0, 0xa5b0, 0x52b8, 0x52b0, 0xa930, 0x74a8,  #2020
    0x6aa0, 0xad50, 0x4da8, 0x4b60, 0x9570, 0xa4e0, 0xd260, 0xe930, 0xd530, 0x5aa0,  #2030
    0x6b50, 0x96d0, 0x4ae8, 0x4ad0, 0xa4d0, 0xd258, 0xd250, 0xd520, 0xdaa0, 0xb5a0,  #2040
    0x56d0, 0x4ad8, 0x49b0, 0xa4b8, 0xa4b0, 0xaa50, 0xb528, 0x6d20, 0xada0, 0x55b0,  #2050
  ]
  #数组gLanarMonth存放阴历1901年到2050年闰月的月份,如没有则为0,每字节存两年
  g_lunar_month = [
    0x00, 0x50, 0x04, 0x00, 0x20,  #1910
    0x60, 0x05, 0x00, 0x20, 0x70,  #1920
    0x05, 0x00, 0x40, 0x02, 0x06,  #1930
    0x00, 0x50, 0x03, 0x07, 0x00,  #1940
    0x60, 0x04, 0x00, 0x20, 0x70,  #1950
    0x05, 0x00, 0x30, 0x80, 0x06,  #1960
    0x00, 0x40, 0x03, 0x07, 0x00,  #1970
    0x50, 0x04, 0x08, 0x00, 0x60,  #1980
    0x04, 0x0a, 0x00, 0x60, 0x05,  #1990
    0x00, 0x30, 0x80, 0x05, 0x00,  #2000
    0x40, 0x02, 0x07, 0x00, 0x50,  #2010
    0x04, 0x09, 0x00, 0x60, 0x04,  #2020
    0x00, 0x20, 0x60, 0x05, 0x00,  #2030
    0x30, 0xb0, 0x06, 0x00, 0x50,  #2040
    0x02, 0x07, 0x00, 0x50, 0x03  #2050
  ]
  START_YEAR = 1901
  # 天干
  gan = '甲乙丙丁戊己庚辛壬癸'
  # 地支
  zhi = '子丑寅卯辰巳午未申酉戌亥'
  # 生肖
  xiao = '鼠牛虎兔龙蛇马羊猴鸡狗猪'
  # 月份
  lm = '正二三四五六七八九十冬腊'
  # 日份
  ld = '初一初二初三初四初五初六初七初八初九初十十一十二十三十四十五十六十七十八十九二十廿一廿二廿三廿四廿五廿六廿七廿八廿九三十'
  # 节气
  jie = '小寒大寒立春雨水惊蛰春分清明谷雨立夏小满芒种夏至小暑大暑立秋处暑白露秋分寒露霜降立冬小雪大雪冬至'
  def __init__(self, dt = None):
    '''初始化:参数为datetime.datetime类实例,默认当前时间'''
    self.localtime = dt if dt else datetime.datetime.today()
  def sx_year(self): # 返回生肖年
    ct = self.localtime #取当前时间
    year = self.ln_year() - 3 - 1 # 农历年份减3 (说明:补减1)
    year = year % 12 # 模12,得到地支数
    return self.xiao[year]
  def gz_year(self): # 返回干支纪年
    ct = self.localtime #取当前时间
    year = self.ln_year() - 3 - 1 # 农历年份减3 (说明:补减1)
    G = year % 10 # 模10,得到天干数
    Z = year % 12 # 模12,得到地支数
    return self.gan[G] + self.zhi[Z]
  def gz_month(self): # 返回干支纪月(未实现)
    pass
  def gz_day(self): # 返回干支纪日
    ct = self.localtime #取当前时间
    C = ct.year // 100 #取世纪数,减一
    y = ct.year % 100 #取年份后两位(若为1月、2月则当前年份减一)
    y = y - 1 if ct.month == 1 or ct.month == 2 else y
    M = ct.month #取月份(若为1月、2月则分别按13、14来计算)
    M = M + 12 if ct.month == 1 or ct.month == 2 else M
    d = ct.day #取日数
    i = 0 if ct.month % 2 == 1 else 6 #取i (奇数月i=0,偶数月i=6)
    #下面两个是网上的公式
    # http://baike.baidu.com/link?url=MbTKmhrTHTOAz735gi37tEtwd29zqE9GJ92cZQZd0X8uFO5XgmyMKQru6aetzcGadqekzKd3nZHVS99rewya6q
    # 计算干(说明:补减1)
    G = 4 * C + C // 4 + 5 * y + y // 4 + 3 * (M + 1) // 5 + d - 3 - 1
    G = G % 10
    # 计算支(说明:补减1)
    Z = 8 * C + C // 4 + 5 * y + y // 4 + 3 * (M + 1) // 5 + d + 7 + i - 1
    Z = Z % 12
    #返回 干支纪日
    return self.gan[G] + self.zhi[Z]
  def gz_hour(self): # 返回干支纪时(时辰)
    ct = self.localtime #取当前时间
    #计算支
    Z = round((ct.hour/2) + 0.1) % 12 # 之所以加0.1是因为round的bug!!
    #返回 干支纪时(时辰)
    return self.zhi[Z]
  def ln_year(self): # 返回农历年
    year, _, _ = self.ln_date()
    return year
  def ln_month(self): # 返回农历月
    _, month, _ = self.ln_date()
    return month
  def ln_day(self): # 返回农历日
    _, _, day = self.ln_date()
    return day
  def ln_date(self): # 返回农历日期整数元组(年、月、日)(查表法)
    delta_days = self._date_diff()
    #阳历1901年2月19日为阴历1901年正月初一
    #阳历1901年1月1日到2月19日共有49天
    if (delta_days < 49):
      year = self.START_YEAR - 1
      if (delta_days <19):
       month = 11;
       day = 11 + delta_days
      else:
        month = 12;
        day = delta_days - 18
      return (year, month, day)
    #下面从阴历1901年正月初一算起
    delta_days -= 49
    year, month, day = self.START_YEAR, 1, 1
    #计算年
    tmp = self._lunar_year_days(year)
    while delta_days >= tmp:
      delta_days -= tmp
      year += 1
      tmp = self._lunar_year_days(year)
    #计算月
    (foo, tmp) = self._lunar_month_days(year, month)
    while delta_days >= tmp:
      delta_days -= tmp
      if (month == self._get_leap_month(year)):
        (tmp, foo) = self._lunar_month_days(year, month)
        if (delta_days < tmp):
          return (0, 0, 0)
          return (year, month, delta_days+1)
        delta_days -= tmp
      month += 1
      (foo, tmp) = self._lunar_month_days(year, month)
    #计算日
    day += delta_days
    return (year, month, day)
  def ln_date_str(self):# 返回农历日期字符串,形如:农历正月初九
    _, month, day = self.ln_date()
    return '农历{}月{}'.format(self.lm[month-1], self.ld[(day-1)*2:day*2])
  def ln_jie(self): # 返回农历节气
    ct = self.localtime #取当前时间
    year = ct.year
    for i in range(24):
      #因为两个都是浮点数,不能用相等表示
      delta = self._julian_day() - self._julian_day_of_ln_jie(year, i)
      if -.5 <= delta <= .5:
        return self.jie[i*2:(i+1)*2]
    return ''
  #显示日历
  def calendar(self):
    pass
  #######################################################
  #      下面皆为私有函数
  #######################################################
  def _date_diff(self):
    '''返回基于1901/01/01日差数'''
    return (self.localtime - datetime.datetime(1901, 1, 1)).days
  def _get_leap_month(self, lunar_year):
    flag = self.g_lunar_month[(lunar_year - self.START_YEAR) // 2]
    if (lunar_year - self.START_YEAR) % 2:
      return flag & 0x0f
    else:
      return flag >> 4
  def _lunar_month_days(self, lunar_year, lunar_month):
    if (lunar_year < self.START_YEAR):
      return 30
    high, low = 0, 29
    iBit = 16 - lunar_month;
    if (lunar_month > self._get_leap_month(lunar_year) and self._get_leap_month(lunar_year)):
      iBit -= 1
    if (self.g_lunar_month_day[lunar_year - self.START_YEAR] & (1 << iBit)):
      low += 1
    if (lunar_month == self._get_leap_month(lunar_year)):
      if (self.g_lunar_month_day[lunar_year - self.START_YEAR] & (1 << (iBit -1))):
         high = 30
      else:
         high = 29
    return (high, low)
  def _lunar_year_days(self, year):
    days = 0
    for i in range(1, 13):
      (high, low) = self._lunar_month_days(year, i)
      days += high
      days += low
    return days
  # 返回指定公历日期的儒略日
  def _julian_day(self):
    ct = self.localtime #取当前时间
    year = ct.year
    month = ct.month
    day = ct.day
    if month <= 2:
      month += 12
      year -= 1
    B = year / 100
    B = 2 - B + year / 400
    dd = day + 0.5000115740 #本日12:00后才是儒略日的开始(过一秒钟)*/
    return int(365.25 * (year + 4716) + 0.01) + int(30.60001 * (month + 1)) + dd + B - 1524.5
  # 返回指定年份的节气的儒略日数
  def _julian_day_of_ln_jie(self, year, st):
    s_stAccInfo =[
       0.00, 1272494.40, 2548020.60, 3830143.80, 5120226.60, 6420865.80,
       7732018.80, 9055272.60, 10388958.00, 11733065.40, 13084292.40, 14441592.00,
       15800560.80, 17159347.20, 18513766.20, 19862002.20, 21201005.40, 22529659.80,
       23846845.20, 25152606.00, 26447687.40, 27733451.40, 29011921.20, 30285477.60]
    #已知1900年小寒时刻为1月6日02:05:00
    base1900_SlightColdJD = 2415025.5868055555
    if (st < 0) or (st > 24):
      return 0.0
    stJd = 365.24219878 * (year - 1900) + s_stAccInfo[st] / 86400.0
    return base1900_SlightColdJD + stJd
# 测试
def test(ct=None):
  ln = Lunar(ct)
  print('公历 {} 北京时间 {}'.format(ln.localtime.date(), ln.localtime.time()))
  print('{} 【{}】 {}年 {}日 {}时'.format(ln.ln_date_str(), ln.gz_year(), ln.sx_year(), ln.gz_day(), ln.gz_hour()))
  print('节气:{}'.format(ln.ln_jie()))
if __name__ == '__main__':
  ct = datetime.datetime(2015,2,19,13,0,15)
  test(ct)
Python 相关文章推荐
PHP webshell检查工具 python实现代码
Sep 15 Python
python从网络读取图片并直接进行处理的方法
May 22 Python
Python使用剪切板的方法
Jun 06 Python
python实现给scatter设置颜色渐变条colorbar的方法
Dec 13 Python
浅谈Pandas:Series和DataFrame间的算术元素
Dec 22 Python
python+numpy实现的基本矩阵操作示例
Jul 19 Python
Python 迭代,for...in遍历,迭代原理与应用示例
Oct 12 Python
python设置代理和添加镜像源的方法
Feb 14 Python
Python的Django框架实现数据库查询(不返回QuerySet的方法)
May 19 Python
解决keras加入lambda层时shape的问题
Jun 11 Python
详解python中的异常捕获
Dec 15 Python
教你如何使用Python开发一个钉钉群应答机器人
Jun 21 Python
Python实现的矩阵类实例
Aug 22 #Python
Python实现图片转字符画的示例
Aug 22 #Python
利用aardio给python编写图形界面
Aug 21 #Python
Python实现图片转字符画的示例代码
Aug 21 #Python
Python微信企业号开发之回调模式接收微信端客户端发送消息及被动返回消息示例
Aug 21 #Python
Python编程实现微信企业号文本消息推送功能示例
Aug 21 #Python
Python编程之微信推送模板消息功能示例
Aug 21 #Python
You might like
图片存储与浏览一例(Linux+Apache+PHP+MySQL)
2006/10/09 PHP
解决file_get_contents无法请求https连接的方法
2013/12/17 PHP
PHP+Ajax实时自动检测是否联网的方法
2015/07/01 PHP
ThinkPHP3.2.3框架实现的空模块、空控制器、空操作,跳转到错误404页面图文详解
2019/04/03 PHP
解决laravel groupBy 对查询结果进行分组出现的问题
2019/10/09 PHP
PHP数组与字符串互相转换实例
2020/05/05 PHP
借用Google的Javascript API Loader来加速你的网站
2009/01/28 Javascript
基于jquery的一个OutlookBar类,动态创建导航条
2010/11/19 Javascript
ExtJS4 Grid改变单元格背景颜色及Column render学习
2013/02/06 Javascript
推荐一款jQuery插件模板
2015/01/09 Javascript
js实现文本框选中的方法
2015/05/26 Javascript
javascript中使用正则表达式清理table样式的代码
2020/04/01 Javascript
JS实现为排序好的字符串找出重复行的方法
2016/03/02 Javascript
使用BootStrap和Metroui设计的metro风格微网站或手机app界面
2016/10/21 Javascript
JavaScript获取select中text值的方法
2017/02/13 Javascript
轻松学习Javascript闭包
2017/03/01 Javascript
jQuery 实现图片的依次加载图片功能
2017/07/06 jQuery
详解webpack babel的配置
2018/01/09 Javascript
js将当前时间格式化为 年-月-日 时:分:秒的实现代码
2018/01/20 Javascript
Vue高版本中一些新特性的使用详解
2018/09/25 Javascript
jQuery实现的简单日历组件定义与用法示例
2018/12/24 jQuery
微信小程序实现上传图片裁剪图片过程解析
2019/08/22 Javascript
three.js 实现露珠滴落动画效果的示例代码
2021/03/01 Javascript
Python实现通讯录功能
2018/02/22 Python
Python实现FTP文件传输的实例
2019/07/07 Python
详解使用PyInstaller将Pygame库编写的小游戏程序打包为exe文件
2019/08/23 Python
pip 安装库比较慢的解决方法(国内镜像)
2019/10/06 Python
拉飞逸官网:Lafayette 148 New York
2020/07/15 全球购物
毕业生造价工程师求职信
2013/10/17 职场文书
小学生读书感言
2014/02/12 职场文书
《回乡偶书》教学反思
2014/04/12 职场文书
2014年学校法制宣传日活动总结
2014/11/01 职场文书
追悼会答谢词
2015/01/05 职场文书
2016年小学圣诞节活动总结
2016/03/31 职场文书
青年人初次创业的“五不要”
2019/08/23 职场文书
Redis性能监控的实现
2021/07/09 Redis