class Lunar{
static $lunarInfo= array(//1900-2050
0x04bd8,0x04ae0,0x0a570,0x054d5,0x0d260,0x0d950,0x16554,0x056a0,0x09ad0,0x055d2,
0x04ae0,0x0a5b6,0x0a4d0,0x0d250,0x1d255,0x0b540,0x0d6a0,0x0ada2,0x095b0,0x14977,
0x04970,0x0a4b0,0x0b4b5,0x06a50,0x06d40,0x1ab54,0x02b60,0x09570,0x052f2,0x04970,
0x06566,0x0d4a0,0x0ea50,0x06e95,0x05ad0,0x02b60,0x186e3,0x092e0,0x1c8d7,0x0c950,
0x0d4a0,0x1d8a6,0x0b550,0x056a0,0x1a5b4,0x025d0,0x092d0,0x0d2b2,0x0a950,0x0b557,
0x06ca0,0x0b550,0x15355,0x04da0,0x0a5d0,0x14573,0x052d0,0x0a9a8,0x0e950,0x06aa0,
0x0aea6,0x0ab50,0x04b60,0x0aae4,0x0a570,0x05260,0x0f263,0x0d950,0x05b57,0x056a0,
0x096d0,0x04dd5,0x04ad0,0x0a4d0,0x0d4d4,0x0d250,0x0d558,0x0b540,0x0b5a0,0x195a6,
0x095b0,0x049b0,0x0a974,0x0a4b0,0x0b27a,0x06a50,0x06d40,0x0af46,0x0ab60,0x09570,
0x04af5,0x04970,0x064b0,0x074a3,0x0ea50,0x06b58,0x055c0,0x0ab60,0x096d5,0x092e0,
0x0c960,0x0d954,0x0d4a0,0x0da50,0x07552,0x056a0,0x0abb7,0x025d0,0x092d0,0x0cab5,
0x0a950,0x0b4a0,0x0baa4,0x0ad50,0x055d9,0x04ba0,0x0a5b0,0x15176,0x052b0,0x0a930,
0x07954,0x06aa0,0x0ad50,0x05b52,0x04b60,0x0a6e6,0x0a4e0,0x0d260,0x0ea65,0x0d530,
0x05aa0,0x076a3,0x096d0,0x04bd7,0x04ad0,0x0a4d0,0x1d0b6,0x0d250,0x0d520,0x0dd45,
0x0b5a0,0x056d0,0x055b2,0x049b0,0x0a577,0x0a4b0,0x0aa50,0x1b255,0x06d20,0x0ada0);
//传入一个 2010-12-28
static public function getLar($date,$ty = false){
$objDate = array();//需要解析的数组
list($objDate['year'],$objDate['mon'],$objDate['mday']) = explode('-', $date);
if(empty($objDate['year']) || $objDate['year']<1900 || $objDate['year']>2050){
return '';
}
if($ty){
$dateArr = self::Lunar2Solar($objDate);
}else{
$dateArr = self::Solar2Lunar($objDate);
}
return $dateArr;
}
//y年的总天数
static private function yearHasDays($y) {
return ($total=365 + self::is_yleap($y));
}
//是否闰年: 返回1(代表闰年)或0(非闰年)
static private function is_yleap($y) {
return($y%100==0 ? $result=($y%400==0 ? 1 : 0) : $result=($y%4==0 ? 1 : 0));
}
//y年m月的总天数
static private function monDays($y,$m) {
switch($m) {
case 1: case 3: case 5: case 7: case 8: case 10: case 12:$total=31; break;
case 4: case 6: case 9: case 11: $total=30; break;
case 2: $total=28+self::is_yleap($y);break;
}
return $total;
}
//====================================== 传回农历 y年m月的总天数
static private function lMonDays($y,$m) {
return ((self::$lunarInfo[$y-1900] & (0x10000>>$m))? 30: 29 );
}
//==== 传回农历 y年的总天数
static private function lYearDays($y) {
$sum = 348;
for($i=0x8000; $i>0x8; $i>>=1) $sum += (self::$lunarInfo[$y-1900] & $i) ? 1: 0;
return ($sum+self::leapDays($y));
}
//==== 传回农历 y年闰月的天数
static private function leapDays($y) {
if(self::leapMonth($y)) return (self::$lunarInfo[$y-1900] & 0x10000)? 30: 29;
else return 0;
}
//==== 传回农历 y年闰哪个月 1-12 , 没闰传回 0
static private function leapMonth($y) {
return(self::$lunarInfo[$y-1900] & 0xf);
}
static private function Lunar2Solar($date){
$offset = $year = $month = 0;
for($i = 1900;$i< $date['year'];$i++) $offset += self::lYearDays($i);
//补回闰年
$is_yun = self::leapMonth($date['year']);
if($is_yun< $date['mon'] && $is_yun>0) $offset +=self::leapDays($date['year']);
for($i = 1;$i< $date['mon'];$i++) $offset += self::lMonDays($date['year'],$i);
$offset += $date['mday']-1;
for($i = 1900; $i<2050 && $offset>0; $i++){
$days = self::yearHasDays($i);
if($i == 1900) $days -=31;
if($offset >= $days){
$year = $i+1;
$offset -= $days;
}else break;
}
for($i=1;$i<12;$i++){
$month = $i;
$localDays = self::monDays($year,$i);
if($offset >= $localDays){
$month += 1;
$offset = $offset - $localDays;
}else break;
}
if($offset == 0) $offset = 1;
return $year . '-' . $month . '-' .$offset;
}
//==== 算出农历, 传入日期物件, 传回农历日期物件
//$objDate (array('year'=>2010,'mday'=>28,'mon'=>12))年日月的
static public function Solar2Lunar($date) {
$offset = 0;
for($i = 1900;$i<$date['year'];$i++) $offset += self::yearHasDays($i);
$offset = $offset -31 + $date['mday'];
for($i=1;$i<$date['mon'];$i++) $offset += self::monDays($date['year'],$i) ;
for($i=1900; $i<2050 && $offset>0; $i++) {$temp = self::lYearDays($i); $offset -= $temp;}
if($offset<0) { $offset += $temp; $i--; }
$year = $i;
$leap = self::leapMonth($i); //闰哪个月
$isLeap = false;
for($i=1; $i<13 && $offset>0; $i++) {
//闰月
if($leap>0 && $i==($leap+1) && $isLeap==false){
--$i; $isLeap = true; $temp = self::leapDays($year);
}
else $temp = self::lMonDays($year, $i);
//解除闰月
if($isLeap==true && $i==($leap+1)) $isLeap = false;
$offset -= $temp;
}
if($offset==0 && $leap>0 && $i==$leap+1){
if($isLeap) $isLeap = false;
else{
$isLeap = true; --$i;
}
}
if($offset<0){
$offset += $temp; --$i;
}
$month = $i;
$day = $offset + 1;
return $year . '-' . $month . '-' .$day;
}
}
日期转换农历
最新推荐文章于 2022-05-24 10:07:19 发布