1 Star 0 Fork 63

jiurenm/carbon

forked from dromara/carbon 
加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
克隆/下载
calendar.lunar.go 15.98 KB
一键复制 编辑 原始数据 按行查看 历史
gouguoyin 提交于 2022-04-24 09:47 . v2.1.3 优化代码
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672
package carbon
import (
"fmt"
"strings"
)
var (
minYear, maxYear = 1900, 2100
lunarNumbers = []string{"零", "一", "二", "三", "四", "五", "六", "七", "八", "九"}
lunarMonths = []string{"正月", "二月", "三月", "四月", "五月", "六月", "七月", "八月", "九月", "十月", "十一月", "腊月"}
lunarTimes = []string{"子时", "丑时", "寅时", "卯时", "辰时", "巳时", "午时", "未时", "申时", "酉时", "戌时", "亥时"}
lunarAnimals = []string{"猴", "鸡", "狗", "猪", "鼠", "牛", "虎", "兔", "龙", "蛇", "马", "羊"}
lunarFestivals = []string{"春节", "元宵节", "端午节", "七夕节", "中元节", "中秋节", "重阳节", "寒衣节", "下元节", "腊八节", "小年"}
lunarTerms = []int{
0x04bd8, 0x04ae0, 0x0a570, 0x054d5, 0x0d260, 0x0d950, 0x16554, 0x056a0, 0x09ad0, 0x055d2, // 1900-1909
0x04ae0, 0x0a5b6, 0x0a4d0, 0x0d250, 0x1d255, 0x0b540, 0x0d6a0, 0x0ada2, 0x095b0, 0x14977, // 1910-1919
0x04970, 0x0a4b0, 0x0b4b5, 0x06a50, 0x06d40, 0x1ab54, 0x02b60, 0x09570, 0x052f2, 0x04970, // 1920-1929
0x06566, 0x0d4a0, 0x0ea50, 0x16a95, 0x05ad0, 0x02b60, 0x186e3, 0x092e0, 0x1c8d7, 0x0c950, // 1930-1939
0x0d4a0, 0x1d8a6, 0x0b550, 0x056a0, 0x1a5b4, 0x025d0, 0x092d0, 0x0d2b2, 0x0a950, 0x0b557, // 1940-1949
0x06ca0, 0x0b550, 0x15355, 0x04da0, 0x0a5d0, 0x14573, 0x052d0, 0x0a9a8, 0x0e950, 0x06aa0, // 1950-1959
0x0aea6, 0x0ab50, 0x04b60, 0x0aae4, 0x0a570, 0x05260, 0x0f263, 0x0d950, 0x05b57, 0x056a0, // 1960-1969
0x096d0, 0x04dd5, 0x04ad0, 0x0a4d0, 0x0d4d4, 0x0d250, 0x0d558, 0x0b540, 0x0b5a0, 0x195a6, // 1970-1979
0x095b0, 0x049b0, 0x0a974, 0x0a4b0, 0x0b27a, 0x06a50, 0x06d40, 0x0af46, 0x0ab60, 0x09570, // 1980-1989
0x04af5, 0x04970, 0x064b0, 0x074a3, 0x0ea50, 0x06b58, 0x05ac0, 0x0ab60, 0x096d5, 0x092e0, // 1990-1999
0x0c960, 0x0d954, 0x0d4a0, 0x0da50, 0x07552, 0x056a0, 0x0abb7, 0x025d0, 0x092d0, 0x0cab5, // 2000-2009
0x0a950, 0x0b4a0, 0x0baa4, 0x0ad50, 0x055d9, 0x04ba0, 0x0a5b0, 0x15176, 0x052b0, 0x0a930, // 2010-2019
0x07954, 0x06aa0, 0x0ad50, 0x05b52, 0x04b60, 0x0a6e6, 0x0a4e0, 0x0d260, 0x0ea65, 0x0d530, // 2020-2029
0x05aa0, 0x076a3, 0x096d0, 0x04bd7, 0x04ad0, 0x0a4d0, 0x1d0b6, 0x0d250, 0x0d520, 0x0dd45, // 2030-2039
0x0b5a0, 0x056d0, 0x055b2, 0x049b0, 0x0a577, 0x0a4b0, 0x0aa50, 0x1b255, 0x06d20, 0x0ada0, // 2040-2049
0x14b63, 0x09370, 0x049f8, 0x04970, 0x064b0, 0x168a6, 0x0ea50, 0x06b20, 0x1a6c4, 0x0aae0, // 2050-2059
0x0a2e0, 0x0d2e3, 0x0c960, 0x0d557, 0x0d4a0, 0x0da50, 0x05d55, 0x056a0, 0x0a6d0, 0x055d4, // 2060-2069
0x052d0, 0x0a9b8, 0x0a950, 0x0b4a0, 0x0b6a6, 0x0ad50, 0x055a0, 0x0aba4, 0x0a5b0, 0x052b0, // 2070-2079
0x0b273, 0x06930, 0x07337, 0x06aa0, 0x0ad50, 0x14b55, 0x04b60, 0x0a570, 0x054e4, 0x0d160, // 2080-2089
0x0e968, 0x0d520, 0x0daa0, 0x16aa6, 0x056d0, 0x04ae0, 0x0a9d4, 0x0a2d0, 0x0d150, 0x0f252, // 2090-2099
0x0d520, // 2100
}
invalidYearError = func(year int) error {
return fmt.Errorf("invalid year %d, currently only 200 years from 1900 to 2100 are supported", year)
}
)
// lunar defines a lunar struct.
// 定义 lunar 结构体
type lunar struct {
year, month, day, hour, minute, second int // 农历年、月、日、时、分、秒
isInvalid bool // 是否不可利用
isLeapMonth bool // 是否是闰月
Error error
}
// Lunar converts the gregorian calendar to the lunar calendar.
// 将公历转为农历
func (c Carbon) Lunar() (l lunar) {
l.isInvalid, l.isLeapMonth = false, false
if c.IsInvalid() {
l.Error = c.Error
l.isInvalid = true
return
}
// leapMonths:闰月总数,daysOfYear:年天数,daysOfMonth:月天数,leapMonth:闰月月份
daysInYear, daysInMonth, leapMonth := 365, 30, 0
// 有效范围检验
year := c.Year()
if year < minYear || year > maxYear {
l.Error = invalidYearError(year)
l.isInvalid = true
return
}
offset := int(c.DiffAbsInDays(c.CreateFromDateTime(minYear, 1, 31, 0, 0, 0)))
for l.year = minYear; l.year <= maxYear && offset > 0; l.year++ {
daysInYear = l.getDaysInYear()
offset -= daysInYear
}
if offset < 0 {
offset += daysInYear
l.year--
}
leapMonth = l.LeapMonth()
for l.month = 1; l.month <= 12 && offset > 0; l.month++ {
if leapMonth > 0 && l.month == (leapMonth+1) && !l.isLeapMonth {
l.month--
l.isLeapMonth = true
daysInMonth = l.getDaysInLeapMonth()
} else {
daysInMonth = l.getDaysInMonth()
}
offset -= daysInMonth
if l.isLeapMonth && l.month == (leapMonth+1) {
l.isLeapMonth = false
}
}
// offset为0时,并且刚才计算的月份是闰月,要校正
if offset == 0 && leapMonth > 0 && l.month == leapMonth+1 {
if l.isLeapMonth {
l.isLeapMonth = false
} else {
l.isLeapMonth = true
l.month--
}
}
// offset小于0时,也要校正
if offset < 0 {
offset += daysInMonth
l.month--
}
l.day = offset + 1
l.hour, l.minute, l.second = c.Time()
return
}
// getDaysInYear gets total days in lunar year.
// 获取该年总天数
func (l lunar) getDaysInYear() int {
var sum = 348
for i := 0x8000; i > 0x8; i >>= 1 {
if (lunarTerms[l.year-minYear] & i) != 0 {
sum++
}
}
return sum + l.getDaysInLeapMonth()
}
// getDaysInMonth gets total days in lunar month.
// 获取该月总天数
func (l lunar) getDaysInMonth() int {
if (lunarTerms[l.year-minYear] & (0x10000 >> uint(l.month))) == 0 {
return 29
}
return 30
}
// getDaysInLeapMonth gets total days in lunar leap month.
// 获取闰月总天数
func (l lunar) getDaysInLeapMonth() int {
if l.LeapMonth() == 0 {
return 0
}
if (lunarTerms[l.year-minYear] & 0x10000) != 0 {
return 30
}
return 29
}
// Animal gets lunar animal name like "猴".
// 获取生肖
func (l lunar) Animal() string {
if l.isInvalid {
return ""
}
return lunarAnimals[l.year%MonthsPerYear]
}
// Festival gets lunar festival name like "春节".
// 获取农历节日
func (l lunar) Festival() (festival string) {
if l.isInvalid {
return
}
month, day := l.month, l.day
switch {
case month == 1 && day == 1:
festival = lunarFestivals[0]
case month == 1 && day == 15:
festival = lunarFestivals[1]
case month == 5 && day == 5:
festival = lunarFestivals[2]
case month == 7 && day == 7:
festival = lunarFestivals[3]
case month == 7 && day == 15:
festival = lunarFestivals[4]
case month == 8 && day == 15:
festival = lunarFestivals[5]
case month == 9 && day == 9:
festival = lunarFestivals[6]
case month == 10 && day == 1:
festival = lunarFestivals[7]
case month == 10 && day == 15:
festival = lunarFestivals[8]
case month == 12 && day == 8:
festival = lunarFestivals[9]
case month == 12 && day == 23:
festival = lunarFestivals[10]
}
return
}
// DateTime gets lunar year, month, day, hour, minute, and second like 2020, 8, 5, 13, 14, 15.
// 获取农历年月日时分秒
func (l lunar) DateTime() (year, month, day, hour, minute, second int) {
if l.isInvalid {
return
}
return l.year, l.month, l.day, l.hour, l.minute, l.second
}
// Date gets lunar year, month and day like 2020, 8, 5.
// 获取农历时分秒
func (l lunar) Date() (year, month, day int) {
if l.isInvalid {
return
}
return l.year, l.month, l.day
}
// Time gets lunar hour, minute, and second like 13, 14, 15.
// 获取农历时分秒
func (l lunar) Time() (hour, minute, second int) {
if l.isInvalid {
return
}
return l.hour, l.minute, l.second
}
// Year gets lunar year like 2020.
// 获取农历年
func (l lunar) Year() int {
if l.isInvalid {
return 0
}
return l.year
}
// Month gets lunar month like 8.
// 获取农历月
func (l lunar) Month() int {
if l.isInvalid {
return 0
}
return l.month
}
// LeapMonth gets lunar leap month like 8.
// 获取农历闰月月份
func (l lunar) LeapMonth() int {
if l.isInvalid {
return 0
}
return lunarTerms[l.year-minYear] & 0xf
}
// Day gets lunar day like 5.
// 获取农历日
func (l lunar) Day() int {
if l.isInvalid {
return 0
}
return l.day
}
// ToYearString outputs a string in lunar year format like "二零二零".
// 获取农历年字符串
func (l lunar) ToYearString() string {
if l.isInvalid {
return ""
}
year := fmt.Sprintf("%d", l.year)
for i, replace := range lunarNumbers {
year = strings.Replace(year, fmt.Sprintf("%d", i), replace, -1)
}
return year
}
// ToMonthString outputs a string in lunar month format like "正月".
// 获取农历月字符串
func (l lunar) ToMonthString() string {
if l.isInvalid {
return ""
}
return lunarMonths[l.month-1]
}
// ToDayString outputs a string in lunar day format like "廿一".
// 获取农历日字符串
func (l lunar) ToDayString() (day string) {
if l.isInvalid {
return
}
num := lunarNumbers[l.day%10]
switch {
case l.day == 30:
day = "三十"
case l.day > 20:
day = "廿" + num
case l.day == 20:
day = "二十"
case l.day > 10:
day = "十" + num
case l.day == 10:
day = "初十"
case l.day < 10:
day = "初" + num
}
return
}
// ToDateString outputs a string in lunar date format like "二零二零年腊月初五".
// 获取农历日期字符串
func (l lunar) ToDateString() string {
if l.isInvalid {
return ""
}
return l.ToYearString() + "年" + l.ToMonthString() + l.ToDayString()
}
// String outputs a string in YYYY-MM-DD HH::ii::ss format, implement Stringer interface.
// 输出 YYYY-MM-DD HH::ii::ss 格式字符串, 实现 Stringer 接口
func (l lunar) String() string {
if l.isInvalid {
return ""
}
return fmt.Sprintf("%d-%02d-%02d %02d:%02d:%02d", l.year, l.month, l.day, l.hour, l.minute, l.second)
}
// IsLeapYear reports whether is leap year.
// 是否是闰年
func (l lunar) IsLeapYear() bool {
if l.isInvalid {
return false
}
return l.LeapMonth() != 0
}
// IsLeapMonth reports whether is leap month.
// 是否是闰月
func (l lunar) IsLeapMonth() bool {
if l.isInvalid {
return false
}
return l.month == l.LeapMonth()
}
// IsRatYear reports whether is year of Rat.
// 是否是鼠年
func (l lunar) IsRatYear() bool {
if l.isInvalid {
return false
}
if l.year%MonthsPerYear == 4 {
return true
}
return false
}
// IsOxYear reports whether is year of Ox.
// 是否是牛年
func (l lunar) IsOxYear() bool {
if l.isInvalid {
return false
}
if l.year%MonthsPerYear == 5 {
return true
}
return false
}
// IsTigerYear reports whether is year of Tiger.
// 是否是虎年
func (l lunar) IsTigerYear() bool {
if l.isInvalid {
return false
}
if l.year%MonthsPerYear == 6 {
return true
}
return false
}
// IsRabbitYear reports whether is year of Rabbit.
// 是否是兔年
func (l lunar) IsRabbitYear() bool {
if l.isInvalid {
return false
}
if l.year%MonthsPerYear == 7 {
return true
}
return false
}
// IsDragonYear reports whether is year of Dragon.
// 是否是龙年
func (l lunar) IsDragonYear() bool {
if l.isInvalid {
return false
}
if l.year%MonthsPerYear == 8 {
return true
}
return false
}
// IsSnakeYear reports whether is year of Snake.
// 是否是蛇年
func (l lunar) IsSnakeYear() bool {
if l.isInvalid {
return false
}
if l.year%MonthsPerYear == 9 {
return true
}
return false
}
// IsHorseYear reports whether is year of Horse.
// 是否是马年
func (l lunar) IsHorseYear() bool {
if l.isInvalid {
return false
}
if l.year%MonthsPerYear == 10 {
return true
}
return false
}
// IsGoatYear reports whether is year of Goat.
// 是否是羊年
func (l lunar) IsGoatYear() bool {
if l.isInvalid {
return false
}
if l.year%MonthsPerYear == 11 {
return true
}
return false
}
// IsMonkeyYear reports whether is year of Monkey.
// 是否是猴年
func (l lunar) IsMonkeyYear() bool {
if l.isInvalid {
return false
}
if l.year%MonthsPerYear == 0 {
return true
}
return false
}
// IsRoosterYear reports whether is year of Rooster.
// 是否是鸡年
func (l lunar) IsRoosterYear() bool {
if l.isInvalid {
return false
}
if l.year%MonthsPerYear == 1 {
return true
}
return false
}
// IsDogYear reports whether is year of Dog.
// 是否是狗年
func (l lunar) IsDogYear() bool {
if l.isInvalid {
return false
}
if l.year%MonthsPerYear == 2 {
return true
}
return false
}
// IsPigYear reports whether is year of Pig.
// 是否是猪年
func (l lunar) IsPigYear() bool {
if l.isInvalid {
return false
}
if l.year%MonthsPerYear == 3 {
return true
}
return false
}
// DoubleHour gets double-hour name like "子时".
// 获取当前时辰
func (l lunar) DoubleHour() (dh string) {
if l.isInvalid {
return ""
}
hour, minute := l.hour, l.minute
switch {
case hour >= 23, hour == 0 && minute <= 59:
dh = lunarTimes[0] // FirstDoubleHour
case hour >= 1 && hour < 3:
dh = lunarTimes[1] // SecondDoubleHour
case hour >= 3 && hour < 5:
dh = lunarTimes[2] // ThirdDoubleHour
case hour >= 5 && hour < 7:
dh = lunarTimes[3] // FourthDoubleHour
case hour >= 7 && hour < 9:
dh = lunarTimes[4] // FifthDoubleHour
case hour >= 9 && hour < 11:
dh = lunarTimes[5] // SixthDoubleHour
case hour >= 11 && hour < 13:
dh = lunarTimes[6] // SeventhDoubleHour
case hour >= 13 && hour < 15:
dh = lunarTimes[7] // EighthDoubleHour
case hour >= 15 && hour < 17:
dh = lunarTimes[8] // NinthDoubleHour
case hour >= 17 && hour < 19:
dh = lunarTimes[9] // TenthDoubleHour
case hour >= 19 && hour < 21:
dh = lunarTimes[10] // EleventhDoubleHour
case hour >= 21 && hour < 23:
dh = lunarTimes[11] // TwelfthDoubleHour
}
return
}
// IsFirstDoubleHour reports whether is FirstDoubleHour.
// 是否是子时
func (l lunar) IsFirstDoubleHour() bool {
if l.isInvalid {
return false
}
hour, minute := l.hour, l.minute
if hour >= 23 {
return true
}
if hour == 0 && minute <= 59 {
return true
}
return false
}
// IsSecondDoubleHour reports whether is SecondDoubleHour.
// 是否是丑时
func (l lunar) IsSecondDoubleHour() bool {
if l.isInvalid {
return false
}
hour := l.hour
if hour >= 1 && hour < 3 {
return true
}
return false
}
// IsThirdDoubleHour reports whether is ThirdDoubleHour.
// 是否是寅时
func (l lunar) IsThirdDoubleHour() bool {
if l.isInvalid {
return false
}
hour := l.hour
if hour >= 3 && hour < 5 {
return true
}
return false
}
// IsFourthDoubleHour reports whether is FourthDoubleHour.
// 是否是卯时
func (l lunar) IsFourthDoubleHour() bool {
if l.isInvalid {
return false
}
hour := l.hour
if hour >= 5 && hour < 7 {
return true
}
return false
}
// IsFifthDoubleHour reports whether is FifthDoubleHour.
// 是否是辰时
func (l lunar) IsFifthDoubleHour() bool {
if l.isInvalid {
return false
}
hour := l.hour
if hour >= 7 && hour < 9 {
return true
}
return false
}
// IsSixthDoubleHour reports whether is SixthDoubleHour.
// 是否是巳时
func (l lunar) IsSixthDoubleHour() bool {
if l.isInvalid {
return false
}
hour := l.hour
if hour >= 9 && hour < 11 {
return true
}
return false
}
// IsSeventhDoubleHour reports whether is SeventhDoubleHour.
// 是否是午时
func (l lunar) IsSeventhDoubleHour() bool {
if l.isInvalid {
return false
}
hour := l.hour
if hour >= 11 && hour < 13 {
return true
}
return false
}
// IsEighthDoubleHour reports whether is EighthDoubleHour.
// 是否是未时
func (l lunar) IsEighthDoubleHour() bool {
if l.isInvalid {
return false
}
hour := l.hour
if hour >= 13 && hour < 15 {
return true
}
return false
}
// IsNinthDoubleHour reports whether is NinthDoubleHour.
// 是否是申时
func (l lunar) IsNinthDoubleHour() bool {
if l.isInvalid {
return false
}
hour := l.hour
if hour >= 15 && hour < 17 {
return true
}
return false
}
// IsTenthDoubleHour reports whether is TenthDoubleHour.
// 是否是酉时
func (l lunar) IsTenthDoubleHour() bool {
if l.isInvalid {
return false
}
hour := l.hour
if hour >= 17 && hour < 19 {
return true
}
return false
}
// IsEleventhDoubleHour reports whether is EleventhDoubleHour.
// 是否是戌时
func (l lunar) IsEleventhDoubleHour() bool {
if l.isInvalid {
return false
}
hour := l.hour
if hour >= 19 && hour < 21 {
return true
}
return false
}
// IsTwelfthDoubleHour reports whether is TwelfthDoubleHour.
// 是否是亥时
func (l lunar) IsTwelfthDoubleHour() bool {
if l.isInvalid {
return false
}
hour := l.hour
if hour >= 21 && hour < 23 {
return true
}
return false
}
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
Go
1
https://gitee.com/jiurenm/carbon.git
git@gitee.com:jiurenm/carbon.git
jiurenm
carbon
carbon
master

搜索帮助