记录UITextField和联系人的两个小坑

坑1:关于UITextField限定长度.

需求很简单.就是限定一个UITextField的字符长度在n个中文之内(中英文均有可能).

首先,需求中有中英文的要求,那么如何计算长度呢,很简单:

lengthOfBytesUsingEncoding

1个中文长度等于3个英文长度.在UITextField的代理里面直接搞定它.

- (BOOL)textField:(UITextField *)textField shouldChangeCharactersInRange:(NSRange)range replacementString:(NSString *)string{
    NSString *strWhenChanged = [textField.text stringByReplacingCharactersInRange:range withString:string];
    return [strWhenChanged lengthOfBytesUsingEncoding:NSUTF8StringEncoding] <= 9;
}

似乎好了,测试一下,好像不那么对劲,当字数满了以后,无法删除了.

ok,继续改进

- (BOOL)textField:(UITextField *)textField shouldChangeCharactersInRange:(NSRange)range replacementString:(NSString *)string{
    if ([string isEqualToString:@""]){
        return YES;
    }
    NSString *strWhenChanged = [textField.text stringByReplacingCharactersInRange:range withString:string];
    return [strWhenChanged lengthOfBytesUsingEncoding:NSUTF8StringEncoding] <= 9;
}

好了,可以删除了.到了规定字符数以后也无法继续输入.可是,问题又来了,全拼键盘的联想仍然能用.

妈蛋,我们关了它

textfiled.autocorrectionType = UITextAutocorrectionTypeNo;

啊,好像没用...还是可以联想.而且有个特别严重的问题,点击联想的文字,代理并不会触发!

替换成通知好了,我们使用通知:

UITextFieldTextDidChangeNotification

在使用一个屌屌的属性:

markedTextRange

就可以啦.

不过还有一个问题,如果规定n个字符,如果超过n个字符是截断呢还是干脆就置空?截断似乎要好很多.

我们需要做一个递归的截断方法:

extension String{
    func limitLengthByUTF8(limitedLenght: Int) -> String{
        if self.lengthOfBytesUsingEncoding(NSUTF8StringEncoding) <= limitedLenght{
            return self
        }
        else{
            let limitedStr = self.substringToIndex(advance(self.endIndex, -1))
            return limitedStr.limitLengthByUTF8(limitedLenght)
        }
    }
}

最终是这样的:

    [[NSNotificationCenter defaultCenter] addObserverForName:UITextFieldTextDidChangeNotification object:nil queue:nil usingBlock:^(NSNotification *note) {
        UITextField *field = (UITextField *)note.object;
        if(field.markedTextRange){
            return;
        }
        field.text = [field.text limitLengthByUTF8:45];
    }];

坑在哪里?

流着眼泪的说,我用的是第三方输入法(百度).测试的时候,这个点根本没出任何问题.第三方输入法有自己的toolbar,联想的时候并不会造成代理不响应的情况.

切记切记,一定在开发中要使用原生的输入法进行测试啊.

坑2:坑爹的联系人

又出问题了!怎么问题这么多!

有一个需求是读取联系人列表,然后进行各项操作,诸如分类(A-Z),筛选之类的.在分类的时候,使用CFStringTransform获取联系人拼音首字母,的确会有一些资源消耗.
在普通情况下,一两百个联系人的时候,并没有明显的卡顿,所以当时也没有做异步的操作.

可是,在测试的时候,居然出现了非常明显的卡顿,这...

遇上bug,首先修改为异步,dispatch_async走起.

然后查询为何会卡顿.

通过查询,不可思议的一幕来了:200来个联系人,居然有3000多个电话号码!

无法理解,无法理解,继续抓虫!

通过打印,发现了如下的情况,在系统电话本中如图所示:

记录UITextField和联系人的两个小坑_第1张图片
记录UITextField和联系人的两个小坑_第2张图片

原来,这就是安全卫士类的软件(360等)拦截骚扰电话的数据库.

那么怎么来的呢?很有可能是从其他地方导入.

目前没有特别好的办法处理,不知道大家有没有好的方案?

我暂时只想到2种:

  1. 是找寻此种号码的规律,将其忽略.不过因为样本不足,此种方法未被采纳.有360,还有百度,还有腾讯,未来还有xxx卫士啊.

  2. 当单个联系人中超过n个号码的话,就判定该联系人为非正常联系人.n根据情况设定.当然这种方法有错杀的风险.
    的确有可能有人会把公司当做一个联系人,所有的同事的联系方式都建立在该公司(联系人)下.不过这种方式蛮蠢的,毕竟无法搜索...

你可能感兴趣的:(记录UITextField和联系人的两个小坑)