先交代一下背景。昨天晚上一个群友在群里问我一个问题。他当时在给Oracle 10g配置C3P0连接池。
最起初呢,这家伙用的是ojdbc14.jar。写法是标准的Spring的写法,但是就是不行。后来我说,你换个包。
换成classes12.jar看看。顺利执行了。
话说,后来又出了一个问题。他问我说,执行是执行了,但是出现个新的问题。
XXXX(1234567890123456) 19:02:17
新问题又出来了
XXXX(1234567890123456) 19:02:28
NewPooledConnection - closed by a client.
java.lang.Exception: DEBUG -- CLOSE BY CLIENT STACK TRACE
at com.mchange.v2.c3p0.impl.NewPooledConnection.close(NewPooledConnection.java:566)
at com.mchange.v2.c3p0.impl.NewPooledConnection.close(NewPooledConnection.java:234)
at com.mchange.v2.c3p0.impl.C3P0PooledConnectionPool$1PooledConnectionResourcePoolManager.
destroyResource(C3P0PooledConnectionPool.java:470)
at com.mchange.v2.resourcepool.BasicResourcePool$1DestroyResourceTask.run(BasicResourcePool.
java:964)
at com.mchange.v2.async.ThreadPoolAsynchronousRunner$PoolThread.run(ThreadPoolAsynchronousRunner.
java:547)
C3P0PooledConnectionPool - Successfully destroyed PooledConnection: com.mchange.v2.c3p0.impl.
NewPool
我看到那个DEBUG,我说,是调试信息,修改一下LOG4J的等级就行了。
这个群友很不解的问,既然成功了,干嘛还要丢异常出来?
这里就不得不说到两个商业开发的原则问题了。
第一,对上家传入数据严加过滤,对传出给下家的数据仔细检查。
第二,合理使用异常。
第一点其实很简单的。也就是模块化开发的一个思想问题。对自己的行为负责。前端返回的数据究竟是
什么,需要进行校。不合格的剔除或者是修正。合格的处理完后,在传出之前也要加以校验,是否合格。具
体到这个问题里,就是来自Spring关于数据源的那个配置文件。
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource" destroy-method="close">
//略去数据源相关信息的配置
</bean>
这个配置文件里,有两个信息是很管用的。一个是class,一个是destroy-method。简单点说,一个是
数据源的实现类,一个是析构方法。Spring在读取这个配置文件以后,需要根据这些信息来实例化一些类,
然后内部再根据中间的那些配置信息来实际构造数据源。比如username啥的。
可是来了个问题。不能保证这里的ComboPooledDataSource数据源一定是可用的,也不能保证close方法
一定能关闭连接,对吧?Spring本身不能检查这个类是否真实有效,毫无Bug。实际上呢,也检查不了。同样
的,close方法是否有效,也需要进行检查。这就是我刚才说的,对上家数据的严加检查。
那好吧,怎么检查呢?最简洁的方法莫过于实际构造一下,连接池,获取数据库连接,执行一个测试
语句,然后关闭连接。如果一切都成功,那就OK。关于那个测试语句,配置过WebSphere数据源的同学们还记
得不?有个SQL语句会默认的写在数据源配置里,是 " SELECT 1 FROM TABLE "。嗯,对的,这个就是测试语
句。
这一套流程能走得通,走的顺,那么就可以在自己能力范围内说这个数据源和连接池是能用的,对吧?
这里补充一个知识。java.sql.Connection,这玩意不是class,是interface。
声明是:public interface Connection extends Wrapper 。
任何一个JDBC数据库连接的实现类都应该实现这个接口的全部方法。比如,close。API里的描述是,
立即释放Connection 对象的数据库和 JDBC 资源,而不是等待它们被自动释放。
熟悉Java的同学们应该记得一点,在规范里有个要求,就是接口的实现类必须实现接口的所有方法。但是
呢,这句话还有个意思,那就是,你可以在实现所有方法之外,再写几个方法。没人会管你。
啊哈,那就有疑问了。虽然API规定了close是关闭连接释放资源的。但这只是你接口的一厢情愿。也许人
家实现厂家觉得close方法不够帅,要改成closeConnection。那。。。Spring总不好傻傻的去死扣close方法来
关闭连接吧?虽然这方法必须实现,但是可没说一定要有内容啊。如果是空方法呢?
所以有了destroy-method这个配置项的出现。Spring说,不碍的,您老人家看哪个爽,告诉我就行。
现在测试完了。一切都成功了。
现在来看看第二个问题。合理使用异常。
又遇到一个问题。既然测试成功了,那总得给用户一点交待吧?难道说,测试成功了,就闷声大发财了?
显然不合适嘛。可以试想一下,你是程序员,然后点了个按钮,测试。结果呢,实际上是测试成功了,但是系
统啥动静都不给你。然后你傻傻的等痴痴的盼,一直等到天荒地老……嗯嗯,扯的有点远。如果你等一个小时
还不见动静,活不见人死不见尸的,你说你会不会骂娘?
那么怎么通知才能保证一定有效呢?println?这个不见得一定能看到。因为别人也可能在同时输出信息,
一下就刷掉了。那么有同学说了,最好是能暂停一下,我输出以后,就暂停了,不动了。
嗯,很好。
大家想想看,输出一大堆东西,然后此程序不动了,不继续执行了,这是啥玩意?
这不就是异常嘛!
只有异常能保证程序员一定能看到这个信息,比如,测试成功。这就是为什么Spring要采用这种方式来通
知的原因。
这里呢,我想更正同学们一个习惯成自然的想法。异常不一定是通知坏消息的。异常就是异常,只要你愿
意,你甚至可以在代码执行成功的时候,throws一个Exception。异常只不过是比较激烈的一种通知方式而已。
无他,仅此而已。
现在又有个问题来了。既然要测试,而且每次执行到此处的时候都要测试一下。那么……难道都卡在这里
不走了啊?显然更不合适啊。
熟悉log4J的同学应该看出来了,这是log4J输出的日志。很明显的,这种日志只应当在开发期间存在,不
应该在发布期间存在。因为开发期间数据库变动很大,比如改表啊,改数据库配置啊。所以需要通知用户是否
成功。但是产品一旦开发完毕,正式发布,这种信息就不应再出现,因为商业化运作的应用不允许乱动配置的,
对不?
所以log4J提供了一种方法。消息级别。INFO的时候,是看不到这个异常的。实现起来也很好办,catch了,
然后不做任何处理,也就是空的catch块。
具体实现的时候可以在catch里判断一下,如果等级是INFO的话,就不做任何事。如果不是,那就按照规则
去做。
结合到 “合理使用异常” 这句话来说呢,就是说,需要抛出异常的时候,就抛出。不需要抛出的时候,
就不抛出。对程序员来说,在必要的时候看到一串异常信息,是最合适的事情了。
关于异常的使用,这里不展开说了。有兴趣的同学可以参见林锐博士的 高质量Java编程。
相关推荐
Z-Stack 3.0.2 是一个专为物联网(IoT)设计的网络协议栈,它由Zigbee联盟开发并广泛应用于智能家居、智能能源、工业自动化等领域。Zigbee是一种低功耗、低成本、自组织的无线通信技术,支持多跳网络,能构建大规模...
TI公司免费提供ZigBee协议栈,这...翻译的内容为这个协议栈的Document里面的Z-Stack API。 在翻译的过程中参考了网上现有的内容,但增加和更新到了最新的ZStack-MSP2618-2.2.0-1.3.0这个协议栈里包含的这个文档的内容。
翻译的内容为这个协议栈的Document里面的Z-Stack Developer's Guide。 在翻译的过程中参考了网上现有的内容,但增加和更新到了最新的ZStack-MSP2618-2.2.0-1.3.0这个协议栈里包含的这个文档的内容。
在给定的"**C3P0连接池jar包.zip**"中包含的"c3p0-0.9.5.2.jar"是C3P0的一个版本,用于解决特定的问题,例如"DEBUG -- CLOSE BY CLIENT STACK TRACE"。 这个错误信息"DEBUG -- CLOSE BY CLIENT STACK TRACE"通常...
堆栈跟踪(Stack Trace)是当程序发生错误或异常时,系统自动生成的一种调试信息,它记录了代码执行的路径,显示了调用函数的层级结构。在开发过程中,原始的堆栈跟踪在编译和缩小后往往变得难以理解和解析,因为...
主要介绍了 Android studio 出现错误Run with --stacktrace option to get the stack trace. Run with --info or --debu的相关资料,需要的朋友可以参考下
redis-stack-server-7.2.0-v9.arm64.snap redis-stack-server-7.2.0-v9.bionic.arm64.tar.gz redis-stack-server-7.2.0-v9.bionic.x86_64.tar.gz redis-stack-server-7.2.0-v9.bullseye.x86_64.tar.gz redis-stack-...
loki相关镜像: grafana-image-renderer
$ ./test.sh 你会得到如下的跟踪日志: ..(snip)..[TEST] run sample program.....(snip)..*** BACK-TRACE ***GOTO stacktrace.sml:7.7-7.42: StackTrace.error[2] (from: stacktrace.sml:8.18-8.26: StackTrace....
3)....Fixed: Handling SEC_I_INCOMPLETE_CREDENTIALS in SSPI code (added searching client certificate) 4)....Fixed: Range-check error when closing WinAPI dialog EurekaLog 7.3 Hotfix 1 (7.3.1.0), 2-...
ndk-stack 调试 cocos2d-x ndk-stack 是一个强大的命令行调试工具,用于调试 Native 代码,especially for Android 平台的.so 库文件。下面我们来详细了解 ndk-stack 在 cocos2d-x 中的应用。 什么是 ndk-stack? ...
【标题】"f-stack-server-client.rar" 是一个包含F-stack客户端和服务器端实现代码的压缩文件,这为我们提供了一个深入理解F-stack网络框架如何在实际应用中处理客户端和服务器通信的绝佳机会。F-stack通常是一个高...
《深入理解前端项目-stacktrace.js》 在前端开发中,错误处理是不可或缺的一部分,而堆栈跟踪(Stack Trace)则是定位和解决问题的关键工具。本文将深入探讨stacktrace.js这一微型库,它允许我们在各种环境下获取...
标题“JCL_Debug_StackTrace_Demo”表明这是一个关于使用JCL进行调试和显示堆栈跟踪的示例项目。堆栈跟踪是在程序运行时记录函数调用序列的一种技术,它可以帮助开发者迅速定位错误发生的具体位置。在Delphi7中,JCL...
异步JS演示描述Ejemplos prácticos de implementación de Javascript asíncrono en sesion de clase del btc Fullstack valencia.Utiliza las funciones asíncronas:-Callbacks-Promesas-Async/await-Delay de 5...
npm install 时,会出现很多错误,包括配置错误等。 以webpack中安装autoprefixer为例: ... stack at ChildProcess.onCpExit (/usr/local/lib/node_modules/npm/node_modules/node-gyp/lib/configure.js:351:16) g
无服务器堆栈演示React App 《是用于创建全堆栈无...$ git clone https://github.com/AnomalyInnovations/serverless-stack-demo-client $ npm install 在本地运行。 $ npm run start 此回购由维护; 和制造商。
堆栈跟踪(stacktrace)是程序运行过程中出现错误或异常时生成的一种信息报告,它记录了错误发生时调用函数的顺序,帮助开发者定位问题所在。通常,原始的堆栈跟踪信息可能比较复杂,难以快速理解。Atom-stacktrace...
和horizontal-stack-in-card替代品 它允许将多张卡分组为一张没有边界的卡。 默认情况下,它将垂直堆叠所有内容。 选项 如果堆栈中的卡定义了--keep-background CSS样式,则不会替换背景。 例如,这对于很有用。 您...