译者注:原文出处http://danielwestheide.com/blog/2012/12/26/the-neophytes-guide-to-scala-part-6-error-handling-with-try.html,翻译:Thomas
在新语言的学习阶段,你通常不会去想如果执行代码出了问题该如何处理。一旦你想要开发一个真实产品时,就必须认真面对错误和异常处理了。由于各种个语言对这方面支持程度的重要性有时候被认人忽视了。
Scala在设计之初就考虑了如何优雅的应对错误场景。在本篇中,我会介绍Scala以Try类型为基础的错误处理机制及内在原理。后面会用到Scala2.10才有的一些特性,2.9.3里也移植了这些特性,所以请确保你的实验环境是在2.9.3或以上版本。
抛出和捕捉异常
在介绍Scala错误处理的惯用方式之前,我们先来看看你在其它语言如Java或Ruby中的错误的方式。和这些语言一样,Scala里你也可以抛出一个异常:
case class Customer(age: Int) class Cigarettes case class UnderAgeException(message: String) extends Exception(message) def buyCigarettes(customer: Customer): Cigarettes = if (customer.age < 16) throw UnderAgeException(s"Customer must be older than 16 but was ${customer.age}") else new Cigarettes
抛出的异常可以用类似Java的方式来捕捉和处理,不一样的是,你定义一个偏函数来进行异常处理。另外,在Scala里try/catch是一个表达式,所以下面的代码返回异常里的信息:
val youngCustomer = Customer(15) try { buyCigarettes(youngCustomer) "Yo, here are your cancer sticks! Happy smokin'!" } catch { case UnderAgeException(msg) => msg }
函数式的错误处理
在你的代码中处处出现这种“传统”的错误处理代码会让人觉得很难看,并且这也不算是函数式编程。并且当程序有很多并发代码时,这也不是个好的方法,比如说你就没法捕捉到执行在另一个线程的一个Actor抛出的异常 - 你可能希望收到一个表示异常/错误情况的消息。
在Scala里,首选的做法是从函数返回一个适当的值来表示发生了一个错误,不必担心,我们并没有回到C语言风格的错误处理,在C语言里,函数和调用方通过约定的错误代码来传递异常。在Scala里我们用一个特别的类型来表示一个可能会发生异常的计算。
在本篇,我们专注于从Scala2.10(2.9.3里也引进了)引进来的类型,Try。有一个叫Either的类似类型也挺有用,会在下一篇介绍到。
Try的含义
解释Try的含义的最好方法是拿Option类型的含义来参考。Option[A]表示一个A类型的值可能存在或不存在,那么Try[A]表示的是一段代码当成功执行时返回A类型的值,或者当错误发生时返回一个Throwable。这种包含可能存在错误的类型的实例可以很容易的在你程序中各个并发代码中互相传递。
Try有两个形态:如果一个Try[A]实例表示计算成功,它会是一个Success[A]的实例,包含着A类型的值,另一个形态是当代码出错时,它是一个Failure[A]的实例,包含一个Throwable,表示一个异常或其它类型的错误。
如果我们确切的知道一段代码可能会出错,我们可以让这个函数返回Try[A]。这显式化的表达了可能的错误并且强制函数的调用者处理这种可能的错误。
举个例子,我们要写一个简单地网页抓取器,用户可以提供要抓取的URL地址。我们应用少不了这样的功能:解析用户输入的URL并据此生成java.net.URL实例:
import scala.util.Try import java.net.URL def parseURL(url: String): Try[URL] = Try(new URL(url))
如你所见,我们返回的是Try[URL]类型。如果提供的url在语法上是个URL,那么返回的是
Success[URL],如果URL构造器抛出MalformedURLException,返回类型将是Failure[URL]。
我们用了Try的伙伴对象的apply工厂方法,这方法输入一个by-name方式的URL类型的参数。在我们的例子中,new URL(url)
只在Try的apply方法里执行,这方法里执行的代码抛出的任何non-fatal异常都会被捕捉并封装在Failure 类型中返回。
因此,parseURL("http://danielwestheide.com")
会返回包含相应URL实例的Success[URL]
,而parseURL("garbage")会返回包含
MalformedURLException的Failure[URL]。
使用Try值
使用Try实例和使用Option类型非常相似。因此这里就没有太多惊喜了。
你可以呼叫isSuccess方法来判断Try是否成功,然后如果成功就调用它的get方法(这和上篇的Option的isDefined相似)。但是相信我,没有太多情况让你想要这样做。
也可以用getOrElse来传递一个默认值,如果Try是一个Failure时将返回这默认值:
val url = parseURL(Console.readLine("URL: ")) getOrElse new URL("http://duckduckgo.com")
如果用户提供的URL格式有错误,我们就用后备的URL,DuckDuckGo。
串联操作
Try类型的一个最重要的特性就是,和Option一样,它也支持你在所有集合类型里看到的高阶方法。在下面的例子中你会看到,这让你以一种可读性很强的方式串联很多Try并且捕获其中任何的异常。
Map和flat map
把一个为Success[A]
的Try[A] map到
Try[B]的结果
是Success[B]
实例。
如果实例为Failure[A]
, 那么Try[B]的map结果就是
Failure[B]
, 也就是把原来Failure[A]里的异常传递下来
:
parseURL("http://danielwestheide.com").map(_.getProtocol) // results in Success("http") parseURL("garbage").map(_.getProtocol) // results in Failure(java.net.MalformedURLException: no protocol: garbage)
如果你串联多个map操作,结果就会变成嵌套的
Try
结构,者往往不是你想要的。考虑下面的方法,该方法从给定的URL中获取内容的InputStream:
import java.io.InputStream def inputStreamForURL(url: String): Try[Try[Try[InputStream]]] = parseURL(url).map { u => Try(u.openConnection()).map(conn => Try(conn.getInputStream)) }因为传递给两次
map方法的匿名函数返回的是
Try
, 所以最终返回类型就是Try[Try[Try[InputStream]]]。这时候就该让
flatMap派上用场了,Try[A]的flatMap的参数为一个函数,这函数以A类型作为参数,输出Try[B]类型。如果Try[A]为一个Failure[A],那么将返回Failure[B],如果Try[A]是一个Success[A],flatMap则会解包出A并且返回一个Try[B]。
这意味着我们通过串联任意多个flatMap构造一个管道来进行一系列的计算,这些计算需要封装在Success中得值,管道中的任何一个操作丢出封装在Failure中的异常,就意味着最终结果会是Failure。我们用flatMap来重写一下inputStreamForURL方法:
def inputStreamForURL(url: String): Try[InputStream] = parseURL(url).flatMap { u => Try(u.openConnection()).flatMap(conn => Try(conn.getInputStream)) }现在我们会得到
Try[InputStream]的结果,如果任意一个flatMap中得方法失败,结果是封装在Failure中的异常
, 反之是封装在Success中得InputStream。
过滤和foreach
你还可以过滤一个Try或者呼叫它的foreach方法。这两个的用法和你在前面Option中学到的基本一样。
当Try是一个Failure或过滤方法返回false时,filter方法返回一个Failure。如果呼叫一个Success的filter方法并且过滤方法返回true,那么原始的Success会被返回:
def parseHttpURL(url: String) = parseURL(url).filter(_.getProtocol == "http") parseHttpURL("http://apache.openmirror.de") // results in a Success[URL] parseHttpURL("ftp://mirror.netcologne.de/apache.org") // results in a Failure[URL]仅当Try是Success时,传递给foreach方法的函数才被调用,foreach方法让你执行一个副作用。Success所包含的值被传递给foreach的函数参数,该函数被执行一次:
parseHttpURL("http://danielwestheide.com").foreach(println)
For语句
因为Try有提供flatMap,map以及filter的方法,那就意味着你可以用For语句的写法来串联Try实例的操作,For语句通常提供更好的可读性。我们用for语法来实现一个给定URL的网页内容的方法:
import scala.io.Source def getURLContent(url: String): Try[Iterator[String]] = for { url <- parseURL(url) connection <- Try(url.openConnection()) is <- Try(connection.getInputStream) source = Source.fromInputStream(is) } yield source.getLines()
有三个地方可能会发生异常,所有这些地方都用Try来加以保护。首先是先前实现的parseURL方法,它返回Try[URL],只有当它返回Success[URL],我们才去尝试打开一个连接并构造一个新的inputStream。如果打开连接和构造inputStream都成功了,我们继续进行,最终产生网页的所有行。用这样高效的方式调用flatMap,最终结果会是Try[Iterator[String]]。
请注意上面的代码有两处可改善的,首先整个功能可以简化成只调用Source.fromURL即可,另外我们在程序结束时没有关闭所构造的inputStream,我还是这样写代码是想把焦点放在我们应该关注的方面,即Try。
模式匹配
在你的代码里,你可能经常需要知道从一个函数返回的Try实例代表着成功还是失败,并据此执行不同的代码分支,这时候你需要模式匹配,因为Success和Failure都是case class,所以这很容易实现。
我们想要把网页的内容显示出来,或者如果中间出现异常就打印一个错误信息:
import scala.util.Success import scala.util.Failure getURLContent("http://danielwestheide.com/foobar") match { case Success(lines) => lines.foreach(println) case Failure(ex) => println(s"Problem rendering URL content: ${ex.getMessage}") }如果你想要为Failure的场景设定一个默认行为,无需使用getOrElse,替代的方式是用recover方法,该方法需要传入一个偏函数,并返回另一个Try,如果recover被Success实例呼叫,拿它直接返回该实例,否则,返回包在Success里的偏函数的返回值。
import java.net.MalformedURLException import java.io.FileNotFoundException val content = getURLContent("garbage") recover { case e: FileNotFoundException => Iterator("Requested page does not exist") case e: MalformedURLException => Iterator("Please make sure to enter a valid URL") case _ => Iterator("An unexpected error has occurred. We are so sorry!") }现在我们可以放心的对content调用它的get方法以获取 Try[Iterator[String]]里包着的结果了。调用content.get.foreach(println)将会输出“Please make sure to enter a valid URL”。
结论
Scala里的惯用的错误处理机制和其它的语言如Java或Ruby差别很大。Try类型让你可以封装代码计算结果的出错并且可以以一种优雅的方式将计算结果作串联操作。你可以像用集合或Option一样的方式来进行错误处理。
限于篇幅,我没有介绍Try的所有方法。像Option一样,Try也有orElse方法。transform和recoveryWith方法也值得一看,我建议你去查看Scala的文档并试着写写代码。
在下一篇,我们会接触到Either,她是封装代码异常的另一种方案,不过除了进行错误处理,它还用在其它场景。
相关推荐
中国人工智能产业发展联盟金融大模型落地路线图研究报告2024年56页.pdf
USB运动控制开源系统揭秘:五轴雕刻机核心技术全开源,支持RTCP算法,PCB生产便捷,C++源码可复制,USB运动控制五轴雕刻机系统完全开源资料,含PCB生产支持及多版本C++源码,USB运动控制 (五轴雕刻机系统)全部开源 不保留任何关键技术,PCB可直接生产,C++6.0源码,从13.7-18.2所有版本,本产品为可复制资料,支持五轴联动,支持RTCP算法,全部开源。 1、为电子资料 2、PCB底板+原理图+源码 ,核心关键词:USB运动控制; 五轴雕刻机系统; 开源技术; 不保留关键技术; C++6.0源码; 版本范围(13.7-18.2); 可复制资料; 五轴联动; RTCP算法; PCB底板; 原理图。,开源五轴雕刻机系统:USB运动控制全解析
系统选用B/S模式,后端应用springboot框架,前端应用vue框架, MySQL为后台数据库。 本系统基于java设计的各项功能,数据库服务器端采用了Mysql作为后台数据库,使Web与数据库紧密联系起来。 在设计过程中,充分保证了系统代码的良好可读性、实用性、易扩展性、通用性、便于后期维护、操作方便以及页面简洁等特点。
基于16QAM的SIMULINK与MATLAB联合仿真系统:调制解调波形分析与应用拓展,基于MATLAB和SIMULINK平台的16QAM调制与解调仿真研究及波形分析,16QAM SIMULINK 基于SIMULINK和MATLAB的16QAM调制和解调。 采用SIMULINK搭建框图,MATLAB调用模型得出波形图。 (可自行简单修改在SIMULINK中加scope,无须MATLAB调用) ,核心关键词: 16QAM; SIMULINK; MATLAB; 调制; 解调; 波形图; 框图; Scope,基于SIMULINK的16QAM调制解调系统研究
基于PMSM模型的四种控制策略对比研究:传统滑膜控制与扰动观测器的优化与应用,基于滑膜控制扰动观测器的PMSM模型:四控制策略对比分析与实践应用研究 [附带视频与出图程序],基于滑膜控制扰动观测器的永磁同步电机PMSM模型 四个控制对比: 1、PID控制器 2、传统滑模控制器 3、最优滑模控制器 4、改进补偿滑膜控制器 [1]附带简单讲解视频 如下图 [2]附带出图程序,四个控制对比的说明文档(2篇,非次品) ,核心关键词:滑膜控制; 扰动观测器; 永磁同步电机PMSM模型; PID控制器; 传统滑模控制器; 最优滑模控制器; 改进补偿滑膜控制器; 简单讲解视频; 出图程序; 对比说明文档。,PMSM模型下的滑膜控制:四法比拼,解析与可视化
Abaqus USDFLD子程序:实现积分点间材料弹性连续变化仿真的高效方法,Abaqus USDFLD子程序:实现积分点间材料弹性连续变化仿真的高效方法,Abaqus USDFLD子程序实现积分点间材料弹性连续变化仿真 ,Abaqus; USDFLD子程序; 积分点; 材料弹性; 连续变化仿真;,Abaqus USDFLD实现材料弹性连续变化仿真
内容概要:本文档为《早中期复习—数字信号处理》的学习指南,详细介绍了数字信号处理的相关概念和方法,旨在梳理并巩固相关领域的知识点。文档内容涵盖数字信号处理基本概念及时域离散信号和系统的分析方法;重点探讨时域离散信号、离散傅里叶变换及其快速算法(FFT);详细介绍了基于离散信号变换方法的不同类型滤波器的设计;此外还列举了部分经典的面试题目及其解答方向,以辅助备考者准备面试。文档有助于深入理解和掌握这一学科,提高对信号分析技能的认知和应用。 适合人群:本指南主要面向正在备战考试或从事相关工作的初学者,尤其是需要系统性复习并加强理论理解和实际操作技巧的学生和工程师。 使用场景及目标:可用于准备研究生入学面试或者作为工程师日常工作中处理复杂工程问题时的参考手册。目标是帮助使用者加深对数字信号处理的认识,掌握关键技术和应用场景,以便更好地应对学术和工业挑战。 其他说明:文档结构清晰、条理性强,配合大量例题和图示,有利于读者理解和记忆。同时,提供了实用的小贴士和思考题,引导读者积极思考,拓展视野,培养独立解决问题的能力。
题目2.5(模拟浏览器操作程序):标准Web浏览器具有在最近访问的网页间后退和前进的功能。实现这些功能的个方法是:使用两个栈,追踪可以后退和前进而能够到达的网页。
SensorTower2024年AI应用市场洞察报告31页.pdf
chromedriver-win32-136.0.7055.0.zip
COMSOL热流耦合拓扑优化:最大化放热量与功率耗散策略解析,Comsol热流耦合拓扑优化技术:以最大化放热量与功率耗散为目标函数的优化策略,Comsol热流耦合拓扑优化。 目标函数采用最大化放热量和功率耗散。 ,Comsol;热流耦合;拓扑优化;目标函数;最大化放热量;功率耗散,Comsol热流耦合优化:最大化放热与功率耗散
内容概要:本文介绍了将假肢测试与实时混合子结构(RTHS)方法相结合的技术背景。RTHS方法用于将完整的动态系统分解为数值部分(numerical part)和实验部分(experimental part),并在Simulink中进行建模。数值部分包括模拟截肢者的模型,而实验部分则涉及真实的机械臂和假肢。两者通过传输系统耦合,实现了步行阶段的动态交互。文章具体描述了不同步态阶段的动力学模拟流程,包括飞行阶段(抬脚离地)和接触阶段(脚触地)。为了实现有效的仿线,提出了对机械臂的四个关键要求:能够执行接口运动、承受界面力、低延迟高精度以及实现实时通信。 适合人群:从事生物力学、医疗器械和机器人技术研究的专业人士及科研人员。 使用场景及目标:适用于需要对假肢进行动态性能测试的研发机构或企业,目标是选择合适的机械臂并构建完整的假肢测试平台,提高仿线的准确性和可靠性。 阅读建议:重点理解和掌握RTHS方法的工作原理以及机械臂在仿真实验中的角色,在实践中注意验证机械臂是否符合所列出的各项要求。
FLUENT与MATLAB协同:基于UDP的复杂数据联合仿真计算与交互处理方案,FLUENT与MATLAB协同:基于UDP的复杂数据联合仿真处理系统,FLUENT与MATLAB联合仿真计算,基于UDP,可在MATLAB实现复杂数据计算处理。 提供两个软件数据交互方法和接口,FLUENT数据传递给MATLAB后,可以用任意方法处理,最后再回传给FLUENT处理后的数据。 本案例只是简单演示效果,可以实现复杂功能。 ,联合仿真计算; UDP接口通信; 数据处理; 交互方法; 回传数据; 复杂功能演示。,FLUENT与MATLAB协同:UDP接口数据交互与复杂处理
postgresql安装教程.md
IPMSM数学模型深度解析:双环模拟技术,预测电机对多样输入的响应,精准输出电流、转速与转矩,IPMSM模型分析电机响应,IPMSM数学模型,模拟电机对不同输入的响应,包含速度环和电流环,输出电流转速和转矩。 ,IPMSM数学模型; 电机响应模拟; 速度环和电流环; 输出电流转速和转矩; 电机控制,IPMSM模型模拟电机响应:双环控制下电流转速与转矩输出
基于CNN-RBF神经网络的优化数据分类预测模型——以交叉验证防止过拟合的Matlab代码实现,Matlab结合CNN-RBF进行数据分类优化,基于卷积神经网络结合径向基函数神经网络(CNN-RBF)的数据分类预测 CNN-RBF数据分类 优化参数为扩散速度,采用交叉验证防止过拟合 matlab代码 注:要求 Matlab 2019A 及以上版本 ,核心关键词: 卷积神经网络(CNN); 径向基函数神经网络(RBF); 数据分类预测; 优化参数; 扩散速度; 交叉验证; 过拟合; MATLAB代码 2019A以上版本,基于CNN-RBF的优化参数数据分类预测Matlab代码实现
多变量模式分析在脑电数据中的深度应用:从磁共振到时频域的神经表征研究,多变量模式分析在脑电数据中的深度应用:从磁共振到时频域的神经表征研究,多变量模式分析最早应用在磁共振数据中,用来考察某些脑区在编码不同条件的刺激时是否存在表征上的显著不同。 后来逐渐运用到脑电数据中,虽然脑电数据的空间分辨率较低,但时间分辨率很高,因此可以帮助确定在哪一段时间内,个体对不同条件刺激的表征有显著差异。 目前已经有很多工具箱支持脑电数据MVPA的分析,例如matlab中ADAM,python中的NeuroRA等(这两个相对来说比较好上手)。 方法共包括基础的时间序列解码,以及衍生方法跨时域解码与权重投射等。 MVPA不仅可以应用在原始时域数据上,也可以应用在时频域数据上,来观察不同频段的能量对于编码不同刺激过程中的贡献。 ,多变量模式分析(MVPA);磁共振数据;脑电数据;时间分辨率;工具箱支持;ADAM;NeuroRA;时间序列解码;跨时域解码;权重投射;时频域分析。,多变量模式分析在脑电数据中的应用:从磁共振到时频域的表征研究
更多毕业设计https://cv2022.blog.csdn.net/article/details/124463185
系统选用B/S模式,后端应用springboot框架,前端应用vue框架, MySQL为后台数据库。 本系统基于java设计的各项功能,数据库服务器端采用了Mysql作为后台数据库,使Web与数据库紧密联系起来。 在设计过程中,充分保证了系统代码的良好可读性、实用性、易扩展性、通用性、便于后期维护、操作方便以及页面简洁等特点。
内容概要:本文档详细介绍了基于单片机的点阵电子显示屏的设计与实现项目。项目涵盖从理论背景、项目目标、硬件设计、软件开发到最后的应用与未来扩展等多个方面。首先阐述了点阵显示屏的广泛应用和技术背景,接着定义了项目的具体目标,包括实现文字滚动、图像展示等功能,同时深入解析了硬件部分的选型和连接细节,如单片机(STM32)、LED点阵模块及其驱动电路。在软件部分,文中演示了I2C通信协议的代码示例,并展示了如何通过嵌入式编程控制显示屏的效果,如显示固定字符、实现文字滚动等。此外,文章还讨论了调试和优化的方法论及注意事项,最后对未来发展方向提出了展望,如多显示屏协同、触摸屏输入的支持等。 适合人群:从事嵌入式系统开发的专业技术人员、电子工程专业的大学生及研究人员。 使用场景及目标:①嵌入式系统开发课程的案例研究;②工业设计公司或科研单位的项目参考;③高校实验室开展相关课题的研究资料。 阅读建议:对于初次接触嵌入式开发的新手,建议先掌握基础知识,熟悉所使用的单片机型及其外设接口功能后再开始本项目的学习;而对于有经验的研发人员,则可以重点关注具体的实现技术和遇到问题时的解决方案部分,结合自身需求进行灵活运用。