標籤:
iOS開發實用技巧—身份證的正則校正
說明:
- 在ios項目的開發中可能很多地方都需要用到身份證校正,一般在開發的時候很多人都是直接百度去網上蕩相關的Regex和校正代碼,但是網上瘋狂粘貼複製的校正代碼本身也可能並不準確,可能會有風險,比如2013年1月1號起停止使用15位的身份證,網上的身份證校正普遍支援15位的號碼。
- 在開發過程中,進行類似處理的時候,還是需要一些甄別的能力的,當然也要考慮自己的項目的實際情況。該文貼出了最近項目中使用到得身份證校正代碼,以方便有需要的人“謹慎”擷取。
一、規則
下面是iOS身份證校正規則,對於第6點就值得商榷,按道理出生年份前兩位是20也應該是合理的。如果要校正投保人需要年滿18歲,需要另行檢查,不應放在身份證校正裡面。
- 長度必須是18位,前17位必須是數字,第十八位可以是數字或X(校正時不區分大小寫)
- 前兩位必須是以下35種情形中的一種: 11,12,13,14,15,21,22,23,31,32,33,34,35,36,37,41,42,43,44,45,46,50,51,52,53,54,61,62,63,64,65,71,81,82,91
- 第7到第14位出生年月日。第7到第10位為出生年份;11到12位表示月份,範圍為01~12;13到14位為合法的日期,比如月份是04,範圍應是01~30
- 第17位表示性別,必須是0或1,0表示女,1表示男
- 第18位為前17位的校正位 演算法如下:
- 總和 = (n1 + n11) * 7 + (n2 + n12) * 9 + (n3 + n13) * 10 + (n4 + n14) * 5 + (n5 + n15) * 8 + (n6 + n16) * 4 + (n7 + n17) * 2 + n8 + n9 * 6 + n10 * 3,其中n1表示1位元字,其它類似
- 用總和除以11,看餘數是多少, 餘數只可能有0 1 2 3 4 5 6 7 8 9 10這11個數字。其分別對應的最後一位身份證的號碼為1 0 X 9 8 7 6 5 4 3 2
- 第7位必須為1,第8位必須為9,即:出生年份的前兩位必須是19
二、程式碼範例:
對於這種需要在很多地方都用到的代碼,可以使用一個工具類或者是分類來寫。這裡把校正的代碼寫在了一個工具類中。
VerifyRegexTool.h檔案代碼:
1 #import <Foundation/Foundation.h> 2 3 @interface VerifyRegexTool : NSObject 4 5 + (BOOL)verifyIsNotEmpty:(NSString *)str; //驗證是否不為空白 6 7 + (BOOL)verifyText:(NSString *)text withRegex:(NSString *)regex; //正則驗證 8 9 + (BOOL)verifyIDCardNumber:(NSString *)value; //驗證身份證10 11 + (BOOL)verifyCardNumberWithSoldier:(NSString *)value; //驗證軍官證或警官證12 13 + (BOOL)verifyIDCardHadAdult:(NSString *)card; //驗證身份證是否成年且小於100歲****這個方法中不做身份證校正,請確保傳入的是正確身份證14 15 + (BOOL)verifyIDCardMoreThanPointDate:(NSString *)card withNumber:(NSInteger)number withAddTimeInterval:(NSTimeInterval)interval withDateType:(DateType)dateType; //驗證身份證加上指定天數是否大於指定number的類型16 17 + (BOOL)verifyIDCardLessThanPointDate:(NSString *)card withNumber:(NSInteger)number withAddTimeInterval:(NSTimeInterval)interval withDateType:(DateType)dateType; //驗證身份證是否小於指定number的類型18 19 20 + (NSString *)getIDCardBirthday:(NSString *)card; //得到身份證的生日****這個方法中不做身份證校正,請確保傳入的是正確身份證21 + (NSInteger)getIDCardSex:(NSString *)card; //得到身份證的性別(1男0女)****這個方法中不做身份證校正,請確保傳入的是正確身份證22 23 @end
VerifyRegexTool.m檔案代碼:
1 #import "VerifyRegexTool.h" 2 3 @implementation VerifyRegexTool 4 5 //驗證是否不為空白 6 + (BOOL)verifyIsNotEmpty:(NSString *)str 7 { 8 if (!str) return NO; 9 10 str = [str stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceCharacterSet]]; 11 if (![str isEqualToString:@""]) { 12 return YES; 13 } else { 14 return NO; 15 } 16 } 17 18 //正則驗證 19 + (BOOL)verifyText:(NSString *)text withRegex:(NSString *)regex 20 { 21 return [[text stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceCharacterSet]] isMatchedByRegex:regex]; 22 } 23 24 //驗證身份證 25 //必須滿足以下規則 26 //1. 長度必須是18位,前17位必須是數字,第十八位可以是數字或X 27 //2. 前兩位必須是以下情形中的一種:11,12,13,14,15,21,22,23,31,32,33,34,35,36,37,41,42,43,44,45,46,50,51,52,53,54,61,62,63,64,65,71,81,82,91 28 //3. 第7到第14位出生年月日。第7到第10位為出生年份;11到12位表示月份,範圍為01-12;13到14位為合法的日期 29 //4. 第17位表示性別,雙數表示女,單數表示男 30 //5. 第18位為前17位的校正位 31 //演算法如下: 32 //(1)校正和 = (n1 + n11) * 7 + (n2 + n12) * 9 + (n3 + n13) * 10 + (n4 + n14) * 5 + (n5 + n15) * 8 + (n6 + n16) * 4 + (n7 + n17) * 2 + n8 + n9 * 6 + n10 * 3,其中n數值,表示第幾位的數字 33 //(2)餘數 = 校正和 % 11 34 //(3)如果餘數為0,校正位應為1,餘數為1到10校正位應為字串“0X98765432”(不包括分號)的第餘數位的值(比如餘數等於3,校正位應為9) 35 //6. 出生年份的前兩位必須是19或20 36 + (BOOL)verifyIDCardNumber:(NSString *)value 37 { 38 value = [value stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceAndNewlineCharacterSet]]; 39 if ([value length] != 18) { 40 return NO; 41 } 42 NSString *mmdd = @"(((0[13578]|1[02])(0[1-9]|[12][0-9]|3[01]))|((0[469]|11)(0[1-9]|[12][0-9]|30))|(02(0[1-9]|[1][0-9]|2[0-8])))"; 43 NSString *leapMmdd = @"0229"; 44 NSString *year = @"(19|20)[0-9]{2}"; 45 NSString *leapYear = @"(19|20)(0[48]|[2468][048]|[13579][26])"; 46 NSString *yearMmdd = [NSString stringWithFormat:@"%@%@", year, mmdd]; 47 NSString *leapyearMmdd = [NSString stringWithFormat:@"%@%@", leapYear, leapMmdd]; 48 NSString *yyyyMmdd = [NSString stringWithFormat:@"((%@)|(%@)|(%@))", yearMmdd, leapyearMmdd, @"20000229"]; 49 NSString *area = @"(1[1-5]|2[1-3]|3[1-7]|4[1-6]|5[0-4]|6[1-5]|82|[7-9]1)[0-9]{4}"; 50 NSString *regex = [NSString stringWithFormat:@"%@%@%@", area, yyyyMmdd , @"[0-9]{3}[0-9Xx]"]; 51 52 NSPredicate *regexTest = [NSPredicate predicateWithFormat:@"SELF MATCHES %@", regex]; 53 if (![regexTest evaluateWithObject:value]) { 54 return NO; 55 } 56 int summary = ([value substringWithRange:NSMakeRange(0,1)].intValue + [value substringWithRange:NSMakeRange(10,1)].intValue) *7 57 + ([value substringWithRange:NSMakeRange(1,1)].intValue + [value substringWithRange:NSMakeRange(11,1)].intValue) *9 58 + ([value substringWithRange:NSMakeRange(2,1)].intValue + [value substringWithRange:NSMakeRange(12,1)].intValue) *10 59 + ([value substringWithRange:NSMakeRange(3,1)].intValue + [value substringWithRange:NSMakeRange(13,1)].intValue) *5 60 + ([value substringWithRange:NSMakeRange(4,1)].intValue + [value substringWithRange:NSMakeRange(14,1)].intValue) *8 61 + ([value substringWithRange:NSMakeRange(5,1)].intValue + [value substringWithRange:NSMakeRange(15,1)].intValue) *4 62 + ([value substringWithRange:NSMakeRange(6,1)].intValue + [value substringWithRange:NSMakeRange(16,1)].intValue) *2 63 + [value substringWithRange:NSMakeRange(7,1)].intValue *1 + [value substringWithRange:NSMakeRange(8,1)].intValue *6 64 + [value substringWithRange:NSMakeRange(9,1)].intValue *3; 65 NSInteger remainder = summary % 11; 66 NSString *checkBit = @""; 67 NSString *checkString = @"10X98765432"; 68 checkBit = [checkString substringWithRange:NSMakeRange(remainder,1)];// 判斷校正位 69 return [checkBit isEqualToString:[[value substringWithRange:NSMakeRange(17,1)] uppercaseString]]; 70 } 71 72 //驗證軍官證或警官證 73 //必須是下面兩種格式中的一種 74 //格式一:4到20位元字 75 //格式二:大於或等於10位並且小於等於20位(中文按兩位),並滿足以下規則 76 //1)必須有“字第”兩字 77 //2)“字第”前面有至少一個字元 78 //3)“字第”後是4位以上數字 79 + (BOOL)verifyCardNumberWithSoldier:(NSString *)value 80 { 81 NSString *s1 = @"^\\d*$"; 82 NSString *s2 = @"^.{1,}字第\\d{4,}$"; 83 //NSString *s3 = @"^([A-Za-z0-9\\u4e00-\\u9fa5])*$"; 84 value = [value stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceAndNewlineCharacterSet]]; 85 if ([VerifyRegexTool verifyText:value withRegex:s1]) { 86 NSString *s11 = @"^\\d{4,20}$"; 87 return [value isMatchedByRegex:s11]; 88 } else if ([self lengthUsingChineseCharacterCountByTwo:value] >= 10 89 && [self lengthUsingChineseCharacterCountByTwo:value] <= 20) { 90 return [value isMatchedByRegex:s2]; 91 } 92 93 return NO; 94 } 95 96 + (NSUInteger)lengthUsingChineseCharacterCountByTwo:(NSString *)string{ 97 NSUInteger count = 0; 98 for (NSUInteger i = 0; i< string.length; ++i) { 99 if ([string characterAtIndex:i] < 256) {100 count++;101 } else {102 count += 2;103 }104 }105 return count;106 }107 108 //驗證身份證是否成年且小於100歲****這個方法中不做身份證校正,請確保傳入的是正確身份證109 + (BOOL)verifyIDCardHadAdult:(NSString *)card110 {111 NSString *birtday = [VerifyRegexTool getIDCardBirthday:card]; //****年**月**日112 //轉換為****-**-**113 birtday = [birtday stringByReplacingOccurrencesOfString:@"年" withString:@"-"];114 birtday = [birtday stringByReplacingOccurrencesOfString:@"月" withString:@"-"];115 birtday = [birtday stringByReplacingOccurrencesOfString:@"日" withString:@""];116 birtday = [NSString stringWithFormat:@"%@ 00:00",birtday];117 NSDate *dateBirtday = [NSDate dateFromString:birtday];118 NSInteger year = [[NSDate date] getIntervalTime:dateBirtday withDateType:DateTypeYear]; //得到年數119 120 if (year >= 18 && year < 100) {121 return YES;122 } else {123 return NO;124 }125 126 }127 128 //驗證身份證加上指定天數是否大於指定number的類型129 + (BOOL)verifyIDCardMoreThanPointDate:(NSString *)card withNumber:(NSInteger)number withAddTimeInterval:(NSTimeInterval)interval withDateType:(DateType)dateType {130 NSString *birtday = [VerifyRegexTool getIDCardBirthday:card]; //****年**月**日131 //轉換為****-**-**132 birtday = [birtday stringByReplacingOccurrencesOfString:@"年" withString:@"-"];133 birtday = [birtday stringByReplacingOccurrencesOfString:@"月" withString:@"-"];134 birtday = [birtday stringByReplacingOccurrencesOfString:@"日" withString:@""];135 birtday = [NSString stringWithFormat:@"%@ 00:00",birtday];136 NSDate *dateBirtday = [NSDate dateFromString:birtday];137 NSDate *today = [NSDate dateFromStringWithoutTime:[[NSDate date] strDate]];138 NSDate *pointDate = [today dateByAddingTimeInterval:interval];139 140 //tempDate為number後的日期141 NSDate *tempDate = [pointDate dateAddNumber:-number withDateType:dateType]; //臨界日期142 if ([dateBirtday earlierDate:tempDate] == dateBirtday) { //如果生日+指定天數大雨tempdate,則通過143 return YES;144 } else {145 return NO;146 }147 }148 149 //驗證身份證加上指定天數是否小於指定number的類型150 + (BOOL)verifyIDCardLessThanPointDate:(NSString *)card withNumber:(NSInteger)number withAddTimeInterval:(NSTimeInterval)interval withDateType:(DateType)dateType151 {152 NSString *birtday = [VerifyRegexTool getIDCardBirthday:card]; //****年**月**日153 //轉換為****-**-**154 birtday = [birtday stringByReplacingOccurrencesOfString:@"年" withString:@"-"];155 birtday = [birtday stringByReplacingOccurrencesOfString:@"月" withString:@"-"];156 birtday = [birtday stringByReplacingOccurrencesOfString:@"日" withString:@""];157 birtday = [NSString stringWithFormat:@"%@ 00:00",birtday];158 NSDate *dateBirtday = [NSDate dateFromString:birtday];159 NSDate *today = [NSDate dateFromStringWithoutTime:[[NSDate date] strDate]];160 NSDate *pointDate = [today dateByAddingTimeInterval:interval];161 162 //tempDate為number後的日期163 NSDate *tempDate = [pointDate dateAddNumber:-number withDateType:dateType]; //臨界日期164 if ([tempDate earlierDate:dateBirtday] == tempDate) { //如果生日+指定天數小於tempdate,則通過165 return YES;166 } else {167 return NO;168 }169 }170 171 //得到身份證的生日****這個方法中不做身份證校正,請確保傳入的是正確身份證172 + (NSString *)getIDCardBirthday:(NSString *)card {173 card = [card stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceAndNewlineCharacterSet]];174 if ([card length] != 18) {175 return nil;176 }177 NSString *birthady = [NSString stringWithFormat:@"%@年%@月%@日",[card substringWithRange:NSMakeRange(6,4)], [card substringWithRange:NSMakeRange(10,2)], [card substringWithRange:NSMakeRange(12,2)]];178 return birthady;179 }180 181 //得到身份證的性別(1男0女)****這個方法中不做身份證校正,請確保傳入的是正確身份證182 + (NSInteger)getIDCardSex:(NSString *)card {183 card = [card stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceAndNewlineCharacterSet]];184 NSInteger defaultValue = 0;185 if ([card length] != 18) {186 return defaultValue;187 }188 NSInteger number = [[card substringWithRange:NSMakeRange(16,1)] integerValue];189 if (number % 2 == 0) { //偶數為女190 return 0;191 } else {192 return 1;193 }194 }195 196 @end
身份證校正調用樣本:
1 if (![VerifyRegexTool verifyIDCardNumber:self.IdentityCardNOTextfield.text]) { //驗證身份證是否真實2 [self showMessageHUD:@"請輸入正確的社會安全號碼碼!" withTimeInterval:kShowMessageTime];3 return;4 }
iOS開發實用技巧—身份證的正則校正