首页 > 系统 > iOS > 正文

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

2019-11-09 16:43:55
字体:
来源:转载
供稿:网友

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;

外部调用下这两个方法。


发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表