- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
在为我的项目决定异常处理结构时,我一直在概念上陷入困境。
假设你有,例如:
public abstract class Data {
public abstract String read();
}
还有两个子类 FileData,它从某个指定的文件中读取您的数据,以及 StaticData,它只返回一些预定义的常量数据。
现在,在读取文件时,可能会在 FileData 中抛出 IOException,但 StaticData 永远不会抛出。大多数风格指南建议将异常传播到调用堆栈中,直到有足够数量的上下文可用于有效地处理它。
但我真的不想在抽象的 read() 方法中添加 throws 子句。为什么?因为数据和使用它的复杂机器对文件一无所知,它只知道数据。此外,可能还有其他数据子类(以及更多子类)从不抛出异常并完美地传送数据。
另一方面,IOException 是必要的,因为如果磁盘不可读(或类似的),则必须 抛出错误。所以我看到的唯一出路是捕获 IOException 并在其位置抛出一些 RuntimeException。
这是正确的理念吗?
最佳答案
你是对的。
异常应该在使用的抽象层次上。这就是为什么自 java 1.4 Throwable 开始支持异常链的原因。对于使用数据库的服务或与“存储”无关的服务,没有必要抛出 FileNotFoundException。
可能是这样的:
public abstract class Data {
public abstract String read() throws DataUnavailableException;
}
class DataFile extends Data {
public String read() throws DataUnavailableException {
if( !this.file.exits() ) {
throw new DataUnavailableException( "Cannot read from ", file );
}
try {
....
} catch( IOException ioe ) {
throw new DataUnavailableException( ioe );
} finally {
...
}
}
class DataMemory extends Data {
public String read() {
// Everything is performed in memory. No exception expected.
}
}
class DataWebService extends Data {
public string read() throws DataUnavailableException {
// connect to some internet service
try {
...
} catch( UnknownHostException uhe ) {
throw new DataUnavailableException( uhe );
}
}
}
请记住,如果您在编程时考虑到继承,则应针对特定场景仔细设计并使用这些场景测试实现。显然,编写一个通用库会更难,因为你不知道它将如何被使用。但大多数时候,应用程序仅限于特定领域。
您的新异常应该是运行时异常还是已检查异常?视情况而定,一般规则是针对编程错误抛出运行时并检查可恢复条件。
如果可以通过正确编程避免异常(例如 NullPointerException 或 IndexOutOfBounds),请使用运行时
如果异常是由于程序员无法控制的某些外部资源造成的(例如网络中断)并且可以做一些事情(显示一条 5 分钟后重试的消息或其他),那么检查异常应该使用。
如果异常超出了程序员的控制范围,但又无能为力,则可以使用 RuntimeException。例如,你应该写入一个文件,但文件已被删除,你无法重新创建它或重新尝试,那么程序应该会失败(你无能为力)很可能是运行时。
请参阅 Effective Java 中的这两项:
希望对您有所帮助。
关于Java 风格 : Properly handling exceptions,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/425281/
我想使用foreach 来等待线程终止。但是,出现以下错误,没有实现。请告诉我。 cannot move out of `*handle` which is behind a shared refer
如果在TypoScript中未配置给定的typeNum,则TYPO3将抛出Exception/CMS/1294587217。 背景:从另一个系统迁移到TYPO3后,我们遇到了许多此类异常,因为在那里使
我需要一个带有 2 个 handle 的 slider ,一个可拖动,另一个固定。我正在使用 Jquery UI slider 。这是我到目前为止尝试过的:http://jsfiddle.net/8K
给定文件的HANDLE(例如C:\\FolderA\\file.txt),我想要一个函数,该函数会将HANDLE返回到包含的目录(在前面的示例中,它将是C:\\FolderA的HANDLE)。例如:
我想通过Automic在Unix中检查文件。如果该文件不存在,则应切换主机并检查文件是否存在。 问题是,我现在不执行错误处理。 每当脚本对象正在处理并且找不到文件时,skript都会中止。我在skri
鉴于: fruitid('Apple', 'Granny Smith', 1). fruitid('Apple', 'Cox', 2). fruitid('Pear', 'Bartlett', 3).
我有一个基于Spring的Wicket应用程序。 有一个池化的数据源bean。 现在,当MySQL死了时,我得到了带有堆栈跟踪的默认Wicket错误页面。 我想处理这种情况,只允许某些页面完全显示(静
我希望能够一次查询多个句柄,其中表格具有相同的格式,例如: 句柄:8000,8001,8003表:foo 想要做这样的事情: x:hopen `8000`8001`8003 x select from
我对在Swift 3中引发自定义异常有些困惑。 在C++中,我可以执行此操作以立即停止方法中的进程,抛出错误并进行处理,而无需进一步进行操作。 void foo() { try {
我一直在阅读MSDN开发人员COM指南。但是this page上的代码令人困惑。在此处复制: The following code sample shows the recommended way o
我有一个计划的批处理文件每天都会启动的过程。如果有错误,我需要内置错误处理才能重启进程。所有这些在大多数情况下都有效,但是我每个月都会收到一次超时错误,所以这是不可避免的。该进程不会将错误级别输出到b
我正在尝试从 chartlyrics API 获取歌词。我编写了一个可以运行但不能在循环内运行的 R 函数。我的脚本是: library(httr) library(RCurl) library(XM
在libuv事件循环中调用prepare handle callback和check handle callback的原因是什么? 最佳答案 I/O 操作发生在这两者之间,因此您可能希望在阻塞 I/O
我正在尝试在 R 中安装 BTYplus 包。 devtools::install_github("mplatzer/BTYDplus", dependencies=TRUE) library(BTY
我有一个Arduino,可以使用pySerialTransfer库通过串行与Mac正常通信,并且可以运行数小时。然后是某种形式的串行中断-尽管一夜间发生时我一直无法确定原因,但只要从笔记本电脑上拔下A
我是hooks和async/await的新手。我正在尝试处理Axios调用中的错误,并且不确定如何使用then/catch或try/catch处理我的API调用中的错误。 在基于类的React中,我将
我正在尝试向脚本中添加一些内容,以便让我知道我复制的文件是否已被完全复制。 基本上,我要压缩一堆文件,然后将它们发送到网络上的映射驱动器。然后,一旦文件被成功复制,我将脚本删除原始位置的文件。该脚本可
我有一个圆形 slider ,其中绘制了一条贝塞尔弧,一个圆弧在 slider 的起点和终点有两个 handle ,圆弧是在圆形 slider 中绘制的。 借助开始和结束 handle ,我可以沿着圆
删除 NULL 指针是安全的。 int* p = NULL; delete p; // ok, secure 句柄是什么? HANDLE h = NULL; CloseHandle(h
如果您没有在 dojo.connect 期间返回的“句柄”,您如何删除 dojo 连接事件? 我的示例涉及将一组事件动态分配给一组对象。 (为简单起见,事件是 onclick 和 ondblclick
我是一名优秀的程序员,十分优秀!