美文网首页君赏博客GB 移动小组
建议大型项目用上Try Catch

建议大型项目用上Try Catch

作者: 君赏 | 来源:发表于2016-11-15 10:32 被阅读182次

    建议大型项目用上Try Catch

    我们在平时项目做功能的时候,经常会遇到崩溃的情况。如果是我们在开发测试阶段,我们可以找到原因修复。但是遇到已经上线,出现这种问题。要么使用JSPatch进行热修复,但是使用热修复修复苦只有写的人知道。

    本来就是一个方法一个单词写错,一个值没做异常处理,就要重写整个方法。对于像我这样根本不懂JS语法和不精通JSPatch的人来说,真的是蓝瘦,香菇。

    最近写的项目用Swift语法进行编写的,对于之前我们在Object-CNSError**类型的指针标识遇到了什么错误,现在转成Swift方法直接进行throws进行抛异常。

    这样我们就必须使用Do Catch进行捕获异常了,我觉得苹果这样做真的挺好的。

    比如我刚刚写的项目,就用上了,感觉用完顿时高大上了许多。

    import Cocoa
    
    enum OFileMagerImportError:Error {
      case cannel //点击了取消的按钮
      case error(message:String)
    }
    
    class OFileManger: NSObject {
      /*
       * 导入配置文件
       * return 返回一个数组对象 可能返回为空
       */
      class func importAction() throws -> [Any] {
          let openPannel = NSOpenPanel()
          openPannel.allowedFileTypes = ["ork"]; // 只允许读取.ork的文件类型
          openPannel.allowsMultipleSelection = false // 设置不允许多选
          let buttonIndex = openPannel.runModal()
          guard buttonIndex == NSFileHandlingPanelOKButton else {
              throw OFileMagerImportError.cannel
          }
          guard openPannel.urls.count > 0 else {
              throw OFileMagerImportError.error(message:"你还没有选择任何的配置文件!")
          }
          guard let fileName = openPannel.urls.first else {
              throw OFileMagerImportError.error(message:"你还没有选择任何的配置文件!")
          }
          guard let jsonData = try? Data(contentsOf: fileName) else {
              throw OFileMagerImportError.error(message:"读取文件内容失败!")
          }
          guard let jsonObj = try? JSONSerialization.jsonObject(with: jsonData, options: .allowFragments) else {
              throw OFileMagerImportError.error(message:"配置文件内容不是一个JSON文本!")
          }
          guard let configList:[Any] = jsonObj as? [Any] else {
              throw OFileMagerImportError.error(message:"配置文件内容不是一个数组对象!")
          }
          return configList
      }
    
    
    }
    
    

    调用的时候我们可以这么写。

    @IBAction func importAction(_ sender: Any) {
            do {
                let configList:[Any] = try OFileManger.importAction()
                print(configList)
            } catch OFileMagerImportError.error(let message) {
                OAlertMessage.alert(message: message)
            } catch {
                print("抛出了其他不给用户看到的错误!")
            }
        }
    

    我们就可以做一些错误的处理。但是只是这样,我心里面还不是十分强烈想用。

    后来线上的App在双11那天晚上突然出现打不开崩溃的情况,在外面准备吃饭庆祝的强行喊回去解决问题。

    回去觉得应该是那里崩溃了吧,应该也是十分的容易解决。

    后来发现因为接口因为最近上线的功能出问题了,就回滚了代码。导致之前一些接口的字段没有返回,恰好那个接口没有用Model进行映射,所以就出现崩溃了。

    经理一直问这个接口回滚好几天了为啥突然出问题了,原来是接口做了缓存,会根据后台返回是否需要更新才会重新的请求接口。

    双11那天正好运维更新了数据,导致接口重新请求,少了字段导致崩溃。最后让后台强行把拿两个字段返回,最快的解决问题。

    虽然让后台加上了字段,但是我觉得我们App这里没有做好兼容,才导致这一次事故。幸亏没影响多久,再说客户都是外国人,那个时间还没睡醒。

    我们平时遇到的一些崩溃问题所在

    • 数组越界
    • 给一个可变数组强行赋值不可变,之后当做可变数组用
    • 字典值不存在
    • 后台返回的字段不统一,本来有值是字典,没值是字符串。

    导致的崩溃自然不止这些,但是这些占据了崩溃里面大部分,前三个我们可以做一下异常处理,最后一个最好还是让后台规范起来,这是长远之际。或者使用Model进行映射。

    怎么做异常处理

    • 做很多if else判断
    • 使用Try catch进行处理

    虽然两个都要写很多的代码,if else的代码更加优美一点,但是就怕有一种情况你没写出来。觉得最好的办法还是用try catch进行。

    @try@catch@throw,和 @finally。这四个关键词进行异常处理。

    我们看一下常见的一个数组越界的例子:

    NSArray *tests = @[];
    NSNumber *num = tests[0];
    

    我们不用运行就可以知道这个地方会崩溃,因为我们初始化的数组是空的。但是我们强行取一个元素,导致崩溃。

    假设我们现在用上了try catch的代码,会如何呢。

    NSArray *tests = @[];
    @try {
        NSNumber *num = tests[0];
    } @catch (NSException *exception) {
        NSLog(@"%@",exception.userInfo);
    } @finally {
        NSLog(@"我做一些必须的处理");
    }
    

    经过测试,这个代码会在Debug的时候断点,但是强行运行还是可以运行的。但是在Release发布之后运行完全没有任何的影响。

    虽然我们平时写代码上面,不可能让所有出现异常的地方写上这样的代码。但是对着我们的经验,我们对于一眼就知道会出崩溃的地方加上,还是能让线上的应用少出很多问题的。

    关于Try catch的资料可以查看下面资料:

    iOS捕获异常,常用的异常处理方法

    Swift中错误和异常处理

    相关文章

      网友评论

      本文标题:建议大型项目用上Try Catch

      本文链接:https://www.haomeiwen.com/subject/ribkpttx.html