原生js实现公历转农历(转载)

网上原生JS实现公历转农历的代码参差不齐,个人经过项目实践后发现有个博主的写的很好使用起来也很方便,于是一起分享给大家。
原博主的代码是在vue框架下写的,具体如下:

  /* 公历转农历代码思路:
1、建立农历年份查询表
2、计算输入公历日期与公历基准的相差天数
3、从农历基准开始遍历农历查询表,计算自农历基准之后每一年的天数,并用相差天数依次相减,确定农历年份
4、利用剩余相差天数以及农历每个月的天数确定农历月份
5、利用剩余相差天数确定农历哪一天 */// 农历1949-2100年查询表
export function sloarToLunar(sy, sm, sd) {let lunarYearArr = [0x0b557, //19490x06ca0, 0x0b550, 0x15355, 0x04da0, 0x0a5b0, 0x14573, 0x052b0, 0x0a9a8, 0x0e950, 0x06aa0, //1950-19590x0aea6, 0x0ab50, 0x04b60, 0x0aae4, 0x0a570, 0x05260, 0x0f263, 0x0d950, 0x05b57, 0x056a0, //1960-19690x096d0, 0x04dd5, 0x04ad0, 0x0a4d0, 0x0d4d4, 0x0d250, 0x0d558, 0x0b540, 0x0b6a0, 0x195a6, //1970-19790x095b0, 0x049b0, 0x0a974, 0x0a4b0, 0x0b27a, 0x06a50, 0x06d40, 0x0af46, 0x0ab60, 0x09570, //1980-19890x04af5, 0x04970, 0x064b0, 0x074a3, 0x0ea50, 0x06b58, 0x055c0, 0x0ab60, 0x096d5, 0x092e0, //1990-19990x0c960, 0x0d954, 0x0d4a0, 0x0da50, 0x07552, 0x056a0, 0x0abb7, 0x025d0, 0x092d0, 0x0cab5, //2000-20090x0a950, 0x0b4a0, 0x0baa4, 0x0ad50, 0x055d9, 0x04ba0, 0x0a5b0, 0x15176, 0x052b0, 0x0a930, //2010-20190x07954, 0x06aa0, 0x0ad50, 0x05b52, 0x04b60, 0x0a6e6, 0x0a4e0, 0x0d260, 0x0ea65, 0x0d530, //2020-20290x05aa0, 0x076a3, 0x096d0, 0x04afb, 0x04ad0, 0x0a4d0, 0x1d0b6, 0x0d250, 0x0d520, 0x0dd45, //2030-20390x0b5a0, 0x056d0, 0x055b2, 0x049b0, 0x0a577, 0x0a4b0, 0x0aa50, 0x1b255, 0x06d20, 0x0ada0, //2040-20490x14b63, 0x09370, 0x049f8, 0x04970, 0x064b0, 0x168a6, 0x0ea50, 0x06b20, 0x1a6c4, 0x0aae0, //2050-20590x0a2e0, 0x0d2e3, 0x0c960, 0x0d557, 0x0d4a0, 0x0da50, 0x05d55, 0x056a0, 0x0a6d0, 0x055d4, //2060-20690x052d0, 0x0a9b8, 0x0a950, 0x0b4a0, 0x0b6a6, 0x0ad50, 0x055a0, 0x0aba4, 0x0a5b0, 0x052b0, //2070-20790x0b273, 0x06930, 0x07337, 0x06aa0, 0x0ad50, 0x14b55, 0x04b60, 0x0a570, 0x054e4, 0x0d160, //2080-20890x0e968, 0x0d520, 0x0daa0, 0x16aa6, 0x056d0, 0x04ae0, 0x0a9d4, 0x0a2d0, 0x0d150, 0x0f252, //2090-20990x0d520 //2100],lunarMonth = ['正', '二', '三', '四', '五', '六', '七', '八', '九', '十', '冬', '腊'],lunarDay = ['一', '二', '三', '四', '五', '六', '七', '八', '九', '十', '初', '廿'],tianGan = ['甲', '乙', '丙', '丁', '戊', '己', '庚', '辛', '壬', '癸'],diZhi = ['子', '丑', '寅', '卯', '辰', '巳', '午', '未', '申', '酉', '戌', '亥'];// 公历转农历函数
function sloarToLunar(sy, sm, sd) {// 输入的月份减1处理sm -= 1;// 计算与公历基准的相差天数// Date.UTC()返回的是距离公历1970年1月1日的毫秒数,传入的月份需要减1let daySpan = (Date.UTC(sy, sm, sd) - Date.UTC(1949, 0, 29)) / (24 * 60 * 60 * 1000) + 1;let ly, lm, ld;// 确定输出的农历年份for (let j = 0; j < lunarYearArr.length; j++) {daySpan -= lunarYearDays(lunarYearArr[j]);if (daySpan <= 0) {ly = 1949 + j;// 获取农历年份确定后的剩余天数daySpan += lunarYearDays(lunarYearArr[j]);break}}// 确定输出的农历月份for (let k = 0; k < lunarYearMonths(lunarYearArr[ly - 1949]).length; k++) {daySpan -= lunarYearMonths(lunarYearArr[ly - 1949])[k];if (daySpan <= 0) {// 有闰月时,月份的数组长度会变成13,因此,当闰月月份小于等于k时,lm不需要加1if (hasLeapMonth(lunarYearArr[ly - 1949]) && hasLeapMonth(lunarYearArr[ly - 1949]) <= k) {if (hasLeapMonth(lunarYearArr[ly - 1949]) < k) {lm = k;} else if (hasLeapMonth(lunarYearArr[ly - 1949]) === k) {lm = '闰' + k;} else {lm = k + 1;}} else {lm = k + 1;}// 获取农历月份确定后的剩余天数daySpan += lunarYearMonths(lunarYearArr[ly - 1949])[k];break}}// 确定输出农历哪一天ld = daySpan;// 将计算出来的农历月份转换成汉字月份,闰月需要在前面加上闰字if (hasLeapMonth(lunarYearArr[ly - 1949]) && (typeof (lm) === 'string' && lm.indexOf('闰') > -1)) {lm = `闰${lunarMonth[/\d/.exec(lm) - 1]}`} else {lm = lunarMonth[lm - 1];}// 将计算出来的农历年份转换为天干地支年ly = getTianGan(ly) + getDiZhi(ly);// 将计算出来的农历天数转换成汉字if (ld < 11) {ld = `${lunarDay[10]}${lunarDay[ld-1]}`} else if (ld > 10 && ld < 20) {ld = `${lunarDay[9]}${lunarDay[ld-11]}`} else if (ld === 20) {ld = `${lunarDay[1]}${lunarDay[9]}`} else if (ld > 20 && ld < 30) {ld = `${lunarDay[11]}${lunarDay[ld-21]}`} else if (ld === 30) {ld = `${lunarDay[2]}${lunarDay[9]}`}return {lunarYear: ly,lunarMonth: lm,lunarDay: ld,}
}// 计算农历年是否有闰月,参数为存储农历年的16进制
// 农历年份信息用16进制存储,其中16进制的最后1位可以用于判断是否有闰月
function hasLeapMonth(ly) {// 获取16进制的最后1位,需要用到&与运算符if (ly & 0xf) {return ly & 0xf} else {return false}
}// 如果有闰月,计算农历闰月天数,参数为存储农历年的16进制
// 农历年份信息用16进制存储,其中16进制的第1位(0x除外)可以用于表示闰月是大月还是小月
function leapMonthDays(ly) {if (hasLeapMonth(ly)) {// 获取16进制的第1位(0x除外)return (ly & 0xf0000) ? 30 : 29} else {return 0}
}// 计算农历一年的总天数,参数为存储农历年的16进制
// 农历年份信息用16进制存储,其中16进制的第2-4位(0x除外)可以用于表示正常月是大月还是小月
function lunarYearDays(ly) {let totalDays = 0;// 获取正常月的天数,并累加// 获取16进制的第2-4位,需要用到>>移位运算符for (let i = 0x8000; i > 0x8; i >>= 1) {let monthDays = (ly & i) ? 30 : 29;totalDays += monthDays;}// 如果有闰月,需要把闰月的天数加上if (hasLeapMonth(ly)) {totalDays += leapMonthDays(ly);}return totalDays
}// 获取农历每个月的天数
// 参数需传入16进制数值
function lunarYearMonths(ly) {let monthArr = [];// 获取正常月的天数,并添加到monthArr数组中// 获取16进制的第2-4位,需要用到>>移位运算符for (let i = 0x8000; i > 0x8; i >>= 1) {monthArr.push((ly & i) ? 30 : 29);}// 如果有闰月,需要把闰月的天数加上if (hasLeapMonth(ly)) {monthArr.splice(hasLeapMonth(ly), 0, leapMonthDays(ly));}return monthArr
}// 将农历年转换为天干,参数为农历年
function getTianGan(ly) {let tianGanKey = (ly - 3) % 10;if (tianGanKey === 0) tianGanKey = 10;return tianGan[tianGanKey - 1]
}// 将农历年转换为地支,参数为农历年
function getDiZhi(ly) {let diZhiKey = (ly - 3) % 12;if (diZhiKey === 0) diZhiKey = 12;return diZhi[diZhiKey - 1]
}
return sloarToLunar(sy, sm, sd)
}

然后调用的时候只需要像下图代码中调用即可:

 sloarToLunar(2021, 10, 10).lunarYear +"年" +sloarToLunar(2021, 10, 10).lunarMonth +"月" +sloarToLunar(2021, 10, 10).lunarDay

原博主博客链接如下:
原生js实现公历转农历

js获取当前农历年月 原生js实现公历转农历相关推荐

  1. class 原生js获取父元素_原生js获取class

    //使用原生js时,通过class名称就可以得到相应的class名称标签组封装的函数 //定义一个函数getClass(oParent,aClass); function getClass(oPare ...

  2. 原生js获取html元素高度,原生JS获取元素宽高实践详解

    开篇的话 任何不是亲身实践中求得的知识,都不是属于你的. 任何求得的知识不去时常温习运用,也不是属于你的. 记录由来 在做个上拉广告功能中遇到了一个"理所当然"觉得对的用法,慢慢才 ...

  3. js 封装ajax方法吗,原生JS封装ajax方法

    jquery框架的ajax方法固然好用,但是假如某天我们的项目不能引入jquery或项目需求很简单,没有很多交互功能,只需要ajax,这时引入jquery库会造成资源浪费,也会显得页面臃肿.这时我们就 ...

  4. 原生js写三级联动 java_原生js三级联动的简单实现代码

    本文实例为大家分享了js查询天气应用,供大家参考,具体内容如下 实现功能:打开网页时显示用户所在城市的天气状况,在输入框输入城市可查询其它城市. 实现过程:先调用百度地图的API来获取用户所在的城市, ...

  5. js异步请求php数据,原生JS发送异步数据请求实例详解

    这篇文章主要为大家详细介绍了原生JS发送异步数据请求的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下 在做项目的时候,有时候需要用到异步数据请求,但是如果这个时候没有框架的依赖,就需要用到 ...

  6. js 写html代码编辑器,原生JS写一个功能强大的编辑器

    因为一个同学,要做一个能加入图片的留言板功能,类型与QQ空间留言板和百度贴吧发帖的那种形式,同时在网上找了找发生网上对这方面的交流很少,所以发表这篇文章抛砖引玉,希望能帮助广大的学习者,也同时希望大佬 ...

  7. vue如何使用原生js写动画效果_原生js写一个无缝轮播图插件(支持vue)

    轮播图插件(Broadcast.js) 前言:写这个插件的原因 前段时间准备用vue加上网易云的nodejs接口,模拟网易云音乐移动端.因为想自己写一遍所有的代码以及加固自己的flex布局,所以没有使 ...

  8. html原生js实现图片轮播,原生JS实现图片轮播切换效果

    原生JS实现图片轮播切换效果 2019-01-06 编程之家 https://www.jb51.cc 编程之家收集整理的这篇文章主要介绍了原生JS实现图片轮播切换效果,编程之家小编觉得挺不错的,现在分 ...

  9. js获取classname值_利用js获取元素class值的两种方法

    我们有时为了达到某种效果,需要以元素的class值为条件做判断. 我们如何利用JavaScript获取元素class的值?我们先看下面代码: x=document.getElementsByTagNa ...

最新文章

  1. SAP MM 采购发票上的价格与采购订单上价格的差异
  2. 划分vlan,制作trunk口。使同一vlan能互相通讯
  3. 给自己的Sublime Text换上Soda Theme主题后清爽了好多
  4. Webpack 的 HtmlWebpackPlugin 如何控制某个 chunks 的 inject 位置?
  5. linux模式匹配,sed的模式匹配用法探讨
  6. 阿里云应用实战-IoT工业数据上云优化实践
  7. docker-compose 使用小例
  8. 单机最大负载_电流互感器允许接入的实际最大二次负载(注电案例1865)
  9. 第5章第24节:如何在幻灯片中播放视频文件 [PowerPoint精美幻灯片实战教程]
  10. JAVA 经纬度转换成直角坐标系,以及直角坐标系转换成经纬度算法
  11. vcard微信电子名片
  12. 图片怎么识别文字?这几个方法很实用
  13. tp5使用xunsearch
  14. Qt 应用程序输出中文乱码+UI界面输出中文乱码
  15. Docker-配置私有仓库
  16. 学习笔记之-51单片机IO口详解
  17. Tableau 中国教育水平发展指标历史数据(一)教育公共开支总额和GDP
  18. facebook新闻页ListView优化
  19. Android中layout-sw600dp、layout-w600dp和layout-h600dp的区别
  20. 解决 FROM keyword not found where expected

热门文章

  1. 08-F. 电视机与遥控器(友元类)
  2. having 聚合函数
  3. qq本次操作由于本台计算机,本次操作由于这台计算机限制而被取消(最新解决方案).doc...
  4. sql 后台运行远程服务器,在SQLServer中通过.NET远程的执行SQL文件
  5. 还在人工巡查教室?在线巡课一步到位
  6. 线性代数笔记20——行列式和代数余子式
  7. 交通车辆工程机械高级仪器设备用丙烯酸磁漆 耐化学品性能好
  8. 两款!微软将在E3展上公布新Xbox主机细节
  9. vue-cli创建脚手架工程
  10. NS版暗黑破坏神3金手指开发教程(17)