日期转换农历

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;
    }


}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值