正則表達(dá)式读虏,又稱(chēng)規(guī)則表達(dá)式嚼鹉。(英語(yǔ):Regular Expression,在代碼中常簡(jiǎn)寫(xiě)為regex憨奸、regexp或RE)革屠,計(jì)算機(jī)科學(xué)的一個(gè)概念。正則表達(dá)式通常被用來(lái)檢索排宰、替換那些符合某個(gè)模式(規(guī)則)的文本似芝。具體內(nèi)容可以查看這篇文章。
rangeOfString
示例代碼:
NSString *phoneNo = @"霸王別姬板甘,霸王吃雞党瓮,霸王買(mǎi)雞";
NSString *substring = @"霸";
NSRange range = [phoneNo rangeOfString:substring options:NSRegularExpressionSearch];
if (range.location != NSNotFound) NSLog(@"%@", [phoneNo substringWithRange:range]);
輸出結(jié)果:霸
rangeOfString:(NSString *)searchString options:(NSStringCompareOptions)mask本質(zhì)上是利用正則表達(dá)式匹配,返回的是第一次匹配到的結(jié)果盐类。
正則表達(dá)式NSRegularExpression類(lèi)
方法聲明
NSRegularExpression類(lèi)提供了類(lèi)方法和實(shí)例方法進(jìn)行聲明
/**
* pattern是正則匹配模式
* options為NSRegularExpressionOptions枚舉類(lèi)型
* error匹配條件選項(xiàng)和錯(cuò)誤地址
*/
+ (nullable NSRegularExpression *)regularExpressionWithPattern:(NSString *)pattern options:(NSRegularExpressionOptions)options error:(NSError **)error;
- (nullable instancetype)initWithPattern:(NSString *)pattern options:(NSRegularExpressionOptions)options error:(NSError **)error NS_DESIGNATED_INITIALIZER;
NSRegularExpressionOptions
typedef NS_OPTIONS(NSUInteger, NSRegularExpressionOptions) {
NSRegularExpressionCaseInsensitive = 1 << 0, //不區(qū)分字母大小寫(xiě)的模式寞奸,aBc 會(huì)匹配到abc.
NSRegularExpressionAllowCommentsAndWhitespace = 1 << 1, //忽略掉正則表達(dá)式中的空格和#號(hào)之后的字符,表達(dá)式[a-z]在跳,會(huì)匹配到[a-z]
NSRegularExpressionIgnoreMetacharacters = 1 << 2, //將正則表達(dá)式整體作為字符串處理枪萄。表達(dá)式 a b c 會(huì)匹配到abc,ab#c會(huì)匹配到ab硬毕。
NSRegularExpressionDotMatchesLineSeparators = 1 << 3, //允許.匹配任何字符呻引,包括換行符
NSRegularExpressionAnchorsMatchLines = 1 << 4, //允許^和$符號(hào)匹配行的開(kāi)頭和結(jié)尾
NSRegularExpressionUseUnixLineSeparators = 1 << 5, //設(shè)置\n為唯一的行分隔符,否則所有的都有效吐咳。
NSRegularExpressionUseUnicodeWordBoundaries = 1 << 6 //使用Unicode TR#29標(biāo)準(zhǔn)作為詞的邊界逻悠,否則所有傳統(tǒng)正則表達(dá)式的詞邊界都有效
};
NSMatchingOptions
typedef NS_OPTIONS(NSUInteger, NSMatchingOptions) {
NSMatchingReportProgress = 1 << 0, //找到最長(zhǎng)的匹配字符串后調(diào)用block回調(diào)
NSMatchingReportCompletion = 1 << 1, //找到任何一個(gè)匹配串后都回調(diào)一次block
NSMatchingAnchored = 1 << 2, //從匹配范圍的開(kāi)始出進(jìn)行極限匹配
NSMatchingWithTransparentBounds = 1 << 3, //允許匹配的范圍超出設(shè)置的范圍
NSMatchingWithoutAnchoringBounds = 1 << 4 //禁止^和$自動(dòng)匹配行還是和結(jié)束
};
NSMatchingFlags
typedef NS_OPTIONS(NSUInteger, NSMatchingFlags) {
NSMatchingProgress = 1 << 0, //匹配到最長(zhǎng)串是被設(shè)置
NSMatchingCompleted = 1 << 1, //全部分配完成后被設(shè)置
NSMatchingHitEnd = 1 << 2, //匹配到設(shè)置范圍的末尾時(shí)被設(shè)置
NSMatchingRequiredEnd = 1 << 3, //當(dāng)前匹配到的字符串在匹配范圍的末尾時(shí)被設(shè)置
NSMatchingInternalError= 1 << 4 //由于錯(cuò)誤導(dǎo)致的匹配失敗時(shí)被設(shè)置 };
NSRegularExpression常用方法:
@interface NSRegularExpression (NSMatching)
/*
遍歷的模式元践,正則表達(dá)式匹配在指定options和range模式下匹配指定string,傳入block中可以獲取結(jié)果信息
*/
- (void)enumerateMatchesInString:(NSString *)string options:(NSMatchingOptions)options range:(NSRange)range usingBlock:(void (NS_NOESCAPE ^)(NSTextCheckingResult * _Nullable result, NSMatchingFlags flags, BOOL *stop))block;
/*
在指定options和range模式下匹配指定string童谒,通過(guò)正則匹配返回一個(gè)匹配結(jié)果的數(shù)組
*/
- (NSArray<NSTextCheckingResult *> *)matchesInString:(NSString *)string options:(NSMatchingOptions)options range:(NSRange)range;
/*
返回滿(mǎn)足條件的匹配次數(shù)
*/
- (NSUInteger)numberOfMatchesInString:(NSString *)string options:(NSMatchingOptions)options range:(NSRange)range;
/*
匹配返回的第一個(gè)結(jié)果单旁,NSTextCheckingResult類(lèi)型
*/
- (nullable NSTextCheckingResult *)firstMatchInString:(NSString *)string options:(NSMatchingOptions)options range:(NSRange)range;
/*
匹配返回的第一個(gè)結(jié)果的NSRange范圍信息
*/
- (NSRange)rangeOfFirstMatchInString:(NSString *)string options:(NSMatchingOptions)options range:(NSRange)range;
@end
示例代碼:
+(NSMutableAttributedString *)getTagStringFromString:(NSString *)string{
NSString *pattern = @"(\\[\\[[^\\]]*\\]\\])";
NSRegularExpression * regex = [[NSRegularExpression alloc]initWithPattern:pattern options:0 error:nil];
__block int count = 0;
NSString *newString = [string stringByReplacingOccurrencesOfString:@"]]" withString:@""];
NSString *otherString = [newString stringByReplacingOccurrencesOfString:@"[[" withString:@""];
NSMutableAttributedString * attributedString = [[NSMutableAttributedString alloc] initWithString:otherString];
[regex enumerateMatchesInString:string options:0 range:NSMakeRange(0, string.length) usingBlock:^(NSTextCheckingResult * _Nullable result, NSMatchingFlags flags, BOOL * _Nonnull stop) {
if (flags != NSMatchingInternalError) {
[attributedString addAttribute:NSForegroundColorAttributeName value:HexColor(@"#FF2525") range:NSMakeRange(result.range.location-4*count, result.range.length-4)];
count++;
}
}];
return attributedString;
}
效果展示:利用NSRegularExpression類(lèi)進(jìn)行匹配,可以進(jìn)行多次匹配饥伊,并以數(shù)組的形式返回給我們象浑,如果需要進(jìn)行多次的結(jié)果匹配,這個(gè)方法效率相對(duì)比較高琅豆。
NSPredicate(謂詞)匹配
常用關(guān)鍵詞:
邏輯運(yùn)算符號(hào)運(yùn)用:
邏輯運(yùn)算符號(hào)(> , < , = , >= , <=)還可以跟邏輯運(yùn)算符一起使用的: &&, || ,AND, OR 謂詞不區(qū)分大小寫(xiě)
NSArray *datas = @[[PeoplesModel itemWithName:@"小李子" age:20 height:170],
[PeoplesModel itemWithName:@"小棒槌" age:22 height:165],
[PeoplesModel itemWithName:@"小凳子" age:18 height:180],
[PeoplesModel itemWithName:@"小桌子" age:23 height:178]];
// NSPredicate *predicate = [NSPredicate predicateWithFormat:@"age >= 20"]; //創(chuàng)建NSPredicate對(duì)象 查找年紀(jì)大于二十的對(duì)象
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"height > 175 && age > 20"]; //查找身高高于175且年紀(jì)大于二十的
NSArray *newDatas = [[datas filteredArrayUsingPredicate:predicate] mutableCopy]; //使用數(shù)組的filteredArrayUsingPredicate方法 獲取符合我們指定條件的對(duì)象
[newDatas enumerateObjectsUsingBlock:^(PeoplesModel *model, NSUInteger idx, BOOL * _Nonnull stop) {
NSLog(@"%@\t%ld",model.name,(long)model.age);
}];
輸出結(jié)果為:小桌子**** 23
范圍運(yùn)算符(IN,BETWEEN):
NSArray *datas = @[[PeoplesModel itemWithName:@"xiaolizi" age:20 height:170],
[PeoplesModel itemWithName:@"xiaobangchui" age:22 height:165],
[PeoplesModel itemWithName:@"dongfangbubai" age:18 height:180],
[PeoplesModel itemWithName:@"qiaofeng" age:23 height:178]];
// NSPredicate *predicate = [NSPredicate predicateWithFormat:@"age BETWEEN {20,30}"];
// 相當(dāng)于30>=age>=20
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"name IN {'xiaolizi', '20'}"];//查找出名字叫小李子且20的人
NSArray *newDatas = [[datas filteredArrayUsingPredicate:predicate] mutableCopy]; //使用數(shù)組的filteredArrayUsingPredicate方法 獲取符合我們指定條件的對(duì)象
[newDatas enumerateObjectsUsingBlock:^(PeoplesModel *model, NSUInteger idx, BOOL * _Nonnull stop) {
NSLog(@"%@\t%ld",model.name,(long)model.age);
}];
輸出結(jié)果為:xiaolizi 20
其它關(guān)鍵詞:
BEGINSWITH:以**開(kāi)頭
ENDSWITH:以**結(jié)尾
CONTAINS:是否包含某個(gè)字符串
LIKE:模糊查詢(xún)
這些關(guān)鍵詞后面還可以跟上一些格式符號(hào) 如:BEGINSWITH[cd] c表示不區(qū)分大小寫(xiě) d表示不區(qū)分發(fā)音 符號(hào) cd就可以表示即不區(qū)分大小寫(xiě)也不區(qū)分發(fā)音符號(hào)
NSArray *datas = @[[PeoplesModel itemWithName:@"xiaolizi" age:20 height:170],
[PeoplesModel itemWithName:@"xiaobangchui" age:22 height:165],
[PeoplesModel itemWithName:@"dongfangbubai" age:18 height:180],
[PeoplesModel itemWithName:@"qiaofeng" age:23 height:178]];
// NSPredicate *predicate = [NSPredicate predicateWithFormat:@"name CONTAINS 'x'"];
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"name LIKE '*xiao*'"];
// NSPredicate *predicate = [NSPredicate predicateWithFormat:@"name LIKE[cd] '???er*"]; //LIKE中的?表示一個(gè)任意字符, *表示通配符
NSArray *newDatas = [[datas filteredArrayUsingPredicate:predicate] mutableCopy]; //使用數(shù)組的filteredArrayUsingPredicate方法 獲取符合我們指定條件的對(duì)象
[newDatas enumerateObjectsUsingBlock:^(PeoplesModel *model, NSUInteger idx, BOOL * _Nonnull stop) {
NSLog(@"%@\t%ld",model.name,(long)model.age);
}];
輸出結(jié)果為:xiaolizi 20 xiaobangchui 22
字符串匹配查詢(xún)(SELF) 數(shù)組中都是字符串,即非屬性匹配查詢(xún), 需要用到SELF
正則表達(dá)式(MATCHES) NSPredicate 使用MATCHES 匹配正則表達(dá)式愉豺,正則表達(dá)式的寫(xiě)法采用international components for Unicode (ICU)的正則語(yǔ)法
/** 郵箱校驗(yàn) */
+ (BOOL)checkEmail:(NSString *)email{
NSString *regex = @"^[a-zA-Z0-9_-]+@[a-zA-Z0-9_-]+(\\.[a-zA-Z0-9_-]+)+$";
NSPredicate *emailTest = [NSPredicate predicateWithFormat:@"SELF MATCHES %@", regex];
return [emailTest evaluateWithObject:email];
}
常用的正則表達(dá)式:
一、校驗(yàn)數(shù)字的表達(dá)式
1 數(shù)字:^[0-9]*$
2 n位的數(shù)字:^\d{n}$
3 至少n位的數(shù)字:^\d{n,}$
4 m-n位的數(shù)字:^\d{m,n}$
5 零和非零開(kāi)頭的數(shù)字:^(0|[1-9][0-9]*)$
6 非零開(kāi)頭的最多帶兩位小數(shù)的數(shù)字:^([1-9][0-9]*)+(.[0-9]{1,2})?$
7 帶1-2位小數(shù)的正數(shù)或負(fù)數(shù):^(-)?\d+(.\d{1,2})?$
8 正數(shù)茫因、負(fù)數(shù)蚪拦、和小數(shù):^(-|+)?\d+(.\d+)?$
9 有兩位小數(shù)的正實(shí)數(shù):^[0-9]+(.[0-9]{2})?$
10 有1~3位小數(shù)的正實(shí)數(shù):^[0-9]+(.[0-9]{1,3})?$
11 非零的正整數(shù):^[1-9]\d 或 ^+?[1-9][0-9]$
12 非零的負(fù)整數(shù):^-[1-9][]0-9"*
13 非負(fù)整數(shù):^\d+
14 非正整數(shù):^-[1-9]\d*|0
15 非負(fù)浮點(diǎn)數(shù):^\d+(.\d+)?
16 非正浮點(diǎn)數(shù):^((-\d+(.\d+)?)|(0+(.0+)?))
17 正浮點(diǎn)數(shù):^[1-9]\d.\d|0.\d[1-9]\d
18 負(fù)浮點(diǎn)數(shù):^-([1-9]\d.\d|0.\d[1-9]\d)
19 浮點(diǎn)數(shù):^(-?\d+)(.\d+)?
二、校驗(yàn)字符的表達(dá)式
1 漢字:^[\u4e00-\u9fa5]{0,}$
2 英文和數(shù)字:^[A-Za-z0-9]+
3 長(zhǎng)度為3-20的所有字符:^.{3,20}$
4 由26個(gè)英文字母組成的字符串:^[A-Za-z]+$
5 由26個(gè)大寫(xiě)英文字母組成的字符串:^[A-Z]+$
6 由26個(gè)小寫(xiě)英文字母組成的字符串:^[a-z]+$
7 由數(shù)字和26個(gè)英文字母組成的字符串:^[A-Za-z0-9]+$
8 由數(shù)字冻押、26個(gè)英文字母或者下劃線(xiàn)組成的字符串:^\w+
9 中文驰贷、英文、數(shù)字包括下劃線(xiàn):^[\u4E00-\u9FA5A-Za-z0-9_]+$
10 中文洛巢、英文括袒、數(shù)字但不包括下劃線(xiàn)等符號(hào):^[\u4E00-\u9FA5A-Za-z0-9]+
11 可以輸入含有^%&',;=?\x22]+
12 禁止輸入含有的字符:[^\x22]+
三、 特殊需求表達(dá)式
1 Email地址:^\w+([-+.]\w+)@\w+([-.]\w+).\w+([-.]\w+)*$
2 域名:[a-zA-Z0-9][-a-zA-Z0-9]{0,62}(/.[a-zA-Z0-9][-a-zA-Z0-9]{0,62})+/.?
3 InternetURL:[a-zA-z]+://[^\s]* 或 ^http://([\w-]+.)+[\w-]+(/[\w-./?%&=]*)?$
4 手機(jī)號(hào)碼:^(13[0-9]|14[5|7]|15[0|1|2|3|5|6|7|8|9]|18[0|1|2|3|5|6|7|8|9])\d{8}$
5 電話(huà)號(hào)碼("XXX-XXXXXXX"稿茉、"XXXX-XXXXXXXX"锹锰、"XXX-XXXXXXX"、"XXX-XXXXXXXX"漓库、"XXXXXXX"和"XXXXXXXX):^((\d{3,4}-)|\d{3.4}-)?\d{7,8}$
6 國(guó)內(nèi)電話(huà)號(hào)碼(0511-4405222城须、021-87888822):\d{3}-\d{8}|\d{4}-\d{7}
7 身份證號(hào)(15位、18位數(shù)字):^\d{15}|\d{18}$
8 短身份證號(hào)碼(數(shù)字米苹、字母x結(jié)尾):^([0-9]){7,18}(x|X)?
9 帳號(hào)是否合法(字母開(kāi)頭糕伐,允許5-16字節(jié),允許字母數(shù)字下劃線(xiàn)):^[a-zA-Z][a-zA-Z0-9_]{4,15}$
10 密碼(以字母開(kāi)頭蘸嘶,長(zhǎng)度在6~18之間良瞧,只能包含字母、數(shù)字和下劃線(xiàn)):^[a-zA-Z]\w{5,17}$
11 強(qiáng)密碼(必須包含大小寫(xiě)字母和數(shù)字的組合训唱,不能使用特殊字符褥蚯,長(zhǎng)度在8-10之間):^(?=.\d)(?=.[a-z])(?=.*[A-Z]).{8,10}$
12 日期格式:^\d{4}-\d{1,2}-\d{1,2}
13 一年的12個(gè)月(01~09和1~12):^(0?[1-9]|1[0-2])$
14 一個(gè)月的31天(01~09和1~31):^((0?[1-9])|((1|2)[0-9])|30|31)$
15 錢(qián)的輸入格式:
16 1.有四種錢(qián)的表示形式我們可以接受:"10000.00" 和 "10,000.00", 和沒(méi)有 "分" 的 "10000" 和 "10,000":^[1-9][0-9]*$
17 2.這表示任意一個(gè)不以0開(kāi)頭的數(shù)字,但是,這也意味著一個(gè)字符"0"不通過(guò),所以我們采用下面的形式:^(0|[1-9][0-9]*)$
18 3.一個(gè)0或者一個(gè)不以0開(kāi)頭的數(shù)字.我們還可以允許開(kāi)頭有一個(gè)負(fù)號(hào):^(0|-?[1-9][0-9]*)$
19 4.這表示一個(gè)0或者一個(gè)可能為負(fù)的開(kāi)頭不為0的數(shù)字.讓用戶(hù)以0開(kāi)頭好了.把負(fù)號(hào)的也去掉,因?yàn)殄X(qián)總不能是負(fù)的吧.下面我們要加的是說(shuō)明可能的小數(shù)部分:^[0-9]+(.[0-9]+)?$
20 5.必須說(shuō)明的是,小數(shù)點(diǎn)后面至少應(yīng)該有1位數(shù),所以"10."是不通過(guò)的,但是 "10" 和 "10.2" 是通過(guò)的:^[0-9]+(.[0-9]{2})?$
21 6.這樣我們規(guī)定小數(shù)點(diǎn)后面必須有兩位,如果你認(rèn)為太苛刻了,可以這樣:^[0-9]+(.[0-9]{1,2})?$
22 7.這樣就允許用戶(hù)只寫(xiě)一位小數(shù).下面我們?cè)摽紤]數(shù)字中的逗號(hào)了,我們可以這樣:^[0-9]{1,3}(,[0-9]{3})*(.[0-9]{1,2})?$
23 8.1到3個(gè)數(shù)字,后面跟著任意個(gè) 逗號(hào)+3個(gè)數(shù)字,逗號(hào)成為可選,而不是必須:^([0-9]+|[0-9]{1,3}(,[0-9]{3})*)(.[0-9]{1,2})?$
24 備注:這就是最終結(jié)果了,別忘了"+"可以用"*"替代如果你覺(jué)得空字符串也可以接受的話(huà)(奇怪,為什么?)最后,別忘了在用函數(shù)時(shí)去掉去掉那個(gè)反斜杠,一般的錯(cuò)誤都在這里
25 xml文件:^([a-zA-Z]+-?)+[a-zA-Z0-9]+\.[x|X][m|M][l|L]$
26 中文字符的正則表達(dá)式:[\u4e00-\u9fa5]
27 雙字節(jié)字符:[^\x00-\xff] (包括漢字在內(nèi),可以用來(lái)計(jì)算字符串的長(zhǎng)度(一個(gè)雙字節(jié)字符長(zhǎng)度計(jì)2况增,ASCII字符計(jì)1))
28 空白行的正則表達(dá)式:\n\s*\r (可以用來(lái)刪除空白行)
29 HTML標(biāo)記的正則表達(dá)式:<(\S?)[^>]>.?</\1>|<.? /> (網(wǎng)上流傳的版本太糟糕赞庶,上面這個(gè)也僅僅能部分,對(duì)于復(fù)雜的嵌套標(biāo)記依舊無(wú)能為力)
30 首尾空白字符的正則表達(dá)式:^\s|\s) (可以用來(lái)刪除行首行尾的空白字符(包括空格、制表符歧强、換頁(yè)符等等)澜薄,非常有用的表達(dá)式)
31 騰訊QQ號(hào):[1-9][0-9]{4,} (騰訊QQ號(hào)從10000開(kāi)始)
32 中國(guó)郵政編碼:[1-9]\d{5}(?!\d) (中國(guó)郵政編碼為6位數(shù)字)
33 IP地址:\d+.\d+.\d+.\d+ (提取IP地址時(shí)有用)
??這些表達(dá)式可能存在漏洞,請(qǐng)謹(jǐn)慎使用
參考資料:
https://www.cnblogs.com/jijiYY/p/4083914.html