FunDA(17)- 示范:异常处理与事后处理 - Exceptions handling and Finalizers
作为一个能安全运行的工具库,为了保证占用资源的安全性,对异常处理(exception handling)和事后处理(final clean-up)的支持是不可或缺的。FunDA的数据流FDAPipeLine一般是通过读取数据库数据形成数据源开始的。为了保证每个数据源都能被安全的使用,FunDA提供了事后处理finalizing程序接口来实现数据流使用完毕后的清理及异常处理(error-handling)程序接口来捕获和处理使用过程中出现的异常情况。首先,事后处理程序(finalizer)保证了在任何情况下的FunDA数据流终止运算:包括元素耗尽,强制中断以及异常中断,finalizer都会被调用。在这篇讨论里我们将会测试和示范FunDA Exception-handling和Final-cleanup。下面的样板代码设定了一个静态集合数据源viewState和一个动态数据流streamState:
val db = Database.forConfig("h2db")
implicit def toState(row: StateTable#TableElementType) =
StateModel(row.id,row.name)
val viewLoader = FDAViewLoader(slick.jdbc.H2Profile)(toState _)
val streamLoader = FDAStreamLoader(slick.jdbc.H2Profile)(toState _)
val stateSeq = viewLoader.fda_typedRows(StateQuery.result)(db).toSeq
val viewState = fda_staticSource(stateSeq)(println("***Finally*** the end of viewState!!!"))
val streamState = streamLoader.fda_typedStream(StateQuery.result)(db)(,)(println("***Finally*** the end of streamState!!!"))
在上面的代码例子里我们可以看到fda_staticSource和fad_typedStream都挂接了事后处理程序,我们简单的用println代表一段完整的程序来证实对事后处理程序的调用。所以说事后处理程序的挂接是在构建view或者stream时进行的。我们先看看它们在正常终止或者强行中断是是否发生调用:
viewState.startRun
viewState.take().startRun
streamState.startRun
streamState.take().startRun
// ***Finally*** the end of viewState!!!
// ***Finally*** the end of viewState!!!
// ***Finally*** the end of streamState!!!
// ***Finally*** the end of streamState!!!
那么如果在出现了异常中断是是否同样会被调用呢?我们先设计下面两个用户自定义函数:
def trackRows: FDAUserTask[FDAROW] = row => {
row match {
case m@StateModel(id,name) =>
println(s"State: $id $name")
println( "----------------")
fda_next(m)
case m@_ => fda_next(m)
}
}
def errorRow: FDAUserTask[FDAROW] = row => {
row match {
case StateModel(id,name) =>
val idx = id / (id - )
fda_next(StateModel(idx,name))
case m@_ => fda_next(m)
}
}
trackRows跟踪显示当前数据行,errorRow人为的会在第三行出现异常。我们用streamState来测试一下:
streamState.appendTask(errorRow).appendTask(trackRows).startRun
// State: 0 Alabama
// ----------------
// State: -2 Alaska
// ----------------
// Exception in thread "main" java.lang.ArithmeticException: / by zero
// at examples.ExceptionsAndFinalizers$$anonfun$errorRow$1.apply(ExceptionsAndFinalizers.scala:46)
// ...
// at java.lang.Thread.run(Thread.java:745)
// ***Finally*** the end of streamState!!!
的确在正常显示了两行数据后,第三行出错中断,直接调用了finalizer。这就保证了无论发生任何情况,当完成使用数据源后都给予编程人员一个空间去进行事后处理如释放资源、中断连接、关闭文件等。
我们可以用onError来挂接异常处理程序,如下:
val s = streamState.appendTask(errorRow).appendTask(trackRows)
val s1 = s.onError {case e: Exception => println(s"Caught Error in streamState!!![${e.getMessage}]"); fda_appendRow(FDANullRow)}
注意:onError必须挂接在stream的最尾端以确保所有环节的异常情况都可以正确地得到处理。看看运行结果:
State: Alabama
----------------
State: - Alaska
----------------
***Finally*** the end of streamState!!!
Caught Error in streamState!!![/ by zero]
以上例子捕获了异常情况,同时在异常中断情况后还是调用了finalizer。
有时我们需要自定义一些特殊情况,我们希望能捕获这些情况的发生。但我们同时希望这些情况发生时不会中断运算。首先我们可以先自定义一个异常行类型:
case class DivideZeroError(msg: String, e: Exception) extends FDAROW
注意:切不可忘记extends FDAROW。我们把上面的errorRow函数修改成一个自捕获异常的函数:
def catchError: FDAUserTask[FDAROW] = row => {
row match {
case StateModel(id,name) =>
try {
val idx = id / (id - )
fda_next(StateModel(idx, name))
} catch {
case e: Exception => //pass an error row
fda_next(DivideZeroError(s"Divide by zero excption at ${id}",e))
}
case m@_ => fda_next(m)
}
}
必须修改trackRows能分辨DivideZeroError行:
def trackRows: FDAUserTask[FDAROW] = row => {
row match {
case m@StateModel(id,name) =>
println(s"State: $id $name")
println( "----------------")
fda_next(m)
case DivideZeroError(msg, e) => //error row
println(s"***Error:$msg***")
fda_skip
case m@_ => fda_next(m)
}
}
运算下面的程序:
val s = streamState.take().appendTask(catchError).appendTask(trackRows)
val s1 = s.onError {case e: Exception => println(s"Caught Error in streamState!!![${e.getMessage}]"); fda_appendRow(FDANullRow)}
s1.startRun
产生下面的结果:
State: Alabama
----------------
State: - Alaska
----------------
***Error:Divide by zero excption at ***
State: Arkansas
----------------
State: California
----------------
***Finally*** the end of streamState!!! Process finished with exit code
没有出现异常中断,捕获并处理了自定义异常,并且调用了事后处理程序finalizer。
下面就是这次示范的源代码:
import slick.jdbc.H2Profile.api._
import com.bayakala.funda.samples.SlickModels._
import com.bayakala.funda._
import api._
import scala.language.implicitConversions object ExceptionsAndFinalizers extends App { val db = Database.forConfig("h2db")
implicit def toState(row: StateTable#TableElementType) =
StateModel(row.id,row.name)
val viewLoader = FDAViewLoader(slick.jdbc.H2Profile)(toState _)
val streamLoader = FDAStreamLoader(slick.jdbc.H2Profile)(toState _) val stateSeq = viewLoader.fda_typedRows(StateQuery.result)(db).toSeq
val viewState = fda_staticSource(stateSeq)(println("***Finally*** the end of viewState!!!"))
val streamState = streamLoader.fda_typedStream(StateQuery.result)(db)(,)(println("***Finally*** the end of streamState!!!")) /*
viewState.startRun
viewState.take(2).startRun
streamState.startRun
streamState.take(3).startRun
// ***Finally*** the end of viewState!!!
// ***Finally*** the end of viewState!!!
// ***Finally*** the end of streamState!!!
// ***Finally*** the end of streamState!!!
*/ def trackRows: FDAUserTask[FDAROW] = row => {
row match {
case m@StateModel(id,name) =>
println(s"State: $id $name")
println( "----------------")
fda_next(m)
case DivideZeroError(msg, e) => //error row
println(s"***Error:$msg***")
fda_skip
case m@_ => fda_next(m)
}
} def errorRow: FDAUserTask[FDAROW] = row => {
row match {
case StateModel(id,name) =>
val idx = id / (id - )
fda_next(StateModel(idx,name))
case m@_ => fda_next(m)
}
} case class DivideZeroError(msg: String, e: Exception) extends FDAROW
def catchError: FDAUserTask[FDAROW] = row => {
row match {
case StateModel(id,name) =>
try {
val idx = id / (id - )
fda_next(StateModel(idx, name))
} catch {
case e: Exception => //pass an error row
fda_next(DivideZeroError(s"Divide by zero excption at ${id}",e))
}
case m@_ => fda_next(m)
}
} /*
streamState.appendTask(errorRow).appendTask(trackRows).startRun
// State: 0 Alabama
// ----------------
// State: -2 Alaska
// ----------------
// Exception in thread "main" java.lang.ArithmeticException: / by zero
// at examples.ExceptionsAndFinalizers$$anonfun$errorRow$1.apply(ExceptionsAndFinalizers.scala:46)
// ...
// at java.lang.Thread.run(Thread.java:745)
// ***Finally*** the end of streamState!!!
*/
/*
val v = viewState.appendTask(errorRow).appendTask(trackRows)
val v1 = v.onError {case e: Exception => println(s"Caught Error in viewState!!![${e.getMessage}]"); fda_appendRow(FDANullRow)}
v1.startRun val s = streamState.appendTask(errorRow).appendTask(trackRows)
val s1 = s.onError {case e: Exception => println(s"Caught Error in streamState!!![${e.getMessage}]"); fda_appendRow(FDANullRow)}
s1.startRun
*/ val s = streamState.take().appendTask(catchError).appendTask(trackRows)
val s1 = s.onError {case e: Exception => println(s"Caught Error in streamState!!![${e.getMessage}]"); fda_appendRow(FDANullRow)}
s1.startRun }
FunDA(17)- 示范:异常处理与事后处理 - Exceptions handling and Finalizers的更多相关文章
- Python学习(17)异常处理
目录 Python 异常处理 Python 标准异常 异常处理 使用except而不带任何异常类型 使用except而带多种异常类型 try-finally 语句 异常参数 异常的参数 用户自定义参数 ...
- DRF之权限认证,过滤分页,异常处理
1. 认证Authentication 在配置文件中配置全局默认的认证方案 REST_FRAMEWORK = { 'DEFAULT_AUTHENTICATION_CLASSES': ( 'rest_f ...
- DRF的异常处理
默认情况下,DRF框架通过内置的 exception_handler 方法,处理了如下异常: django内置异常 Http404 PermissionDenied DRF框架异常 APIExcept ...
- drf07 过滤 排序 分页 异常处理 自动生成接口文档
4. 过滤Filtering 对于列表数据可能需要根据字段进行过滤,我们可以通过添加django-fitlter扩展来增强支持. pip install django-filter 在配置文件sett ...
- Cannot load connection class because of underlying exception: com.mysql.cj.exceptions.WrongArgumentException: Malformed database URL, failed to parse the connection string near ';characterEncoding=UTF
今天在使用springboot整合SSM的时候,配置好以后启动项目,报了一个这样的异常 java.sql.SQLNonTransientConnectionException: Cannot load ...
- django-rest-framework-源码解析002-序列化/请求模块/响应模块/异常处理模块/渲染模块/十大接口
简介 当我们使用django-rest-framework框架时, 项目必定是前后端分离的, 那么前后端进行数据交互时, 常见的数据类型就是xml和json(现在主流的是json), 这里就需要我们d ...
- MasaFramework -- 异常处理
前言 在程序设计中,我们会遇到各种各样的异常问题,一个异常处理不仅仅可以帮助开发者快速的定位问题,也可以给用户更好的使用体验,那么我们在AspNetCore项目中如何捕获以及处理异常呢? 而对应Asp ...
- (05)odoo数据库和业务操作
以一个例子开头* To-do 向导 # 配置文件 __openerp_.py: { 'name': 'To-do Tasks Management Assistant' ...
- linux中断源码分析 - 初始化(二)
本文为原创,转载请注明:http://www.cnblogs.com/tolimit/ 本篇文章主要讲述源码中是如何对中断进行一系列的初始化的. 回顾 在上一篇概述中,介绍了几个对于中断来说非常重要的 ...
随机推荐
- 【NIFI】 开发自定义Nifi Processor
本例需要基础知识:[NIFI] Apache NiFI 安装及简单的使用 Nifi不光可以使用自带的Processor,还可以自定义Processor.本例简单介绍开发一个Processor 开发 1 ...
- gcc -ldl 选项作用
如果你的程序中使用dlopen.dlsym.dlclose.dlerror 显示加载动态库,需要设置链接选项 -ldl 加载动态链接库,首先为共享库分配物理内存,然后在进程对应的页表项中建立虚拟页和物 ...
- word中括号中公式对齐
在写论文中使用到括号,但没有对齐,如: 但是我需要对齐,如: 此时只需要在每行需要对齐的地方输入‘&’即可(此符号在专业型括号是不可见的,在线性中可见,上图为专业型),同样也可以有多个对齐点, ...
- Java 数组拷贝方法 System.arraycopy
System类提供的数组拷贝方法: public static native void arraycopy(Object src, int srcPos, Object dest, int destP ...
- 2018.10.27 codeforces402D. Upgrading Array(数论+贪心)
传送门 唉我觉得这题数据范围1e5都能做啊... 居然只出了2000 考完听zxyzxyzxy说我的贪心可以卡但过了? 可能今天本来是0+10+00+10+00+10+0只是运气好T1T1T1骗了10 ...
- Java的函数重载必须满足的条件
1.函数名相同 2.参数个数不同或者参数类型不同 3.函数重载和返回值类型无关 //函数的重载 public static void get() { System.out.println(" ...
- LPCSTR与CString转换
1.LPCSTR是Win32和VC++所使用的一种字符串数据类型,L表示long,P表示指针,C表示常量,STR表示字符串. 2.LPCSTR转化为CString: LPCSTR lpStr=&qu ...
- char与CString相互转换
Char -> CStringchar ch[] = "Hello";CString str;str.Format("%s",ch);CString -& ...
- identify.class.php<======>token加密方法
class Identify { static private $cert = "1111111"; static public function writeSecret($mob ...
- dj 分页器组件
django内置的分页器组件,能够帮我们实现对查询的数据进行自动分页,并返回分页对象 from django.core.paginator import Paginator, EmptyPage Pa ...