医院服务器性能,服务器诊所:编写出色的异常

本文探讨了异常处理在编程中的重要性,强调了理解异常如何解决实际问题、何时捕获以及如何抛出异常。作者指出,异常处理不应仅仅关注语法和局部语义,更应注重提供清晰的错误信息,以提升用户体验并保障应用程序的安全性。通过示例,展示了如何避免不匹配的捕获、提供详细的异常信息以及正确使用异常层次结构。同时,提出了在设计异常时避免过度复杂的继承结构,建议使用参数化异常而非过多的子类化。
摘要由CSDN通过智能技术生成

如何考虑异常

教科书和类似的参考资料一直以来都集中在异常的语法和局部语义方面。通过它们的充分介绍,大多数程序员都能阅读带有异常的代码并能解释其作用。它们所欠缺的是一种对有效风格的感觉。要找到这种感觉,您需要,

重点了解您要求异常为您解决的问题,

如何捕获异常,以及,

如何抛出异常。

本月的专栏文章列举了几个示例来说明如何实现上述三点。

在研究这几个示例之前,不妨采用一种可能与您第一次学习异常时不同的方式来考虑异常,“热热身”。由您喜欢的语言提供的异常系统是不适合于最终用户查看的。相反,可以把异常当作脚手架,在完成应用程序之后,再“拆除”这些“脚手架”。也许您曾在课堂上学过阅读这样的异常,如下所示

caughtexceptioninmain()

java.lang.SomeException:uglyinput

at...

当然,这个技巧对程序员是有价值的。可是它绝不能被强加给最终用户。一个完整的应用程序应该从来不说“有异常”;所有呈现给最终用户的报告都应该用下面的这种本机语言来写,或许更接近于这样

Theconfigurationfile'folder/thing.cfg'

appearstobecorrupt,asline#17cannot

beparsed.

正是这种清晰性对应用程序的安全性施加了直接的压力。其原因在于:用户及其管理员一次又一次地证明他们对不能理解的事物的反应,是简化系统直至得到自己期望的行为。如果他们读到“未发现文件(filenotfound)”,他们会随意地从别处复制一些文件,而不考虑特权或许可权。而保证应用程序安全性的最可靠的方法之一就是使应用程序工作,这样用户才会理解它的运作。聪明的用户会因为急于“让程序工作”而破坏几乎所有安全性设置。

不是所有程序员都认同我这种观点。有不少高级软件工程师冷静地提议说,用户输入错误的数据或错用应用程序都是咎由自取。我在这里不是要讨论这种态度的道德问题;只是注意到,在开发人员和最终用户采取这种互相对立的姿态时,安全性正在不断地被破坏。

因此,在某个特定开发项目中,使用异常的第一步,也往往是最容易被忽视的一步,就是确定程序对异常的需求。这一点一定要搞清楚。当客户或主管在指示程序应如何处理格式良好的输入数据时,抓住机会,对万一发生错误时程序的具体操作细节同他们达成共识。给自己足够的时间去会见客户。想象一下:最终客户可能会把同程序“接触时间”的大部分都消耗在查看程序所显示的错误消息上。这并不是骇人听闻,对许多应用程序来说“正常”操作是相当快的,而对于错误的响应,人们需要花不少时间来思考。错误消息及对应操作同程序的其它部分相比,值得花同样多的技术。

事实上,我会对那些让最好的人才集中精力于错误处理方面,而不是传统编程中比较“花哨”的方面(比如图形用户界面(GUI)外观编程)的项目,更感到高兴。理由在于:一个有错误但带有优秀的错误处理机制的应用程序,比近乎完美但其错误处理机制却不友好的应用程序,更能赢得最终用户的欢心。

在完成了第一轮需求分析后,您手中拥有的这些叙述性说明会使程序的异常处理设计更为合理且更有价值。现在的挑战则是这篇专栏文章的读者所感兴趣的技术问题。

捕获

Python作为一种方便的工具,可用来表达示例用法。我经常遇到类似于这样的缺陷:

清单1.不匹配的捕获

try:

process(some_file)

except:

alert("errorinopening"'%s'%some_file)

发现问题没有?异常的语法和语义不匹配,这有点类似于一个公务员,在向选民承诺要注意他们最关心的问题,尤其是游泳池开放时间。尽管这样的语句形式上正确,其失衡却会使听众感到震惊,暗示有更深层的问题。

上面这个不匹配的捕获也存在类似问题:它捕获了所有错误,但仅仅只报告了“打开文件时出现错误(errorinopening)”。这样写会好一点:

清单2.均衡性较好的捕获

try:

process(some_file)

exceptIOError:

alert("errorinopening"'%s'%some_file)

许多程序员由衷地认为这两个例子是等价的,因为一种可能的理由是,“记录process只是用来生成IOError”。在这层意义上讲,应用程序在这两个示例中的执行,的确毫无差别。可是源代码不只是给计算机用的;更重要的是必须向身为人类的读者表达其含意。如果您的代码假设某个特定的异常一定是一个IOError,那么利用该语言的精确性,就这么说。

第二个示例仍不能完全防止让最终用户看到“原始”异常的危险。实际上,即使process()在当前版本中被明白无误地记录为仅抛出IOError,但我仍要求在编写该段代码时至少达到下面这样的详细程度:

清单3.形式均衡且全面的捕获

try:

process(some_file)

exceptIOError:

alert("errorinopening"'%s'%some_file)

except:

alert("internalandcompletelyunexpectedproblem")

当然,对我们而言,拥有完整且正确记录的接口是一种少有的奢侈。在开发工作中许多语言采用了一种有用的技术-使用异常系统内置的内省。这使我们的示例变成这样:

清单4.形式均衡且全面的捕获,并带有信息性的“缺省设置”

try:

process(some_file)

exceptIOError:

alert("errorinopening"'%s'%some_file)

except:

(exc_class,exc_object,exc_traceback)=sys.exc_info()

alert("""internalandcompletelyunexpectedproblem,

manifestedas%s"""%str(exc_class))

举例来说,如果process()的实现产生了一个导致ValueError而不是IOError的错误,上面最后一个处理程序至少会将ValueError作为类名报告上来。

在捕获时还常有另一种含糊不清之处。其代码像这样:

错误处理中过宽的作用域

try:

first_operation()

second_operation()

third_operation()

fourth_operation()

except:

alert("somethingwentwrong")

这里的不足之处在于“横向”的不精确;当出现somethingwentwrong时,没有能立即与引发错误的特定*_operation()连接。一种简单的解决方案是一次只捕获一段,这样前面的编码就变成:

清单6.错误处理中更高的精确度

#Thedocumentationassuresusthese

#twocan'ttossexceptions.

first_operation()

second_operation()

try:

third_operation()

except:

alert("somethingwentwrongin'third_operation()'")

#This,also,cannotthrowanexception.

fourth_operation()

稍微复杂一点的解决方案是让捕获代码的范围更宽一些,但要使用语言的内省能力来报告追溯信息:

清单7.许多语言能管理自己的追溯

try:

first_operation()

second_operation()

third_operation()

fourth_operation()

except:

exc_traceback=sys.exc_info()[2]

stack_list=[]

while1:

stack_list.append(exc_traceback.tb_frame.f_code.co_name)

ifnotexc_traceback.tb_next:

break

exc_traceback=exc_traceback.tb_next

#Thenextisanalmost-human-readable

#descriptionofwherethefaultoccurred

alert("somethingwentwrongin%s"%stack_list)

于是,在捕获异常时,确信获取了全部异常且其处理方式是精确的,并使用您所选择的语言中的可用信息合成出有用的输出。

抛出

同异常的使用相比,生成异常是一个稍许高级的主题。但是,所有服务器诊所的读者都应知道抛出异常的基础:

记录接口。

保持简单的继承层次结构。

继承是由语言支持的用于异常值的出色技术;按IOError、ValueError、AppError等类别组织错误是一种相当有用的方法。然而,没有经验的设计人员常将他们的继承层次结构复杂化,以至于同最优的层次结构相差甚大。如果您发现在异常类中定义了两层以上的继承级别,那么就要检查一下。如果有三层以上,或者在一个异常超类中子类的个数超过了七个,那我打赌一定有什么地方出错了。

在这方面经常犯的一个错误是在Exception或者Error下复制了一棵应用程序对象树。这几乎总是是个错误,但是可以通过将异常参数化而不是子类化来简单地修正这个错误。一个指定了MercuryException和VenusException等的设计可能不是最好的;而用PlanetException进行编码,并附有数据指出哪一个planet是问题所在的设计可能会更好。

有一种稍许微妙的风格,可支持契约式设计(design-by-contact,DbC)方法,即在AssertionError之外

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值