iOS实现简书的账号识别方式(正则表达式)

520 阅读6分钟

###开篇 通过简书iOS客户端登录,我们会看到请输入手机号或者邮箱登录,但是我们随机输入1234567的时候,便会弹出手机格式不正确,同样也会识别我们的邮箱格式,那么我们在项目中怎么实现这种判断呢?

0E471361-060C-4D93-913F-73622F89BC60.png
这就是我们今天要说的正则表达式。 ####正则表达式的介绍 正则表达式有多种使用方法,根据我们的需要,我们是要判断输入是否合法,还是查找指定的内容,还是捕获多个输入的内容,可以选用不同的方法,今天我们主要说一下,判断输入是否合法,用谓词的方式来创建正则表达式,以后如果有用到的,再写一篇吧。 #####谓词(NSPredicate) NSPredicate是什么意思呢。NSPredicate即谓词逻辑。和数据库的SQL语句具有相似性,都是从数据堆中根据条件进行筛选。 谓词,指定过滤器的条件,将符合条件的对象保留下来 ,一般用谓词过滤数组中指定的元素 ,定义谓词对象,谓词对象中包含了过滤元素,使用谓词条件过滤之后得到我们想要的结果。总得来说呢,Cocoa提供了一个类NSPredicate类,该类主要用于指定过滤器的条件,该对象可以准确的描述所需条件,对每个对象通过谓词进行筛选,判断是否与条件相匹配。谓词表示计算真值或假值的函数。 下面进行代码举例,首先我们创建一个名为Student的类,然后给他设定id,name,height。之后我们引入这个类给他赋值,代码如下:

    NSMutableArray *array=[NSMutableArray arrayWithCapacity: 5];
    
    Student *student1=[[Student alloc] init];
    [student1 setPid: 1];
    [student1 setName: @"xiaoming"];
    [student1 setHeight: 168];
    [array addObject: student1];
     
    Student *student2=[[Student alloc] init];
    [student2 setPid: 2];
    [student2 setName: @"dahuang"];
    [student2 setHeight: 178];
    [array addObject: student2];
    
    Student *student3=[[Student alloc] init];
    [student3 setPid: 3];
    [student3 setName: @"erhuang"];
    [student3 setHeight: 188];
    [array addObject: student3];

在上述的代码中,我们创建了一个可变数组,和三个小学生(-_-,不要吐槽我给他们的命名,懒得想..). 下面我们就要创建我们的谓词了,我们想要筛选出一个id 大于1 ,且升高小于180的小学生来,应该怎么做呢。

 NSPredicate *pre = [NSPredicate predicateWithFormat:
                        @" pid>1 and height<188.0"];
     int i=0;
    for(i=0;i<[array count];i++){
        Student *stu=[array objectAtIndex: i];
        //遍历数组,输出符合谓词条件的Student 的name。
        if ([pre evaluateWithObject: stu]) {
            NSLog(@"11--%@",[stu name]);
        }
    }

谓词还可以帮我们做出一些列的筛选,如下:

  //快速筛选数组内容   以及占位符的使用
    NSPredicate *pre2 = [NSPredicate predicateWithFormat:@"pid>%d",1];
    
    NSMutableArray *arrayPre2=[array filteredArrayUsingPredicate: pre2];
    
    NSLog(@"%@",[[arrayPre2 objectAtIndex: 0] name]);

字符串的处理

 //name以x开头的
    NSPredicate  *predicate3 = [NSPredicate predicateWithFormat:@"name BEGINSWITH 'x'"];

 //name以g结尾的
    NSPredicate  *predicate4 = [NSPredicate predicateWithFormat:@"name ENDSWITH 'g'"];

  //name中包含字符a的
    NSPredicate  *predicate5 = [NSPredicate predicateWithFormat:@"name CONTAINS 'a'"];

   //like 匹配任意多个字符
    //name中只要有r字符就满足条件
    NSPredicate  *predicate6 = [NSPredicate predicateWithFormat:@"name like '*r*'"];

  //?代表一个字符,下面的查询条件是:name中第二个字符是r的
    NSPredicate  *predicate7 = [NSPredicate predicateWithFormat:@"name like '*?r*'"];

谓词的使用就先到这里,是不是和数据库使用感觉差不多呢,下面说说正题的时候来了,我们要怎么进行把谓词和正则表达式结合进行相应的判断呢。 ####利用正则表达式实现判断 首先我们建立一个UI,包含一个UITextField输入框和一个提交按钮,当我们点击按钮的时候对我们提交的数据进行判断。 我们创建一个类,写出对手机号和邮箱进行判断的方法

+ (BOOL)GS_isMobileNumber:(NSString *)mobileNum;

+ (BOOL)GS_isEmailQualified:(NSString *)emailStr;

然后实现检测的方法

+ (BOOL)GS_isMobileNumber:(NSString *)mobileNum
{
    /**
     * 手机号码
     * 移动:134[0-8],135,136,137,138,139,150,151,157,158,159,182,187,188
     * 联通:130,131,132,152,155,156,185,186
     * 电信:133,1349,153,180,189
     */
    //    NSString * MOBILE = @"^1(3[0-9]|5[0-35-9]|8[025-9])\\d{8}$";
    
    
    /**
     * 中国移动:China Mobile
     * 134[0-8],135,136,137,138,139,150,151,157,158,159,182,187,188
     */
    //    NSString * CM = @"^1(34[0-8]|(3[5-9]|5[017-9]|8[278])\\d)\\d{7}$"; // China Mobile phoneNum
    
    /**
     * 中国联通:China Unicom
     * 130,131,132,152,155,156,185,186
     */
    //    NSString * CU = @"^1(3[0-2]|5[256]|8[56])\\d{8}$"; // China Unicom phoneNum
    
    
    /**
     * 中国电信:China Telecom
     * 133,1349,153,180,189
     */
    //    NSString * CT = @"^1((33|53|8[09])[0-9]|349)\\d{7}$"; // China Telecom phoneNum
    
    /**
     *  The following 4 predicate can tell which carrier the number is from.
     */
    //    NSPredicate *regextestmobile = [NSPredicate predicateWithFormat:@"SELF MATCHES %@", MOBILE];
    //    NSPredicate *regextestcm = [NSPredicate predicateWithFormat:@"SELF MATCHES %@", CM];
    //    NSPredicate *regextestcu = [NSPredicate predicateWithFormat:@"SELF MATCHES %@", CU];
    //    NSPredicate *regextestct = [NSPredicate predicateWithFormat:@"SELF MATCHES %@", CT];
    
    
    //Only Check if the string is a valid telephone number, ignoring the carrier info.
    
    NSString *isMobileRegex = @"^(((13[0-9]{1})|(15[0-9]{1})|(17[0-9]{1})|(18[0,5-9]{1}))+\\d{8})$";

    
//    NSString *isMobileRegex = @"^((\\+86)|(\\(\\+86\\)))?(((13[0-9]{1})|(15[0-9]{1})|(17[0-9]{1})|(18[0,5-9]{1}))+\\d{8})$";
    

    NSPredicate *mobileRegex = [NSPredicate predicateWithFormat:@"SELF MATCHES %@", isMobileRegex];
    
    if (([mobileRegex evaluateWithObject: mobileNum] == YES))
    {
        return YES;
        
    }else{
        return NO;
    }
}
//检测是否为邮箱
+ (BOOL)GS_isEmailQualified:(NSString *)emailStr
{
    NSString *pattern = @"[A-Z0-9a-z._%+-]+@[A-Za-z0-9.-]+\\.[A-Za-z]{2,4}";
    NSRegularExpression *regex = [[NSRegularExpression alloc] initWithPattern:pattern options:0 error:nil];
    NSArray *results = [regex matchesInString:emailStr options:0 range:NSMakeRange(0, emailStr.length)];
    return results.count > 0;
}

然后我们调用创建的方法,去检测是否为手机号

   if ([GSRegularExpression GS_isMobileNumber:_textfiled.text]) {
        
        NSLog(@"输入的是正常的手机号码");
        
    }else{
        
        NSLog(@"输入的不是正常的手机号码");
        
    }

根据打印结果,显示我们是正确的这样,我们就实现了正则表达式对手机号和邮箱的判定(邮箱的这是这种调用方式)。 ###后记 正则表达式去判断一些结果,是比较常用的,比如,身份证,车牌,机型,IP地址,输入是否为全数字等。利用他我们可以实现一些看来比较复杂的效果,现在我们已经实现了简书的登录对手机号和邮箱识别的功能,希望对大家有所帮助。文中有不对的地方,欢迎大家积极指出。 ###补充

#####补充1: 例如我们给定一个用来判断密码的正则

+(BOOL)judgePassWordLegal:(NSString *)pass{
    BOOL result ;
    // 判断长度大于8位后再接着判断是否同时包含数字和字符
    NSString * regex = @"^(?![0-9]+$)(?![a-zA-Z]+$)[0-9A-Za-z]{6,20}$";
    NSPredicate *pred = [NSPredicate predicateWithFormat:@"SELF MATCHES %@", regex];
    result = [pred evaluateWithObject:pass];
    NSLog(@"%hhd",result);
    return result;
}

对以上正则我们来怎么理解他呢? 首先开头标记 "^" 结尾标记 "$" ,[0-9A-Za-z]{6,20}表示内容包含数字和字母并且限制了他的位数,(?![0-9]+$)这个形式 正向否定预查,具体的格式意义,这里放上一个网上的链接就不一一写了,正则表达式的用法 。 #####补充2: 关于评论里面的登录的问题这里做一下补充:基本流程仅作参考,有不当的欢迎指出。

  • 已授权过—— 授权之后,请求接口——>已绑定过——>返回登录信息

  • 未授权—— 授权之后,请求接口——>没有绑定过——>(手机号已经注册)——>(openid和userid或者手机号绑定,然后返回登录信息)

  • 未授权——> 授权之后,请求接口——>没有绑定过——>(没注册过的)——>创建账号,绑定id——>返回登录信息