关于iOS 阴历阳历转化的那些事儿

Apple有时候是非常有人性化的,他们的家的iphone居然还支持藏文,但有时候就是很坑的,简直毫无人了可言。

      Apple有一个 UIDatePicker 这UI控件可以显示中国的农历,但是去获取现实的数据的时候却是给了一个NSDate类型的数据,然后你如果要现实农历还需要去转换,最坑的是UIDatePicker和UIPickerView还不是同一个东西,一个是继承UIControl, 一个是继承 UIView的,UIPickerView可以通过代理去获取,但是UIDatePicker就只能通过addTarget的方式去获取它的value。

这里就需要去转化了。

试过很多的方法,但是网上的很多人说有坑,说的是万年历的数据不全,或者是有错误,我这里尝试了很多套的万年历数据,最后用了一套能解决百分之九十九的日期转换。一般会出错的就是农历闰月的时间转化,闰月转阳历都说少一天,反复切换,日期就不会不断的变早。

我们先去创建一个header文件去放万年历的数据:

//
//  QYJDataConfig.h
//  QYJGYDateCenter
//
//  Created by Isoftstone on 17/2/7.
//  Copyright © 2017年 com.qyji.DateDemo. All rights reserved.
//

#ifndef QYJDataConfig_h
#define QYJDataConfig_h

#include <stdio.h>
#include <string.h>

#import "NSDate+FSExtension.h"

//最小年限
#define MIN_YEAR 1900

//最大年限 取决下面万年历转化的数据
#define MAX_YEAR 2099

#define lunarDays @[@"零",@"初一",@"初二",@"初三",@"初四",@"初五",@"初六",@"初七",@"初八",@"初九",@"初十",@"十一",@"十二",@"十三",@"十四",@"十五", @"十六",@"十七",@"十八",@"十九",@"廿", @"廿一",@"廿二", @"廿三", @"廿四", @"廿五", @"廿六", @"廿七", @"廿八", @"廿九", @"卅"]

#define lunarMonths @[@"零", @"正",@"二", @"三", @"四", @"五", @"六", @"七", @"八", @"九", @"十", @"冬", @"腊"]

static int DAYS_BEFORE_MONTH[13] = {0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334, 365};

unsigned int lunarToSolar[200] = {
    0x84B6BF,/*1900*/
    0x04AE53, 0x0A5748, 0x5526BD, 0x0D2650, 0x0D9544, 0x46AAB9, 0x056A4D, 0x09AD42, 0x24AEB6, 0x04AE4A,/*1901-1910*/
    0x6A4DBE, 0x0A4D52, 0x0D2546, 0x5D52BA, 0x0B544E, 0x0D6A43, 0x296D37, 0x095B4B, 0x749BC1, 0x049754,/*1911-1920*/
    0x0A4B48, 0x5B25BC, 0x06A550, 0x06D445, 0x4ADAB8, 0x02B64D, 0x095742, 0x2497B7, 0x04974A, 0x664B3E,/*1921-1930*/
    0x0D4A51, 0x0EA546, 0x56D4BA, 0x05AD4E, 0x02B644, 0x393738, 0x092E4B, 0x7C96BF, 0x0C9553, 0x0D4A48,/*1931-1940*/
    0x6DA53B, 0x0B554F, 0x056A45, 0x4AADB9, 0x025D4D, 0x092D42, 0x2C95B6, 0x0A954A, 0x7B4ABD, 0x06CA51,/*1941-1950*/
    0x0B5546, 0x555ABB, 0x04DA4E, 0x0A5B43, 0x352BB8, 0x052B4C, 0x8A953F, 0x0E9552, 0x06AA48, 0x6AD53C,/*1951-1960*/
    0x0AB54F, 0x04B645, 0x4A5739, 0x0A574D, 0x052642, 0x3E9335, 0x0D9549, 0x75AABE, 0x056A51, 0x096D46,/*1961-1970*/
    0x54AEBB, 0x04AD4F, 0x0A4D43, 0x4D26B7, 0x0D254B, 0x8D52BF, 0x0B5452, 0x0B6A47, 0x696D3C, 0x095B50,/*1971-1980*/
    0x049B45, 0x4A4BB9, 0x0A4B4D, 0xAB25C2, 0x06A554, 0x06D449, 0x6ADA3D, 0x0AB651, 0x095746, 0x5497BB,/*1981-1990*/
    0x04974F, 0x064B44, 0x36A537, 0x0EA54A, 0x86B2BF, 0x05AC53, 0x0AB647, 0x5936BC, 0x092E50, 0x0C9645,/*1991-2000*/
    0x4D4AB8, 0x0D4A4C, 0x0DA541, 0x25AAB6, 0x056A49, 0x7AADBD, 0x025D52, 0x092D47, 0x5C95BA, 0x0A954E,/*2001-2010*/
    0x0B4A43, 0x4B5537, 0x0AD54A, 0x955ABF, 0x04BA53, 0x0A5B48, 0x652BBC, 0x052B50, 0x0A9345, 0x474AB9,/*2011-2020*/
    0x06AA4C, 0x0AD541, 0x24DAB6, 0x04B64A, 0x6a573D, 0x0A4E51, 0x0D2646, 0x5E933A, 0x0D534D, 0x05AA43,/*2021-2030*/
    0x36B537, 0x096D4B, 0xB4AEBF, 0x04AD53, 0x0A4D48, 0x6D25BC, 0x0D254F, 0x0D5244, 0x5DAA38, 0x0B5A4C,/*2031-2040*/
    0x056D41, 0x24ADB6, 0x049B4A, 0x7A4BBE, 0x0A4B51, 0x0AA546, 0x5B52BA, 0x06D24E, 0x0ADA42, 0x355B37,/*2041-2050*/
    0x09374B, 0x8497C1, 0x049753, 0x064B48, 0x66A53C, 0x0EA54F, 0x06AA44, 0x4AB638, 0x0AAE4C, 0x092E42,/*2051-2060*/
    0x3C9735, 0x0C9649, 0x7D4ABD, 0x0D4A51, 0x0DA545, 0x55AABA, 0x056A4E, 0x0A6D43, 0x452EB7, 0x052D4B,/*2061-2070*/
    0x8A95BF, 0x0A9553, 0x0B4A47, 0x6B553B, 0x0AD54F, 0x055A45, 0x4A5D38, 0x0A5B4C, 0x052B42, 0x3A93B6,/*2071-2080*/
    0x069349, 0x7729BD, 0x06AA51, 0x0AD546, 0x54DABA, 0x04B64E, 0x0A5743, 0x452738, 0x0D264A, 0x8E933E,/*2081-2090*/
    0x0D5252, 0x0DAA47, 0x66B53B, 0x056D4F, 0x04AE45, 0x4A4EB9, 0x0A4D4C, 0x0D1541, 0x2D92B5          /*2091-2099*/
};


unsigned int solarTolunar[] = {
    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,0x0a5b0,0x14573,0x052b0,0x0a9a8,0x0e950,0x06aa0,
    0x0aea6,0x0ab50,0x04b60,0x0aae4,0x0a570,0x05260,0x0f263,0x0d950,0x05b57,0x056a0,
    0x096d0,0x04dd5,0x04ad0,0x0a4d0,0x0d4d4,0x0d250,0x0d558,0x0b540,0x0b6a0,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,
    
    0x04b63,0x0937f,0x049f8,0x04970,0x064b0,0x068a6,0x0ea5f,0x06b20,0x0a6c4,0x0aaef,
    0x092e0,0x0d2e3,0x0c960,0x0d557,0x0d4a0,0x0da50,0x05d55,0x056a0,0x0a6d0,0x055d4,
    0x052d0,0x0a9b8,0x0a950,0x0b4a0,0x0b6a6,0x0ad50,0x055a0,0x0aba4,0x0a5b0,0x052b0,
    0x0b273,0x06930,0x07337,0x06aa0,0x0ad50,0x04b55,0x04b6f,0x0a570,0x054e4,0x0d260,
    0x0e968,0x0d520,0x0daa0,0x06aa6,0x056df,0x04ae0,0x0a9d4,0x0a4d0,0x0d150,0x0f252,
    0x0d520};

#endif /* QYJDataConfig_h */
这里使用两套的数据源,分开来使用。

转换的代码:

阳历转阴历(公转农历)

#pragma mark - lunar To solar
// 传入农历转换成新历

+ (NSString *)toSolar:(id)lunar {
    NSString *lunarStr;
    if ([lunar isKindOfClass:[NSDate class]]) {
        NSDateFormatter *formatter = [[NSDateFormatter alloc] init];
        formatter.dateFormat = @"yyyy-MM-dd";
        lunarStr = [formatter stringFromDate:lunar];
        return lunarStr;
    } else {
        lunarStr = lunar;
    }
    
    NSCharacterSet *set = [NSCharacterSet characterSetWithCharactersInString:@"年月"];
    NSArray *year_month_day = [lunarStr componentsSeparatedByCharactersInSet:set];
    int year = [year_month_day.firstObject intValue];
    NSString *monthStr = year_month_day[1];
    int month = 0;
    BOOL reserved = NO;
    
    if ([monthStr rangeOfString:@"闰"].length > 0) {
        //闰年
        monthStr = [monthStr substringFromIndex:1];
        month = (int)[lunarMonths indexOfObject:monthStr];
        reserved = YES;
    } else {
        //平年
        month = (int)[lunarMonths indexOfObject:monthStr];
        reserved = NO;
    }
    
    int day = (int)[lunarDays indexOfObject:year_month_day[2]];
    return [LunarToSolar lunarToSolarYear:year month:month monthDay:day leapMonth:reserved];
}

+ (NSString *)lunarToSolarYear:(int)year month:(int)month monthDay:(int)monthDay leapMonth:(BOOL)isLeapMonth {
    int dayOffset;
    int leapMonth;
    int i;
    
    if (year < MIN_YEAR || year > MAX_YEAR || month < 1 || month > 12
        || monthDay < 1 || monthDay > 30) {
        return @"";
    }
    
    dayOffset = (lunarToSolar[year - MIN_YEAR] & 0x001F) - 1;
    
    if (((lunarToSolar[year - MIN_YEAR] & 0x0060) >> 5) == 2)
        dayOffset += 31;
    
    for (i = 1; i < month; i++) {
        if ((lunarToSolar[year - MIN_YEAR] & (0x80000 >> (i - 1))) == 0)
            dayOffset += 29;
        else
            dayOffset += 30;
    }
    
    dayOffset += monthDay;
    leapMonth = (lunarToSolar[year - MIN_YEAR] & 0xf00000) >> 20;
    
    // 这一年有闰月
    if (leapMonth != 0) {
        if (month > leapMonth || (month == leapMonth && isLeapMonth)) {
            if ((lunarToSolar[year - MIN_YEAR] & (0x80000 >> (month - 1))) == 0)
                dayOffset += 29;
            else
                dayOffset += 30;
        }
    }
    
    if (dayOffset > 366 || (year % 4 != 0 && dayOffset > 365)) {
        year += 1;
        if (year % 4 == 1)
            dayOffset -= 366;
        else
            dayOffset -= 365;
    }
    
    int solarInfo[3];
    for (i = 1; i < 13; i++) {
        int iPos = DAYS_BEFORE_MONTH[i];
        if (year % 4 == 0 && i > 2) {
            iPos += 1;
        }
        
        if (year % 4 == 0 && i == 2 && iPos + 1 == dayOffset) {
            solarInfo[1] = i;
            solarInfo[2] = dayOffset - 31;
            break;
        }
        
        if (iPos >= dayOffset) {
            solarInfo[1] = i;
            iPos = DAYS_BEFORE_MONTH[i - 1];
            if (year % 4 == 0 && i > 2) {
                iPos += 1;
            }
            if (dayOffset > iPos)
                solarInfo[2] = dayOffset - iPos;
            else if (dayOffset == iPos) {
                if (year % 4 == 0 && i == 2)
                    solarInfo[2] = DAYS_BEFORE_MONTH[i] - DAYS_BEFORE_MONTH[i - 1] + 1;
                else
                    solarInfo[2] = DAYS_BEFORE_MONTH[i] - DAYS_BEFORE_MONTH[i - 1];
                
            } else
                solarInfo[2] = dayOffset;
            break;
        }
    }
    solarInfo[0] = year;
    int resultYear = year;
    int resultMonth = solarInfo[1];
    int resultDay = solarInfo[2];
    NSString *result = [NSString stringWithFormat:@"%d-%02d-%02d", resultYear, resultMonth, resultDay];
    return result;
}
阴历转阳历(农历转公历)

#pragma mark - solar To lunar

// 传入新历转换成农历
+ (NSString *)toLunar:(id)solar {
    NSDate *solarDate = nil;
    if ([solar isKindOfClass:[NSDate class]]) {
        solarDate = solar;
    } else {
        NSDateFormatter *formatter = [[NSDateFormatter alloc] init];
        formatter.dateFormat = @"yyyy-MM-dd";
        solarDate = [formatter dateFromString:solar];
    }
    NSInteger year = solarDate.fs_year;
    NSInteger month = solarDate.fs_month;
    NSInteger day = solarDate.fs_day;
    
    return [self solarToLunarCalendarThisDate:solarDate year:year month:month day:day];
}

+ (NSString *)solarToLunarCalendarThisDate:(NSDate *)thisdate year:(int)lunarYear month:(int)lunarMonth day:(int)lunarDay {
    NSString *start = @"1900-01-31";
    NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init];
    [dateFormatter setDateFormat:@"yyyy-MM-dd"];
    NSString *end = [dateFormatter stringFromDate:thisdate];
    
    NSDateFormatter *f = [[NSDateFormatter alloc] init];
    [f setDateFormat:@"yyyy-MM-dd"];
    NSDate *startDate = [f dateFromString:start];
    NSDate *endDate = [f dateFromString:end];
    
    NSCalendar *gregorianCalendar = [[NSCalendar alloc] initWithCalendarIdentifier:NSGregorianCalendar];
    
    NSDateComponents *components = [gregorianCalendar components:NSDayCalendarUnit fromDate:startDate toDate:endDate options:0];
    
    int dayCyclical=(int)(([components day] + 30)/(86400/(3600*24)))+10;
    
    int sumdays = (int)[components day];
    
    int tempdays = 0;
    
    //计算农历年
    for (lunarYear = 1900; lunarYear < 2100 && sumdays > 0; lunarYear++) {
        tempdays = [self LunarYearDays:lunarYear];
        sumdays -= tempdays;
    }
    
    if (sumdays < 0) {
        sumdays += tempdays;
        lunarYear--;
    }
    
    //计算闰月
    int doubleMonth = [self DoubleMonth:lunarYear];
    BOOL isLeap = NO;
    
    //计算农历月
    for (lunarMonth = 1; lunarMonth < 13 && sumdays > 0; lunarMonth++) {
        //闰月
        if (doubleMonth > 0 && lunarMonth == (doubleMonth + 1) && isLeap == false) {
            --lunarMonth;
            isLeap = true;
            tempdays = [self DoubleMonthDays:lunarYear];
        } else {
            tempdays = [self MonthDays:lunarYear:lunarMonth];
        }
        
        //解除闰月
        if (isLeap == true && lunarMonth == (doubleMonth + 1)) {
            isLeap = false;
        }
        sumdays -= tempdays;
    }
    
    //计算农历日
    if (sumdays == 0 && doubleMonth > 0 && lunarMonth == doubleMonth + 1) {
        if (isLeap) {
            isLeap = false;
        } else {
            isLeap = true;
            --lunarMonth;
        }
    }
    
    if (sumdays < 0) {
        sumdays += tempdays;
        --lunarMonth;
    }
    
    lunarDay = sumdays + 1;
    
    NSString *string = [NSString stringWithFormat:@"%d年", lunarYear];
    if (isLeap) {
        string = [string stringByAppendingString:[NSString stringWithFormat:@"闰%@月", lunarMonths[lunarMonth]]];
    } else {
        string = [string stringByAppendingString:[NSString stringWithFormat:@"%@月", lunarMonths[lunarMonth]]];
    }
    
    string = [string stringByAppendingString:[NSString stringWithFormat:@"%@", lunarDays[lunarDay]]];
    return string;
}

+ (int)LunarYearDays:(int)y {
    int i, sum = 348;
    for (i = 0x8000; i > 0x8; i >>= 1)
    {
        if ((solarTolunar[y - 1900] & i) != 0)
            sum += 1;
    }
    return (sum + [self DoubleMonthDays:y]);
}

+ (int)DoubleMonth:(int)y {
    return (solarTolunar[y - 1900] & 0xf);
}

///返回农历年闰月的天数
+ (int)DoubleMonthDays:(int)y {
    if ([self DoubleMonth:y] != 0)
        return (((solarTolunar[y - 1900] & 0x10000) != 0) ? 30 : 29);
    else
        return (0);
}

///返回农历年月份的总天数
+ (int)MonthDays:(int)y :(int)m {
    return (((solarTolunar[y - 1900] & (0x10000 >> m)) != 0) ? 30 : 29);
}

/*!

 @method toLunar:

 @abstract 公历转农历

 @discussion 公历转农历

 @param solar id

 @result NSString

 */

+ (NSString *)toLunar:(id)solar;


/*!

 @method toSolar:

 @abstract 农历转公历

 @discussion 农历转公历

 @param lunar id

 @result NSString

 */

+ (NSString *)toSolar:(id)lunar;

外部调用下这两个方法。


评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值