NSScanner:一个陌生的条件判断利器

2023-10-25 04:08

本文主要是介绍NSScanner:一个陌生的条件判断利器,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

NSScanner官方文档

NSScanner类是一个类簇的抽象父类,该类簇为一个从NSString对象扫描值的对象提供了程序接口。

NSScanner对象把NSString 对象的的字符解释和转化成 number和string 类型的值。在创建NSScanner对象的时候为它分配字符(string ),当你从NSScanner对象获取内容的时候,它会从头到尾遍历字符串(string)。

由于类簇的属性, scanner对象并不是 NSScanner类的实例,而是它一个私有子类的实例。尽管scanner对象的类是私有的,但是它的接口是公开的(抽象父类已经声明)。 NSScanner 的原始方法是string和Configuring a Scanner方法下面列举的所有的方法。

在 NSScanner 对象扫描字符串的时候,你可以通过设置属性charactersToBeSkipped忽略某些字符。在扫描字符串之前,那些位于忽略字符集中的字符将会被跳过。默认的忽略字符是空格和回车字符。

可以通过[[scanner string] substringFromIndex:[scanner scanLocation]]获取未扫描的字符串。

创建 Scanner对象

1
2
3
+ (instancetype)scannerWithString:(NSString *)aString
+ (id)localizedScannerWithString:(NSString *)aString
- (instancetype)initWithString:(NSString *)aString
  • scannerWithString,返回值是 扫描过aString字符串的NSScanner 对象,该方法通过调用initWithString设置扫描字符串;

  • localizedScannerWithString,返回值是 通过用户默认的 locale方式扫描字符串的NSScanner 对象,该方法也是通过调用initWithString设置扫描字符串;

  • initWithString,返回值是NSScanner 对象,该对象通过扫描aString完成初始化

获取Scanner的字符串

1
@property(readonly, copy) NSString *string

配置Scanner

1
2
3
4
@property NSUInteger scanLocation
@property BOOL caseSensitive
@property(copy) NSCharacterSet *charactersToBeSkipped
@property(retain) id locale
  • scanLocation,下次扫描开始的位置,如果该值超出了string的区域,将会引起NSRangeException,该属性在发生错误后重新扫描时非常有用。

  • caseSensitive,是否区分字符串中大小写的标志。默认为NO,注意:该设置不会应用到被跳过的字符集。

  • charactersToBeSkipped,在扫描时被跳过的字符集,默认是空白格和回车键。被跳过的字符集优先于扫描的字符集:例如一个scanner被跳过的字符集为空格,通过scanInt:去查找字符串中的整型数时,首先做的不是扫描,而是跳过空格,直到找到十进制数据或者其他的字符。在字符被扫描的时候,跳过功能就失效了。如果你扫描的字符和跳过的字符是一样的,结果将是未知的。被跳过的字符是一个唯一值,scanner不会将忽略大小写的功能应用于它,也不会用这些字符做一些组合,如果在扫描字符换的时候你想忽略全部的元音字符,就要这么做(比如:将字符集设置成“AEIOUaeiou”};

  • locale,scanner 的locale对它从字符串中区分数值产生影响,它通过locale的十进制分隔符区分浮点型数据的整数和小数部分。一个没有locale的scanner用非定域值。新的scanner若没有设置locale,使用默认locale。

扫描字符串

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
- (BOOL)scanCharactersFromSet:(NSCharacterSet *)scanSet   intoString:(NSString * _Nullable *)stringValue;
- (BOOL)scanUpToCharactersFromSet:(NSCharacterSet *)stopSet    intoString:(NSString * _Nullable *)stringValue;
- (BOOL)scanString:(NSString *)string  intoString:(NSString * _Nullable *)stringValue;
- (BOOL)scanUpToString:(NSString *)stopString   intoString:(NSString * _Nullable *)stringValue;
- (BOOL)scanDecimal:(NSDecimal *)decimalValue;
- (BOOL)scanDouble:(double *)doubleValue;
- (BOOL)scanFloat:(float *)floatValue;
- (BOOL)scanHexDouble:(double *)result;
- (BOOL)scanHexFloat:(float *)result;
- (BOOL)scanHexInt:(unsigned int *)intValue;
- (BOOL)scanHexLongLong:(unsigned long long *)result;
- (BOOL)scanInt:(int *)intValue;
- (BOOL)scanInteger:(NSInteger *)value;
- (BOOL)scanUnsignedLongLong:(unsigned long long *)unsignedLongLongValue;
@property(getter=isAtEnd, readonly) BOOL atEnd;
  • scanCharactersFromSet:intoString:扫描字符串中和NSCharacterSet字符集中匹配的字符,是按字符单个匹配的,例如,NSCharacterSet字符集为@"test123Dmo",scanner字符串为 @" 123test12Demotest",那么字符串中所有的字符都在字符集中,所以指针指向的地址存储的内容为"123test12Demotest"

  • scanUpToCharactersFromSet:intoString:扫描字符串直到遇到NSCharacterSet字符集的字符时停止,指针指向的地址存储的内容为遇到跳过字符集字符之前的内容

  • scanString:intoString:从当前的扫描位置开始扫描,判断扫描字符串是否从当前位置能扫描到和传入字符串相同的一串字符,如果能扫描到就返回YES,指针指向的地址存储的就是这段字符串的内容。例如scanner的string内容为123abc678,传入的字符串内容为abc,如果当前的扫描位置为0,那么扫描不到,但是如果将扫描位置设置成3,就可以扫描到了。

  • scanUpToString:intoString:从当前的扫描位置开始扫描,扫描到和传入的字符串相同字符串时,停止,指针指向的地址存储的是遇到传入字符串之前的内容。例如scanner的string内容为123abc678,传入的字符串内容为abc,存储的内容为123

  • scanDecimal:扫描NSDecimal类型的值,有关NSDecimal类型的值更多的信息可以查看:NSDecimalNumber

  • scanDouble :扫描双精度浮点型字符,溢出和非溢出都被认为合法的浮点型数据。在溢出的情况下scanner将会跳过所有的数字,所以新的扫描位置将会在整个浮点型数据的后面。double指针指向的地址存储的数据为扫描出的值,包括溢出时的HUGE_VAL或者 –HUGE_VAL,即未溢出时的0.0。

  • scanFloat:扫描单精度浮点型字符,具体内容同scanDouble

  • scanHexDouble: 扫描双精度的十六进制类型,溢出和非溢出都被认为合法的浮点型数据。在溢出的情况下scanner将会跳过所有的数字,所以新的扫描位置将会在整个浮点型数据的后面。double指针指向的地址存储的数据为扫描出的值,包括溢出时的HUGE_VAL或者 –HUGE_VAL,即未溢出时的0.0。数据接收时对应的格式为 %a 或%A ,双精度十六进制字符前面一定要加  0x或者 0X。

  • scanHexInt 扫描十六进制无符整型,unsigned int指针指向的地址值为 扫描到的值,包含溢出时的UINT_MAX。

  • scanHexLongLong 同scanHexDouble

  • scanInt 扫描整型,溢出也被认为是有效的整型,int 指针指向的地址的值为扫描到的值,包含溢出时的INT_MAX或INT_MIN。

  • scanInteger 同scanInt

  • scanLongLong 扫描LongLong 型,溢出也被认为是有效的整型,LongLong指针指向的地址的值为扫描到的值,包含溢出时的LLONG_MAX 或 LLONG_MIN。

实例:

1
2
3
4
5
#import
@interface ConditionJudgmentController : NSObject
//输入框中只能输入数字和小数点,且小数点只能输入一位,参数number 可以设置小数的位数,该函数在-(BOOL)textField:(UITextField *)textField shouldChangeCharactersInRange:(NSRange)range replacementString:(NSString *)string调用;
+(BOOL)isValidAboutInputText:(UITextField *)textfield shouldChangeCharactersInRange:(NSRange)range replacementString:(NSString *)string  decimalNumber: (NSInteger) number;
@end
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
#import "ConditionJudgmentController.h"
@implementation ConditionJudgmentController
+(BOOL)isValidAboutInputText:(UITextField *)textField shouldChangeCharactersInRange:(NSRange)range replacementString:(NSString *)string decimalNumber:(NSInteger)number{
NSScanner      *scanner    = [NSScanner scannerWithString:string];
NSCharacterSet *numbers;
NSRange         pointRange = [textField.text rangeOfString:@ "." ];
if  ( (pointRange.length > 0) && (pointRange.location < range.location  || pointRange.location > range.location + range.length) ){
numbers = [NSCharacterSet characterSetWithCharactersInString:@ "0123456789" ];
} else {
numbers = [NSCharacterSet characterSetWithCharactersInString:@ "0123456789." ];
}
if  ( [textField.text isEqualToString:@ "" ] && [string isEqualToString:@ "." ] ){
return  NO;
}
short remain = number;  //保留 number位小数
NSString *tempStr = [textField.text stringByAppendingString:string];
NSUInteger strlen = [tempStr length];
if (pointRange.length > 0 && pointRange.location > 0){  //判断输入框内是否含有“.”。
if ([string isEqualToString:@ "." ]){  //当输入框内已经含有“.”时,如果再输入“.”则被视为无效。
return  NO;
}
if (strlen > 0 && (strlen - pointRange.location) > remain+1){  //当输入框内已经含有“.”,当字符串长度减去小数点前面的字符串长度大于需要要保留的小数点位数,则视当次输入无效。
return  NO;
}
}
NSRange zeroRange = [textField.text rangeOfString:@ "0" ];
if (zeroRange.length == 1 && zeroRange.location == 0){  //判断输入框第一个字符是否为“0”
if (![string isEqualToString:@ "0" ] && ![string isEqualToString:@ "." ] && [textField.text length] == 1){  //当输入框只有一个字符并且字符为“0”时,再输入不为“0”或者“.”的字符时,则将此输入替换输入框的这唯一字符。
textField.text = string;
return  NO;
} else {
if (pointRange.length == 0 && pointRange.location > 0){  //当输入框第一个字符为“0”时,并且没有“.”字符时,如果当此输入的字符为“0”,则视当此输入无效。
if ([string isEqualToString:@ "0" ]){
return  NO;
}
}
}
}
NSString *buffer;
if  ( ![scanner scanCharactersFromSet:numbers intoString:&buffer] && ([string length] != 0) ){
return  NO;
} else {
return  YES;
}
}
@end

注意:上面的方法要结合UITextField的代理方法使用

1
2
3
4
-(BOOL)textField:(UITextField *)textField shouldChangeCharactersInRange:(NSRange)range replacementString:(NSString *)string
{
return  [ConditionJudgmentController isValidAboutInputText:textField shouldChangeCharactersInRange:range replacementString:string decimalNumber:3];
}

结束语:

平时读者在进行条件判断时,可能使用正则表达式或者用if语句比较多一点,其实NSScanner类为我们提供了一个非常好的方式进行判断。把它们结合起来使用,可以获得更好的效果。

这篇关于NSScanner:一个陌生的条件判断利器的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/279912

相关文章

poj 3259 uva 558 Wormholes(bellman最短路负权回路判断)

poj 3259: 题意:John的农场里n块地,m条路连接两块地,w个虫洞,虫洞是一条单向路,不但会把你传送到目的地,而且时间会倒退Ts。 任务是求你会不会在从某块地出发后又回来,看到了离开之前的自己。 判断树中是否存在负权回路就ok了。 bellman代码: #include<stdio.h>const int MaxN = 501;//农场数const int

zoj 1721 判断2条线段(完全)相交

给出起点,终点,与一些障碍线段。 求起点到终点的最短路。 枚举2点的距离,然后最短路。 2点可达条件:没有线段与这2点所构成的线段(完全)相交。 const double eps = 1e-8 ;double add(double x , double y){if(fabs(x+y) < eps*(fabs(x) + fabs(y))) return 0 ;return x + y ;

POJ1269 判断2条直线的位置关系

题目大意:给两个点能够确定一条直线,题目给出两条直线(由4个点确定),要求判断出这两条直线的关系:平行,同线,相交。如果相交还要求出交点坐标。 解题思路: 先判断两条直线p1p2, q1q2是否共线, 如果不是,再判断 直线 是否平行, 如果还不是, 则两直线相交。  判断共线:  p1p2q1 共线 且 p1p2q2 共线 ,共线用叉乘为 0  来判断,  判断 平行:  p1p

Codeforces Round #113 (Div. 2) B 判断多边形是否在凸包内

题目点击打开链接 凸多边形A, 多边形B, 判断B是否严格在A内。  注意AB有重点 。  将A,B上的点合在一起求凸包,如果凸包上的点是B的某个点,则B肯定不在A内。 或者说B上的某点在凸包的边上则也说明B不严格在A里面。 这个处理有个巧妙的方法,只需在求凸包的时候, <=  改成< 也就是说凸包一条边上的所有点都重复点都记录在凸包里面了。 另外不能去重点。 int

键盘快捷键:提高工作效率与电脑操作的利器

键盘快捷键:提高工作效率与电脑操作的利器 在数字化时代,键盘快捷键成为了提高工作效率和优化电脑操作的重要工具。无论是日常办公、图像编辑、编程开发,还是游戏娱乐,掌握键盘快捷键都能带来极大的便利。本文将详细介绍键盘快捷键的概念、重要性、以及在不同应用场景中的具体应用。 什么是键盘快捷键? 键盘快捷键,也称为热键或快捷键,是指通过按下键盘上的一组键来完成特定命令或操作的方式。这些快捷键通常涉及同

【408DS算法题】039进阶-判断图中路径是否存在

Index 题目分析实现总结 题目 对于给定的图G,设计函数实现判断G中是否含有从start结点到stop结点的路径。 分析实现 对于图的路径的存在性判断,有两种做法:(本文的实现均基于邻接矩阵存储方式的图) 1.图的BFS BFS的思路相对比较直观——从起始结点出发进行层次遍历,遍历过程中遇到结点i就表示存在路径start->i,故只需判断每个结点i是否就是stop

linux 判断某个命令是否安装

linux 判断某个命令是否安装 if ! [ -x "$(command -v git)" ]; thenecho 'Error: git is not installed.' >&2exit 1fi

封装MySQL操作时Where条件语句的组织

在对数据库进行封装的过程中,条件语句应该是相对难以处理的,毕竟条件语句太过于多样性。 条件语句大致分为以下几种: 1、单一条件,比如:where id = 1; 2、多个条件,相互间关系统一。比如:where id > 10 and age > 20 and score < 60; 3、多个条件,相互间关系不统一。比如:where (id > 10 OR age > 20) AND sco

JavaScript正则表达式六大利器:`test`、`exec`、`match`、`matchAll`、`search`与`replace`详解及对比

在JavaScript中,正则表达式(Regular Expression)是一种用于文本搜索、替换、匹配和验证的强大工具。本文将深入解析与正则表达式相关的几个主要执行方法:test、exec、match、matchAll、search和replace,并对它们进行对比,帮助开发者更好地理解这些方法的使用场景和差异。 正则表达式基础 在深入解析方法之前,先简要回顾一下正则表达式的基础知识。正则

使用条件变量实现线程同步:C++实战指南

使用条件变量实现线程同步:C++实战指南 在多线程编程中,线程同步是确保程序正确性和稳定性的关键。条件变量(condition variable)是一种强大的同步原语,用于在线程之间进行协调,避免数据竞争和死锁。本文将详细介绍如何在C++中使用条件变量实现线程同步,并提供完整的代码示例和详细的解释。 什么是条件变量? 条件变量是一种同步机制,允许线程在某个条件满足之前进入等待状态,并在条件满