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

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

两个要点:

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

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

源代码:

# 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)

PS:这里再为大家推荐几款日历相关在线工具供大家参考:

在线阴历/阳历转换工具:
http://tools.jb51.net/bianmin/yinli2yangli

网页万年历日历:
http://tools.jb51.net/bianmin/webwannianli

在线万年历日历:
http://tools.jb51.net/bianmin/wannianli

在线万年历黄历flash版:
http://tools.jb51.net/bianmin/flashwnl

另外,本站历史上的今天也有相似的农历日期显示功能:

http://tools.jb51.net/bianmin/lishi

更多关于Python相关内容感兴趣的读者可查看本站专题:《Python日期与时间操作技巧总结》、《Python数学运算技巧总结》、《Python数据结构与算法教程》、《Python函数使用技巧总结》、《Python字符串操作技巧汇总》及《Python入门与进阶经典教程》

希望本文所述对大家Python程序设计有所帮助。

时间: 2017-08-19

Python实用日期时间处理方法汇总

原则, 以datetime为中心, 起点或中转, 转化为目标对象, 涵盖了大多数业务场景中需要的日期转换处理 步骤: 1. 掌握几种对象及其关系 2. 了解每类对象的基本操作方法 3. 通过转化关系转化 涉及对象 1. datetime 复制代码 代码如下: >>> import datetime >>> now = datetime.datetime.now() >>> now datetime.datetime(2015, 1, 12, 23, 9

python实现的阳历转阴历(农历)算法

搜索了好几个python实现的万年历多有部分时间有问题,好多是来自这个代码: 复制代码 代码如下: #!/usr/bin/env python# -*- coding: utf-8 -*-'''Usage:  ccal Month [4-Digit-Year]   or:  ccal 4-Digit-Year Month This Python script is to show Solar and Lunar calender at thesame time. You need to have

Python日期的加减等操作的示例

本文介绍了Python日期的加减等操作的示例,分享给大家,也给自己留个笔记 1. 日期输出格式化 所有日期.时间的api都在datetime模块内. 1. datetime => string now = datetime.datetime.now() now.strftime('%Y-%m-%d %H:%M:%S') #输出2012-03-05 16:26:23.870105 strftime是datetime类的实例方法. 2. string => datetime t_str = '20

python根据日期返回星期几的方法

本文实例讲述了python根据日期返回星期几的方法.分享给大家供大家参考.具体如下: 这个函数给定日期,输出星期几,至于0是星期一还是星期天,这和时区有关,反正我这里是0表示星期一 import time,datetime def get_week_day(date): week_day_dict = { 0 : '星期一', 1 : '星期二', 2 : '星期三', 3 : '星期四', 4 : '星期五', 5 : '星期六', 6 : '星期天', } day = date.weekda

python 计算两个日期相差多少个月实例代码

近期,由于业务需要计算两个日期之前相差多少个月.我在网上找了很久,结果发现万能的python,居然没有一个模块计算两个日期的月数,像Java.C#之类的高级语言,都会有(date1-date2).months的现成方法,觉得不可思议.说句实在的,一直觉得python 的日期处理模块真心不好用. 哦,对了,别跟我说 datetime, calendar, dateutil 这些模块,因为我都试过了,都没用.有个竟然算出来还有错.datetime.timedelta只能计算出日时分秒.对年月却不支持

Python计算两个日期相差天数的方法示例

本文实例讲述了Python计算两个日期相差天数的方法.分享给大家供大家参考,具体如下: #!/usr/bin/python import time import sys def dateinput(): date = raw_input('please input the first date: ') return date def datetrans(tdate): spdate = tdate.replace("/","-") try: datesec = ti

python实现字符串和日期相互转换的方法

本文实例讲述了python实现字符串和日期相互转换的方法.分享给大家供大家参考.具体分析如下: 这里用的分别是time和datetime函数 ''' @author: jiangqh ''' import time,datetime # date to str print time.strftime("%Y-%m-%d %X", time.localtime()) #str to date t = time.strptime("2009 - 08 - 08", &q

python编程开发之日期操作实例分析

本文实例讲述了python编程开发之日期操作.分享给大家供大家参考,具体如下: 在python中对日期进行操作的库有: import datetime import time 对日期格式化信息,可以参考官方API: time.strftime datetime 下面是我做的demo: #datetime import datetime #当前日期 now = datetime.datetime.now() print(now.strftime('%Y-%m-%d %H:%M:%S')) prin

python操作日期和时间的方法

不管何时何地,只要我们编程时遇到了跟时间有关的问题,都要想到 datetime 和 time 标准库模块,今天我们就用它内部的方法,详解python操作日期和时间的方法.1.将字符串的时间转换为时间戳 复制代码 代码如下: 方法:a = "2013-10-10 23:40:00"#将其转换为时间数组import timetimeArray = time.strptime(a, "%Y-%m-%d %H:%M:%S")#转换为时间戳:timeStamp = int(t

利用python获取当前日期前后N天或N月日期的方法示例

前言 最近因为工作原因,发现一个Python的时间组件,很好用分享出来!(忘记作者名字了,在这里先感谢了),下面话不多说,来一起看看详细的介绍吧. 示例代码: # -*- coding: utf-8 -*- '''获取当前日期前后N天或N月的日期''' from time import strftime, localtime from datetime import timedelta, date import calendar year = strftime("%Y", localt

利用Python获取操作系统信息实例

前言 每一位运维人员都应该对自己所管理的机器配置很清楚,因为这对我们快速处理问题很有帮助,比如随着业务增长,突然某些机器负载上涨的厉害,这时候要排查原因,除了从应用程序.架构上分析外,当前硬件性能的分析应该是必不可少的一环,今天我们将不用第三方模块,用python自带模块和系统提供的运行信息来获取我们需要的信息,这个脚本除了硬件外,还抓取了当前系统进程数和网卡流量功能,所以这个版本实现的功能基本对应了之前psutil实现的内容,多的不说了,直接贴代码: #!/usr/bin/env python

python获取当前日期和时间的方法

本文实例讲述了python获取当前日期和时间的方法.分享给大家供大家参考.具体如下: import datetime # Get a datetime object now = datetime.datetime.now() # General functions print "Year: %d" % now.year print "Month: %d" % now.month print "Day: %d" % now.day print &q

利用Python获取赶集网招聘信息前篇

如何获取一个网站的相关信息,获取赶集网的招聘信息,本文为大家介绍利用python获取赶集网招聘信息的关键代码,供大家参考,具体内容如下 import re import urllib import urllib.request #获取赶集网数据 def begin(url): #要伪装成的浏览器(我这个是用的chrome) headers = ('User-Agent','Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML,

利用python获取Ping结果示例代码

前言 本文主要跟大家分享了关于利用python获取Ping结果的相关内容,分享出来供大家参考学习,下面话不多说,来一起看看详细的介绍吧. 示例代码: # -*- coding: utf-8 -*- import subprocess import re def get_ping_result(ip_address): p = subprocess.Popen(["ping.exe", ip_address], stdin = subprocess.PIPE, stdout = subp

python获取文件版本信息、公司名和产品名的方法

本文实例讲述了python获取文件版本信息.公司名和产品名的方法,分享给大家供大家参考.具体如下: 该python代码可得到文件版本信息.公司名和产品名.其他的信息都在返回的字典中.具体代码如下: def _getCompanyNameAndProductName(self, file_path): """ Read all properties of the given file return them as a dictionary. """

Python获取数据库数据并保存在excel表格中的方法

将数据库中的数据保存在excel文件中有很多种方法,这里主要介绍pyExcelerator的使用. 一.前期准备(不详细介绍MySQL) python包pyExcelerator和MySQLdb 导入方法:(以Pycharm为例) 在File->Settings中点击右上角绿色图标"+", 输入pyExcelerator,点击install package,导入成功之后点击OK,就完成了pyExcelerator的导入. 2.使用pyExcelerator对excel进行操作 #

Python获取命令实时输出-原样彩色输出并返回输出结果的示例

经试验显示效果不错. #!/usr/bin/python3 # -*- coding: utf-8 -*- import os import subprocess # 与在命令窗口执行显示效果相同,如有彩色输出可保留,但不能返回结果 def run(command): subprocess.call(command, shell=True) # 实时输出但不可显示彩色,可以返回结果 def sh(command, print_msg=True): p = subprocess.Popen(com

利用python和ffmpeg 批量将其他图片转换为.yuv格式的方法

由于跑编码的需要,所以需要制作一个.yuv格式的图片数据集,但是手头只有.jpg格式的,故记录下转换过程.其他图片格式也可以,代码里修改一下就行. ①安装ffmpeg 官网(各种版本):ffmepg官网 window版:https://ffmpeg.zeranoe.com/builds/ ②安装ffmpeg的python3接口ffmpy3 在命令行中输入以下命令: pip install ffmpy3 ③将下载好的windows版解压,把ffmpeg.exe复制到你将要跑python代码的文件夹

Python获取二维数组的行列数的2种方法

这篇文章主要介绍了Python获取二维数组的行列数的2种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 import numpy as np x = np.array([[1,2,5],[2,3,5],[3,4,5],[2,3,6]]) # 输出数组的行和列数 print x.shape # (4, 3) # 只输出行数 print x.shape[0] # 4 # 只输出列数 print x.shape[1] # 3 或者 In [48]