php類:php
1
2 /**
3 *PHP編程實現陽曆轉換爲陰曆的方法
4 *根據實際狀況所需進行調用
5 *
6 /
7 8 9
10 11 <?php 12 class Lunar 13 { 14 public $MIN_YEAR = 1891; 15 public $MAX_YEAR = 2100; 16 public $lunarInfo = array( 17 array(0, 2, 9, 21936), array(6, 1, 30, 9656), array(0, 2, 17, 9584), array(0, 2, 6, 21168), array(5, 1, 26, 43344), array(0, 2, 13, 59728), 18 array(0, 2, 2, 27296), array(3, 1, 22, 44368), array(0, 2, 10, 43856), array(8, 1, 30, 19304), array(0, 2, 19, 19168), array(0, 2, 8, 42352), 19 array(5, 1, 29, 21096), array(0, 2, 16, 53856), array(0, 2, 4, 55632), array(4, 1, 25, 27304), array(0, 2, 13, 22176), array(0, 2, 2, 39632), 20 array(2, 1, 22, 19176), array(0, 2, 10, 19168), array(6, 1, 30, 42200), array(0, 2, 18, 42192), array(0, 2, 6, 53840), array(5, 1, 26, 54568), 21 array(0, 2, 14, 46400), array(0, 2, 3, 54944), array(2, 1, 23, 38608), array(0, 2, 11, 38320), array(7, 2, 1, 18872), array(0, 2, 20, 18800), 22 array(0, 2, 8, 42160), array(5, 1, 28, 45656), array(0, 2, 16, 27216), array(0, 2, 5, 27968), array(4, 1, 24, 44456), array(0, 2, 13, 11104), 23 array(0, 2, 2, 38256), array(2, 1, 23, 18808), array(0, 2, 10, 18800), array(6, 1, 30, 25776), array(0, 2, 17, 54432), array(0, 2, 6, 59984), 24 array(5, 1, 26, 27976), array(0, 2, 14, 23248), array(0, 2, 4, 11104), array(3, 1, 24, 37744), array(0, 2, 11, 37600), array(7, 1, 31, 51560), 25 array(0, 2, 19, 51536), array(0, 2, 8, 54432), array(6, 1, 27, 55888), array(0, 2, 15, 46416), array(0, 2, 5, 22176), array(4, 1, 25, 43736), 26 array(0, 2, 13, 9680), array(0, 2, 2, 37584), array(2, 1, 22, 51544), array(0, 2, 10, 43344), array(7, 1, 29, 46248), array(0, 2, 17, 27808), 27 array(0, 2, 6, 46416), array(5, 1, 27, 21928), array(0, 2, 14, 19872), array(0, 2, 3, 42416), array(3, 1, 24, 21176), array(0, 2, 12, 21168), 28 array(8, 1, 31, 43344), array(0, 2, 18, 59728), array(0, 2, 8, 27296), array(6, 1, 28, 44368), array(0, 2, 15, 43856), array(0, 2, 5, 19296), 29 array(4, 1, 25, 42352), array(0, 2, 13, 42352), array(0, 2, 2, 21088), array(3, 1, 21, 59696), array(0, 2, 9, 55632), array(7, 1, 30, 23208), 30 array(0, 2, 17, 22176), array(0, 2, 6, 38608), array(5, 1, 27, 19176), array(0, 2, 15, 19152), array(0, 2, 3, 42192), array(4, 1, 23, 53864), 31 array(0, 2, 11, 53840), array(8, 1, 31, 54568), array(0, 2, 18, 46400), array(0, 2, 7, 46752), array(6, 1, 28, 38608), array(0, 2, 16, 38320), 32 array(0, 2, 5, 18864), array(4, 1, 25, 42168), array(0, 2, 13, 42160), array(10, 2, 2, 45656), array(0, 2, 20, 27216), array(0, 2, 9, 27968), 33 array(6, 1, 29, 44448), array(0, 2, 17, 43872), array(0, 2, 6, 38256), array(5, 1, 27, 18808), array(0, 2, 15, 18800), array(0, 2, 4, 25776), 34 array(3, 1, 23, 27216), array(0, 2, 10, 59984), array(8, 1, 31, 27432), array(0, 2, 19, 23232), array(0, 2, 7, 43872), array(5, 1, 28, 37736), 35 array(0, 2, 16, 37600), array(0, 2, 5, 51552), array(4, 1, 24, 54440), array(0, 2, 12, 54432), array(0, 2, 1, 55888), array(2, 1, 22, 23208), 36 array(0, 2, 9, 22176), array(7, 1, 29, 43736), array(0, 2, 18, 9680), array(0, 2, 7, 37584), array(5, 1, 26, 51544), array(0, 2, 14, 43344), 37 array(0, 2, 3, 46240), array(4, 1, 23, 46416), array(0, 2, 10, 44368), array(9, 1, 31, 21928), array(0, 2, 19, 19360), array(0, 2, 8, 42416), 38 array(6, 1, 28, 21176), array(0, 2, 16, 21168), array(0, 2, 5, 43312), array(4, 1, 25, 29864), array(0, 2, 12, 27296), array(0, 2, 1, 44368), 39 array(2, 1, 22, 19880), array(0, 2, 10, 19296), array(6, 1, 29, 42352), array(0, 2, 17, 42208), array(0, 2, 6, 53856), array(5, 1, 26, 59696), 40 array(0, 2, 13, 54576), array(0, 2, 3, 23200), array(3, 1, 23, 27472), array(0, 2, 11, 38608), array(11, 1, 31, 19176), array(0, 2, 19, 19152), 41 array(0, 2, 8, 42192), array(6, 1, 28, 53848), array(0, 2, 15, 53840), array(0, 2, 4, 54560), array(5, 1, 24, 55968), array(0, 2, 12, 46496), 42 array(0, 2, 1, 22224), array(2, 1, 22, 19160), array(0, 2, 10, 18864), array(7, 1, 30, 42168), array(0, 2, 17, 42160), array(0, 2, 6, 43600), 43 array(5, 1, 26, 46376), array(0, 2, 14, 27936), array(0, 2, 2, 44448), array(3, 1, 23, 21936), array(0, 2, 11, 37744), array(8, 2, 1, 18808), 44 array(0, 2, 19, 18800), array(0, 2, 8, 25776), array(6, 1, 28, 27216), array(0, 2, 15, 59984), array(0, 2, 4, 27424), array(4, 1, 24, 43872), 45 array(0, 2, 12, 43744), array(0, 2, 2, 37600), array(3, 1, 21, 51568), array(0, 2, 9, 51552), array(7, 1, 29, 54440), array(0, 2, 17, 54432), 46 array(0, 2, 5, 55888), array(5, 1, 26, 23208), array(0, 2, 14, 22176), array(0, 2, 3, 42704), array(4, 1, 23, 21224), array(0, 2, 11, 21200), 47 array(8, 1, 31, 43352), array(0, 2, 19, 43344), array(0, 2, 7, 46240), array(6, 1, 27, 46416), array(0, 2, 15, 44368), array(0, 2, 5, 21920), 48 array(4, 1, 24, 42448), array(0, 2, 12, 42416), array(0, 2, 2, 21168), array(3, 1, 22, 43320), array(0, 2, 9, 26928), array(7, 1, 29, 29336), 49 array(0, 2, 17, 27296), array(0, 2, 6, 44368), array(5, 1, 26, 19880), array(0, 2, 14, 19296), array(0, 2, 3, 42352), array(4, 1, 24, 21104), 50 array(0, 2, 10, 53856), array(8, 1, 30, 59696), array(0, 2, 18, 54560), array(0, 2, 7, 55968), array(6, 1, 27, 27472), array(0, 2, 15, 22224), 51 array(0, 2, 5, 19168), array(4, 1, 25, 42216), array(0, 2, 12, 42192), array(0, 2, 1, 53584), array(2, 1, 21, 55592), array(0, 2, 9, 54560) 52 ); 53 54 /** 55 * 將陽曆轉換爲陰曆 56 * @param year 公曆-年 57 * @param month 公曆-月 58 * @param date 公曆-日 59 */ 60 public function convertSolarToLunar($year, $month, $date) 61 { 62 $yearData = $this->lunarInfo[$year - $this->MIN_YEAR]; 63 if ($year == $this->MIN_YEAR && $month <= 2 && $date <= 9) return array(1891, '正月', '初一', '辛卯', 1, 1, '兔'); 64 return $this->getLunarByBetween($year, $this->getDaysBetweenSolar($year, $month, $date, $yearData[1], $yearData[2])); 65 } 66 67 public function convertSolarMonthToLunar($year, $month, $date) 68 { 69 $yearData = $this->lunarInfo[$year - $this->MIN_YEAR]; 70 if ($year == $this->MIN_YEAR && $month <= 2 && $date <= 9) return array(1891, '正月', '初一', '辛卯', 1, 1, '兔'); 71 $month_days_ary = array(31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31); 72 $dd = $month_days_ary[$month]; 73 if ($this->isLeapYear($year) && $month == 2) $dd++; 74 $lunar_ary = array(); 75 for ($i = 1; $i < $dd; $i++) { 76 $array = $this->getLunarByBetween($year, $this->getDaysBetweenSolar($year, $month, $i, $yearData[1], $yearData[2])); 77 $array[] = $year . '-' . $month . '-' . $i; 78 $lunar_ary[$i] = $array; 79 } 80 return $lunar_ary; 81 } 82 /** 83 * 將陰曆轉換爲陽曆(潤月處理) 84 * @param year 陰曆-年 85 * @param month 陰曆-月,閏月處理:例如若是當年閏五月,那麼第二個五月就傳六月,至關於陰曆有13個月,只是有的時候第13個月的天數爲0 86 * @param date 陰曆-日 87 */ 88 public function convertLunarToSolarLeapMonth($year, $month, $date) 89 { 90 $leapMonth = $this->getLeapMonth($year);//獲取閏月 91 $month = intval($month); 92 if ($leapMonth > 0) { 93 //判斷若是是閏月而且單前月大於潤月的話 +1 94 if ($month > $leapMonth) { 95 $month = $month + 1; 96 } 97 } 98 $month = strval($month); 99 return $this->convertLunarToSolar($year, $month, $date); 100 } 101 102 /** 103 * 將陰曆轉換爲陽曆 104 * @param year 陰曆-年 105 * @param month 陰曆-月,閏月處理:例如若是當年閏五月,那麼第二個五月就傳六月,至關於陰曆有13個月,只是有的時候第13個月的天數爲0 106 * @param date 陰曆-日 107 */ 108 public function convertLunarToSolar($year, $month, $date) 109 { 110 $yearData = $this->lunarInfo[$year - $this->MIN_YEAR]; 111 $between = $this->getDaysBetweenLunar($year, $month, $date); 112 $res = mktime(0, 0, 0, $yearData[1], $yearData[2], $year); 113 $res = date('Y-m-d', $res + $between * 24 * 60 * 60); 114 $day = explode('-', $res); 115 $year = $day[0]; 116 $month = $day[1]; 117 $day = $day[2]; 118 return array($year, $month, $day); 119 } 120 121 /** 122 * 判斷是不是閏年 123 * @param year 124 */ 125 public function isLeapYear($year) 126 { 127 return (($year % 4 == 0 && $year % 100 != 0) || ($year % 400 == 0)); 128 } 129 130 /** 131 * 獲取干支紀年 132 * @param year 133 */ 134 public function getLunarYearName($year) 135 { 136 $sky = array('庚', '辛', '壬', '癸', '甲', '乙', '丙', '丁', '戊', '己'); 137 $earth = array('申', '酉', '戌', '亥', '子', '醜', '寅', '卯', '辰', '巳', '午', '未'); 138 $year = $year . ''; 139 return $sky[$year{3}] . $earth[$year % 12]; 140 } 141 142 /** 143 * 根據陰曆年獲取生肖 144 * @param year 陰曆年 145 */ 146 public function getYearZodiac($year) 147 { 148 $zodiac = array('猴', '雞', '狗', '豬', '鼠', '牛', '虎', '兔', '龍', '蛇', '馬', '羊'); 149 return $zodiac[$year % 12]; 150 } 151 152 /** 153 * 獲取陽曆月份的天數 154 * @param year 陽曆-年 155 * @param month 陽曆-月 156 */ 157 public function getSolarMonthDays($year, $month) 158 { 159 $monthHash = array('1' => 31, '2' => $this->isLeapYear($year) ? 29 : 28, '3' => 31, '4' => 30, '5' => 31, '6' => 30, '7' => 31, '8' => 31, '9' => 30, '10' => 31, '11' => 30, '12' => 31); 160 return $monthHash["$month"]; 161 } 162 163 /** 164 * 獲取陰曆月份的天數 165 * @param year 陰曆-年 166 * @param month 陰曆-月,從一月開始 167 */ 168 public function getLunarMonthDays($year, $month) 169 { 170 $monthData = $this->getLunarMonths($year); 171 return $monthData[$month - 1]; 172 } 173 174 /** 175 * 獲取陰曆每個月的天數的數組 176 * @param year 177 */ 178 public function getLunarMonths($year) 179 { 180 $yearData = $this->lunarInfo[$year - $this->MIN_YEAR]; 181 $leapMonth = $yearData[0]; 182 $bit = decbin($yearData[3]); 183 for ($i = 0; $i < strlen($bit); $i++) $bitArray[$i] = substr($bit, $i, 1); 184 for ($k = 0, $klen = 16 - count($bitArray); $k < $klen; $k++) array_unshift($bitArray, '0'); 185 $bitArray = array_slice($bitArray, 0, ($leapMonth == 0 ? 12 : 13)); 186 for ($i = 0; $i < count($bitArray); $i++) $bitArray[$i] = $bitArray[$i] + 29; 187 return $bitArray; 188 } 189 190 /** 191 * 獲取農曆每一年的天數 192 * @param year 農曆年份 193 */ 194 public function getLunarYearDays($year) 195 { 196 $yearData = $this->lunarInfo[$year - $this->MIN_YEAR]; 197 $monthArray = $this->getLunarYearMonths($year); 198 $len = count($monthArray); 199 return ($monthArray[$len - 1] == 0 ? $monthArray[$len - 2] : $monthArray[$len - 1]); 200 } 201 202 public function getLunarYearMonths($year) 203 { 204 $monthData = $this->getLunarMonths($year); 205 $res = array(); 206 $temp = 0; 207 $yearData = $this->lunarInfo[$year - $this->MIN_YEAR]; 208 $len = ($yearData[0] == 0 ? 12 : 13); 209 for ($i = 0; $i < $len; $i++) { 210 $temp = 0; 211 for ($j = 0; $j <= $i; $j++) $temp += $monthData[$j]; 212 array_push($res, $temp); 213 } 214 return $res; 215 } 216 217 /** 218 * 獲取閏月 219 * @param year 陰曆年份 220 */ 221 public function getLeapMonth($year) 222 { 223 $yearData = $this->lunarInfo[$year - $this->MIN_YEAR]; 224 return $yearData[0]; 225 } 226 227 /** 228 * 計算陰曆日期與正月初一相隔的天數 229 * @param year 230 * @param month 231 * @param date 232 */ 233 public function getDaysBetweenLunar($year, $month, $date) 234 { 235 $yearMonth = $this->getLunarMonths($year); 236 $res = 0; 237 for ($i = 1; $i < $month; $i++) $res += $yearMonth[$i - 1]; 238 $res += $date - 1; 239 return $res; 240 } 241 242 /** 243 * 計算2個陽曆日期之間的天數 244 * @param year 陽曆年 245 * @param cmonth 246 * @param cdate 247 * @param dmonth 陰曆正月對應的陽曆月份 248 * @param ddate 陰曆初一對應的陽曆天數 249 */ 250 public function getDaysBetweenSolar($year, $cmonth, $cdate, $dmonth, $ddate) 251 { 252 $a = mktime(0, 0, 0, $cmonth, $cdate, $year); 253 $b = mktime(0, 0, 0, $dmonth, $ddate, $year); 254 return ceil(($a - $b) / 24 / 3600); 255 } 256 257 /** 258 * 根據距離正月初一的天數計算陰曆日期 259 * @param year 陽曆年 260 * @param between 天數 261 */ 262 public function getLunarByBetween($year, $between) 263 { 264 $lunarArray = array(); 265 $yearMonth = array(); 266 $t = 0; 267 $e = 0; 268 $leapMonth = 0; 269 $m = ''; 270 if ($between == 0) { 271 array_push($lunarArray, $year, '正月', '初一'); 272 $t = 1; 273 $e = 1; 274 } else { 275 $year = $between > 0 ? $year : ($year - 1); 276 $yearMonth = $this->getLunarYearMonths($year); 277 $leapMonth = $this->getLeapMonth($year); 278 $between = $between > 0 ? $between : ($this->getLunarYearDays($year) + $between); 279 for ($i = 0; $i < 13; $i++) { 280 if ($between == $yearMonth[$i]) { 281 $t = $i + 2; 282 $e = 1; 283 break; 284 } else if ($between < $yearMonth[$i]) { 285 $t = $i + 1; 286 $e = $between - (empty($yearMonth[$i - 1]) ? 0 : $yearMonth[$i - 1]) + 1; 287 break; 288 } 289 } 290 $m = ($leapMonth != 0 && $t == $leapMonth + 1) ? ('閏' . $this->getCapitalNum($t - 1, true)) : $this->getCapitalNum(($leapMonth != 0 && $leapMonth + 1 < $t ? ($t - 1) : $t), true); 291 array_push($lunarArray, $year, $m, $this->getCapitalNum($e, false)); 292 } 293 array_push($lunarArray, $this->getLunarYearName($year));// 天干地支 294 array_push($lunarArray, $t, $e); 295 array_push($lunarArray, $this->getYearZodiac($year));// 12生肖 296 array_push($lunarArray, $leapMonth);// 閏幾月 297 return $lunarArray; 298 } 299 300 /** 301 * 獲取數字的陰曆叫法 302 * @param num 數字 303 * @param isMonth 是不是月份的數字 304 */ 305 public function getCapitalNum($num, $isMonth) 306 { 307 $isMonth = $isMonth || false; 308 $dateHash = array('0' => '', '1' => '一', '2' => '二', '3' => '三', '4' => '四', '5' => '五', '6' => '六', '7' => '七', '8' => '八', '9' => '九', '10' => '十 '); 309 $monthHash = array('0' => '', '1' => '正月', '2' => '二月', '3' => '三月', '4' => '四月', '5' => '五月', '6' => '六月', '7' => '七月', '8' => '八月', '9' => '九月', '10' => '十月', '11' => '冬月', '12' => '臘月'); 310 $res = ''; 311 if ($isMonth) $res = $monthHash[$num]; 312 else { 313 if ($num <= 10) $res = '初' . $dateHash[$num]; 314 else if ($num > 10 && $num < 20) $res = '十' . $dateHash[$num - 10]; 315 else if ($num == 20) $res = "二十"; 316 else if ($num > 20 && $num < 30) $res = "廿" . $dateHash[$num - 20]; 317 else if ($num == 30) $res = "三十"; 318 } 319 return $res; 320 } 321 322 /** 323 * 節氣通用算法 324 */ 325 public function getJieQi($_year, $month, $day) 326 { 327 $year = substr($_year, -2) + 0; 328 $coefficient = array( 329 array(5.4055, 2019, -1),//小寒 330 array(20.12, 2082, 1),//大寒 331 array(3.87),//立春 332 array(18.74, 2026, -1),//雨水 333 array(5.63),//驚蟄 334 array(20.646, 2084, 1),//春分 335 array(4.81),//清明 336 array(20.1),//穀雨 337 array(5.52, 1911, 1),//立夏 338 array(21.04, 2008, 1),//小滿 339 array(5.678, 1902, 1),//芒種 340 array(21.37, 1928, 1),//夏至 341 array(7.108, 2016, 1),//小暑 342 array(22.83, 1922, 1),//大暑 343 array(7.5, 2002, 1),//立秋 344 array(23.13),//處暑 345 array(7.646, 1927, 1),//白露 346 array(23.042, 1942, 1),//秋分 347 array(8.318),//寒露 348 array(23.438, 2089, 1),//霜降 349 array(7.438, 2089, 1),//立冬 350 array(22.36, 1978, 1),//小雪 351 array(7.18, 1954, 1),//大雪 352 array(21.94, 2021, -1)//冬至 353 ); 354 $term_name = array( 355 "小寒", "大寒", "立春", "雨水", "驚蟄", "春分", "清明", "穀雨", 356 "立夏", "小滿", "芒種", "夏至", "小暑", "大暑", "立秋", "處暑", 357 "白露", "秋分", "寒露", "霜降", "立冬", "小雪", "大雪", "冬至"); 358 $idx1 = ($month - 1) * 2; 359 $_leap_value = floor(($year - 1) / 4); 360 $day1 = floor($year * 0.2422 + $coefficient[$idx1][0]) - $_leap_value; 361 if (isset($coefficient[$idx1][1]) && $coefficient[$idx1][1] == $_year) $day1 += $coefficient[$idx1][2]; 362 $day2 = floor($year * 0.2422 + $coefficient[$idx1 + 1][0]) - $_leap_value; 363 if (isset($coefficient[$idx1 + 1][1]) && $coefficient[$idx1 + 1][1] == $_year) $day1 += $coefficient[$idx1 + 1][2]; 364 if ($day == $day1) return $term_name[$idx1]; 365 if ($day == $day2) return $term_name[$idx1 + 1]; 366 return ''; 367 } 368 369 /** 370 * 獲取節日:特殊的節日只能修改此函數來計算 371 */ 372 public function getFestival($today, $nl_info = false, $config = 1) 373 { 374 if ($config == 1) { 375 $arr_lunar = array('01-01' => '春節', '01-15' => '元宵節', '02-02' => '二月二', '05-05' => '端午節', '07-07' => '七夕節', '08-15' => '中秋節', '09-09' => '重陽節', '12-08' => '臘八節', '12-23' => '小年'); 376 $arr_solar = array('01-01' => '元旦', '02-14' => '情人節', '03-12' => '植樹節', '04-01' => '愚人節', '05-01' => '勞動節', '06-01' => '兒童節', '10-01' => '國慶節', '10-31' => '萬聖節', '12-24' => '平安夜', '12-25' => '聖誕節'); 377 }//須要不一樣節日的,用不一樣的$config,而後配置$arr_lunar和$arr_solar 378 $festivals = array(); 379 list($y, $m, $d) = explode('-', $today); 380 if (!$nl_info) $nl_info = $this->convertSolarToLunar($y, intval($m), intval($d)); 381 if ($nl_info[7] > 0 && $nl_info[7] < $nl_info[4]) $nl_info[4] -= 1; 382 $md_lunar = substr('0' . $nl_info[4], -2) . '-' . substr('0' . $nl_info[5], -2); 383 $md_solar = substr_replace($today, '', 0, 5); 384 isset($arr_lunar[$md_lunar]) ? array_push($festivals, $arr_lunar[$md_lunar]) : ''; 385 isset($arr_solar[$md_solar]) ? array_push($festivals, $arr_solar[$md_solar]) : ''; 386 $glweek = date("w", strtotime($today)); //0-6 387 if ($m == 5 && ($d > 7) && ($d < 15) && ($glweek == 0)) array_push($festivals, "母親節"); 388 if ($m == 6 && ($d > 14) && ($d < 22) && ($glweek == 0)) array_push($festivals, "父親節"); 389 $jieqi = $this->getJieQi($y, $m, $d); 390 if ($jieqi) array_push($festivals, $jieqi); 391 return implode('/', $festivals); 392 } 393 }
調用方法:算法
1 $lunar = new Lunar(); 2 $month = $lunar->convertSolarToLunar(date('Y'),date('m'),date('d')); 3 print_r($month);
運行結果:編程
Array ( [0] => 2019 [1] => 七月 [2] => 初一 [3] => 己亥 [4] => 7 [5] => 1 [6] => 豬 [7] => 0 )