iOS开发的那些坑

来源:互联网 发布:linux保存配置 编辑:程序博客网 时间:2024/05/18 04:02

今天是农历小年,程序员大咖祝大家小年夜快乐~

文/TheAlchemist(授权)原文链接:http://www.jianshu.com/p/da80c4b7190e

最近重新拿起了iOS的开发,使用OC和Swift混编,碰到了一些比较棘手的问题,在这里记录下来,方便自己以后或他人不再入坑。这篇文章的内容包含:

UITableViewCell的真实结构在iOS的环境下使用正则表达式如何优雅的隐藏tabbar如何修改navigationbar的颜色运行时修改使用autolayout的viewnavigation controller子view偏移问题CoreData的使用总结解决git的Xcode冲突

1.UITableViewCell的真实结构

目标是想把UITableview在editing模式下的drag按钮去掉,换成自己的样式,但又想保留原生拖动排序的行为,所以研究了很多方法,最后还是找到了UITableViewCell的结构,并将拖动按钮替换。不多说,上代码。

//打印出来的自定义的cell在editing模式下的结构    (lldb) po self<QKZhi_iOS.EditingCell: 0x7c148800; baseClass = UITableViewCell; frame = (0 0; 320 56); autoresize = W; layer = <CALayer: 0x7d086b40>>(lldb) po self.subviews 5 elements  - [0] : <UITableViewCellContentView: 0x7d086d10; frame = (38 0; 242 55.5); opaque = NO; gestureRecognizers = <NSArray: 0x7d0882b0>; layer = <CALayer: 0x7d086de0>>  - [1] : <_UITableViewCellSeparatorView: 0x7d087c40; frame = (15 55; 305 1); layer = <CALayer: 0x7d0852c0>>  - [2] : <_UITableViewCellSeparatorView: 0x7b163240; frame = (15 55.5; 305 0.5); layer = <CALayer: 0x7b162de0>>  - [3] : <UITableViewCellEditControl: 0x7b164700; frame = (0 0; 47 56); opaque = NO; layer = <CALayer: 0x7b164950>>  - [4] : <UITableViewCellReorderControl: 0x7b164d10; frame = (268 0; 52 56); opaque = NO; autoresize = LM; layer = <CALayer: 0x7b164f60>>(lldb) po self.subviews.last   Optional<UIView>    - Some : <UITableViewCellReorderControl: 0x7b164d10; frame = (268 0; 52 56); opaque = NO; autoresize = LM; layer = <CALayer: 0x7b164f60>>(lldb) po self.subviews.last?.subviews   Optional<Array<UIView>>     Some : 1 elements      - [0] : <UIImageView: 0x7b165130; frame = (15 24; 22 8.5); opaque = NO; userInteractionEnabled = NO; tag = 1; layer = <CALayer: 0x7b165210>>(lldb)

可以看到此时contentView左右都向内缩进了一定的距离,最后有一个view叫UITableViewCellReorderControl,我想就是它了,然后看它的subviews,竟然包含了一个UIImageView,果断替换之,代码如下。

override func layoutSubviews() {    super.layoutSubviews()    setupReorderControl()}func setupReorderControl() {    if (self.reorderControl != nil) {        return;    }    for view in self.subviews {        if view.description.containsString("UITableViewCellReorderControl") {            self.reorderControl = view        }    }    if ((self.reorderControl) != nil)    {        let imageOfReorder = self.reorderControl?.subviews[0] as? UIImageView        imageOfReorder?.removeFromSuperview()    }}

此时就完成了将拖动按钮隐藏的功能,但是注意,在这里如果想通过设置reorderControl的frame去改变它的位置是不成功的,我想可能它的布局使用autolayout,并没有深入的再去研究。

2. 在iOS的环境下使用正则表达式

在iOS下使用正则似乎挺麻烦的,如果只是想用正则判断格式,那么可以使用NSPredicate:

let string = "abcd"let predicate = NSPredicate.init(format: "self matches %@", "^a(b|c)d$")predicate.evaluateWithObject(string)

如果要使用正则判断格式,然后获取group的子字符串,就比较麻烦了,并没有Swift的原生方法来做,必须使用NSRegularExpression,话不多说,上代码:

let reg = try! NSRegularExpression.init(pattern: "^a(b|c)d$", options: NSRegularExpressionOptions.CaseInsensitive)let matches = reg.matchesInString(string, options: NSMatchingOptions.init(rawValue: 0), range: NSMakeRange(0, string.characters.count))if matches?.count > 0 {   let returnString = (string as NSString).substringWithRange(matches![0].rangeAtIndex(1))}

3. 如何优雅的隐藏tabbar

很多APP都使用TabBarController套NavigationController的方法来作为应用的框架,那么隐藏TabBar就成了一个必要的功能,目前最简单的方法还是使用hidesBottomBarWhenPushed来实现,最简单的方法就是在要隐藏tab bar的Controller里写入下面的方法,来覆默认值。

- (BOOL) hidesBottomBarWhenPushed {    return (self.navigationController.topViewController == self);}

4. 如何修改navigationbar的颜色

[self.navigationController.navigationBar setTranslucent:NO];self.navigationController.navigationBar.barTintColor = [UIColor redColor];//在此方法中还原颜色-(void)viewWillDisappear:(BOOL)animated{self.navigationController.navigationBar.barTintColor = [UIColor whiteColor];[super viewWillDisappear:animated];}

5. 运行时修改使用autolayout的view

方案是运行时给view添加constraint。

NSLayoutConstraint *c = [NSLayoutConstraint constraintWithItem:view1                                                     attribute:NSLayoutAttributeTop                                                     relatedBy:NSLayoutRelationEqual                                                        toItem: view2                                                     attribute: NSLayoutAttributeTop                                                    multiplier:1                                                      constant:2];view1(的顶部) = view2(的顶部) * 1 + 2

这个代码的意思是把view1的顶部放在view2的顶部向下两个像素的地方(因为此时view1是view2 的subview,所以达到的效果就是view1在view2顶部向下2个像素的地方)

如下是所有属性:

NSLayoutAttributeLeft    视图的左边NSLayoutAttributeRight    视图的右边NSLayoutAttributeTop    视图的上边NSLayoutAttributeBottom    视图的下边NSLayoutAttributeLeading    视图的前边NSLayoutAttributeTrailing    视图的后边NSLayoutAttributeWidth    视图的宽度NSLayoutAttributeHeight    视图的高度NSLayoutAttributeCenterX    视图的中点的X值NSLayoutAttributeCenterY    视图中点的Y值NSLayoutAttributeBaseline    视图的基准线NSLayoutAttributeNotAnAttribute    无属性

6. navigation controller子view偏移问题

场景是这样的:页面使用UItableview布局,由于页面比较复杂,在push进去再出来时第一个cell会向下偏移。解决方案:

 //在viewDidLoad中加入:if ([self respondsToSelector:@selector(edgesForExtendedLayout)]){    self.edgesForExtendedLayout = UIRectEdgeNone;}

7. CoreData的使用总结

////  CoreDataHelper.swift//  TestCode////  Created by lk on 16/7/6.//  Copyright © 2016年 lk. All rights reserved.//import Foundationimport CoreData@objc class CoreDataHelper: NSObject {static let KEY_OF_MAX_ORDER_NUMBER_OF_I = "MIK"static let KEY_OF_MAX_ORDER_NUMBER_OF_O = "MOK"var context : NSManagedObjectContext!static let sharedInstance = CoreDataHelper()private func fetchQuery(request request : NSFetchRequest) -> [NSManagedObject] {    do{        let fetchResults = try context?.executeFetchRequest(request) as! [NSManagedObject]!        return fetchResults    }catch let error as NSError {        print("查询失败: \\(error)")    }    return []}private func save() -> Void {    //保存    do{        try context.save()    } catch let error as NSError {        print("保存失败: \\(error)")    }}//TODO: - 这里可能存在溢出漏洞,以后再说private func addIncrement(key _key : String, newNumber : Int) -> Void {    if _key == CoreDataHelper. KEY_OF_MAX_ORDER_NUMBER_OF_O || _key == CoreDataHelper.KEY_OF_MAX_ORDER_NUMBER_OF_I {        NSUserDefaults.standardUserDefaults().setInteger(newNumber, forKey: _key)    }}//Mark: - 重新计算最大值func recaculateOptionalMaxOrder() -> Void {    let topOptinal = getOptionals(count: 1)    if topOptinal.count > 0 {        NSUserDefaults.standardUserDefaults().setInteger(topOptinal[0].order!.integerValue, forKey: CoreDataHelper.KEY_OF_MAX_ORDER_NUMBER_OF_O)    }}//MARK: - 添加func addXXX(code code: String, name: String) -> Void {    if (context == nil) {        print("添加失败,context为nil")        return    }    let s = NSEntityDescription.insertNewObjectForEntityForName("S", inManagedObjectContext: context) as! S    //对象赋值    s.code = code    s.name = name    s.numericalCode = info.code    s.type = info.type    save()}//MARK: - 删除func removeAll() -> Void {    let fetchRequest = NSFetchRequest(entityName: "S")    let fetchResults = fetchQuery(request: fetchRequest)    for info:NSManagedObject in fetchResults {        //删除对象        context.deleteObject(info)    }    save()}//MARK: - 查询func getObject(code code : String) -> S? {    if (context == nil) {        print("查询失败,context为nil")        return nil;    }    let fetchRequest = NSFetchRequest(entityName: "S")    let predicate = NSPredicate(format: "code == %@ || numericalCode == %@", code, code)    fetchRequest.predicate = predicate    let fetchResults = fetchQuery(request: fetchRequest)    if let ses = fetchResults as? [S] {        if ses.count > 0{            return ses[0]        }else{            return nil        }    }    return nil}//MARK: - 搜索func searchStocks(filterString : String) -> [Stock] {    var attribute = "code";    let num = Int(filterString)    if filterString.lengthOfBytesUsingEncoding(NSUTF8StringEncoding) <= 0 {        return []    } else if num == nil {        //TODO         attribute = "name"    }    let fetchRequest = NSFetchRequest(entityName: "S")    fetchRequest.predicate = NSPredicate.init(format: "%K CONTAINS %@", attribute, filterString)    let fetchResults = fetchQuery(request: fetchRequest)    print("结果:")    print(fetchResults)    var ses = [S]()    for res in fetchResults {        let s = res as! S        ses.append(s)    }    return ses}




原创粉丝点击