首页 > 其他分享 >都说了能不动就别动,非要去调整,出生产事故了吧 → 补充

都说了能不动就别动,非要去调整,出生产事故了吧 → 补充

时间:2024-02-26 09:13:47浏览次数:27  
标签:java 事故 别动 ibatis org apache 非要 com junit

开心一刻

  今天去超市买饮料

  老板说 5 元,我听成了“会员”

  我说没有

  老板跟我对视了 10 秒

  然后,老板说:没有你买啥?

  我说:没有就不让买?

  当时的老板:

 瑕疵回顾

  都说了能不动就别动,非要去调整,出生产事故了吧 中有一个地方讲的有瑕疵,不知道你们发现了没有

  框住的第一句是没问题的,但第二句是不够严谨的,我自罚三耳光!

  为什么这么说,我们细看下当时的异常堆栈信息

  发现了什么?

  在 Mybatis 与 JDBC 之间有 hikari 

   hikari 是什么,一个性能极高的数据库连接池,它是可以有自己的想法的!

  思维再扩散一点,格局再打开一点,是不是就是:在 Mybatis 与 JDBC 之间有 数据库连接池 

  哪些说不用 数据库连接池 的小伙伴,你最好想清楚了再说

  用关系型数据库,而不用数据库连接池的项目多吗?仔细回忆回忆

  回到框住的第二句,严谨的说法应该是:而是交由下游组件

  至于下游组件是 Hikari ,还是 Druid ,亦或是其他的,是不是都囊括进来了?是不是就严谨了?

druid SQLFeatureNotSupportedException

  mybatis-plus/issues/1114 中提到了一个异常: java.sql.SQLFeatureNotSupportedException 

  基于 druid 1.1.16 触发的异常

  我们调整下代码

  1、引入 druid 依赖

  2、修改数据源类型(默认的 hikari 可以不用配置 type )

  数据库数据源就已经切成 druid 了

  我们将 mysql-connector-java 版本调回到最初的 5.1.26 , Mybatis-Plus 仍使用 3.1.1 

  运行 com.qsl.OrderTest#orderListAllTest ,此时的异常是什么,还是 Conversion not supported for type java.time.LocalDateTime 吗?

org.springframework.dao.InvalidDataAccessApiUsageException: Error attempting to get column 'pay_time' from result set.  Cause: java.sql.SQLFeatureNotSupportedException
; null; nested exception is java.sql.SQLFeatureNotSupportedException

    at org.springframework.jdbc.support.SQLExceptionSubclassTranslator.doTranslate(SQLExceptionSubclassTranslator.java:96)
    at org.springframework.jdbc.support.AbstractFallbackSQLExceptionTranslator.translate(AbstractFallbackSQLExceptionTranslator.java:72)
    at org.springframework.jdbc.support.AbstractFallbackSQLExceptionTranslator.translate(AbstractFallbackSQLExceptionTranslator.java:81)
    at org.mybatis.spring.MyBatisExceptionTranslator.translateExceptionIfPossible(MyBatisExceptionTranslator.java:73)
    at org.mybatis.spring.SqlSessionTemplate$SqlSessionInterceptor.invoke(SqlSessionTemplate.java:446)
    at com.sun.proxy.$Proxy54.selectList(Unknown Source)
    at org.mybatis.spring.SqlSessionTemplate.selectList(SqlSessionTemplate.java:230)
    at com.baomidou.mybatisplus.core.override.MybatisMapperMethod.executeForMany(MybatisMapperMethod.java:158)
    at com.baomidou.mybatisplus.core.override.MybatisMapperMethod.execute(MybatisMapperMethod.java:76)
    at com.baomidou.mybatisplus.core.override.MybatisMapperProxy.invoke(MybatisMapperProxy.java:62)
    at com.sun.proxy.$Proxy60.selectList(Unknown Source)
    at com.qsl.OrderTest.orderListAllTest(OrderTest.java:33)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:498)
    at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50)
    at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
    at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47)
    at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17)
    at org.springframework.test.context.junit4.statements.RunBeforeTestExecutionCallbacks.evaluate(RunBeforeTestExecutionCallbacks.java:74)
    at org.springframework.test.context.junit4.statements.RunAfterTestExecutionCallbacks.evaluate(RunAfterTestExecutionCallbacks.java:84)
    at org.springframework.test.context.junit4.statements.RunBeforeTestMethodCallbacks.evaluate(RunBeforeTestMethodCallbacks.java:75)
    at org.springframework.test.context.junit4.statements.RunAfterTestMethodCallbacks.evaluate(RunAfterTestMethodCallbacks.java:86)
    at org.springframework.test.context.junit4.statements.SpringRepeat.evaluate(SpringRepeat.java:84)
    at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325)
    at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:251)
    at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:97)
    at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290)
    at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71)
    at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)
    at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58)
    at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268)
    at org.springframework.test.context.junit4.statements.RunBeforeTestClassCallbacks.evaluate(RunBeforeTestClassCallbacks.java:61)
    at org.springframework.test.context.junit4.statements.RunAfterTestClassCallbacks.evaluate(RunAfterTestClassCallbacks.java:70)
    at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
    at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.run(SpringJUnit4ClassRunner.java:190)
    at org.junit.runner.JUnitCore.run(JUnitCore.java:137)
    at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:69)
    at com.intellij.rt.junit.IdeaTestRunner$Repeater$1.execute(IdeaTestRunner.java:38)
    at com.intellij.rt.execution.junit.TestsRepeater.repeat(TestsRepeater.java:11)
    at com.intellij.rt.junit.IdeaTestRunner$Repeater.startRunnerWithArgs(IdeaTestRunner.java:35)
    at com.intellij.rt.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:232)
    at com.intellij.rt.junit.JUnitStarter.main(JUnitStarter.java:55)
Caused by: java.sql.SQLFeatureNotSupportedException
    at com.alibaba.druid.pool.DruidPooledResultSet.getObject(DruidPooledResultSet.java:1771)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:498)
    at org.apache.ibatis.logging.jdbc.ResultSetLogger.invoke(ResultSetLogger.java:69)
    at com.sun.proxy.$Proxy73.getObject(Unknown Source)
    at org.apache.ibatis.type.LocalDateTimeTypeHandler.getNullableResult(LocalDateTimeTypeHandler.java:38)
    at org.apache.ibatis.type.LocalDateTimeTypeHandler.getNullableResult(LocalDateTimeTypeHandler.java:28)
    at org.apache.ibatis.type.BaseTypeHandler.getResult(BaseTypeHandler.java:81)
    at org.apache.ibatis.executor.resultset.DefaultResultSetHandler.applyAutomaticMappings(DefaultResultSetHandler.java:521)
    at org.apache.ibatis.executor.resultset.DefaultResultSetHandler.getRowValue(DefaultResultSetHandler.java:402)
    at org.apache.ibatis.executor.resultset.DefaultResultSetHandler.handleRowValuesForSimpleResultMap(DefaultResultSetHandler.java:354)
    at org.apache.ibatis.executor.resultset.DefaultResultSetHandler.handleRowValues(DefaultResultSetHandler.java:328)
    at org.apache.ibatis.executor.resultset.DefaultResultSetHandler.handleResultSet(DefaultResultSetHandler.java:301)
    at org.apache.ibatis.executor.resultset.DefaultResultSetHandler.handleResultSets(DefaultResultSetHandler.java:194)
    at org.apache.ibatis.executor.statement.PreparedStatementHandler.query(PreparedStatementHandler.java:65)
    at org.apache.ibatis.executor.statement.RoutingStatementHandler.query(RoutingStatementHandler.java:79)
    at com.baomidou.mybatisplus.core.executor.MybatisSimpleExecutor.doQuery(MybatisSimpleExecutor.java:67)
    at org.apache.ibatis.executor.BaseExecutor.queryFromDatabase(BaseExecutor.java:324)
    at org.apache.ibatis.executor.BaseExecutor.query(BaseExecutor.java:156)
    at org.apache.ibatis.executor.CachingExecutor.query(CachingExecutor.java:109)
    at org.apache.ibatis.executor.CachingExecutor.query(CachingExecutor.java:83)
    at org.apache.ibatis.session.defaults.DefaultSqlSession.selectList(DefaultSqlSession.java:147)
    at org.apache.ibatis.session.defaults.DefaultSqlSession.selectList(DefaultSqlSession.java:140)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:498)
    at org.mybatis.spring.SqlSessionTemplate$SqlSessionInterceptor.invoke(SqlSessionTemplate.java:433)
    ... 39 more
View Code

  和 hikari 对比下

  可以看到,差别还是比较大的

  从异常堆栈信息,我们得知,调用栈是: Mybatis ->  数据库连接池 ->  mysql-connector-java 

  当数据库连接池是 druid 1.1.16 时,在调用栈的第二环( druid )就异常了

  而当数据库连接池是 hikari 3.4.5 时,在调用栈的第三环才异常,而在第二环( hikari )并未异常

  那就来看看 druid 为何会异常

  为何异常

  相信看了 都说了能不动就别动,非要去调整,出生产事故了吧 的小伙伴,能够很快定位到关键代码

  对,你想的没错,就是从异常堆栈中找关键位置

  我们就从 LocalDateTimeTypeHandler.java:38 开始,来看看异常是怎么产生的

  打个断点,然后再 debug 下

  按 F7 ,进入 org.apache.ibatis.logging.jdbc.ResultSetLogger#invoke 

  红框框住的代码,大家能看懂吗?我给大家拆分下

   method 信息如下

   rs 对象

   method.invoke(rs, params); 作用是不是明显了?

  就是反射调用 DruidPooledResultSet 的方法: getObject(String columnLabel, Class<T> type) 

  跟进去看下该方法的具体实现

  哦豁,直接抛出异常 SQLFeatureNotSupportedException 

  原因是不是找到了: druid 1.1.16 不支持根据JAVA类型获取列值

  如何修复

  1、降低 Mybatis 版本

  将 Mybatis 版本降低至 3.5.0 或以下

  因为项目中用的是 Mybatis-Plus ,我们将其降至 3.1.0 

  运行 com.qsl.OrderTest#orderListAllTest ,没异常,结果也正确

  此时各个组件的版本: Mybatis-Plus 3.1.0 (即 Mybatis 3.5.0 ), druid 1.1.16 , mysql-connector-java 5.1.26 

  为何将 Mybatis 的版本调整至 3.5.0 就可以了?

  这其实跟 Mybatis 3.5.1 对 LocalDateTimeTypeHandler.java 的调整有关

   Mybatis 3.5.0 依赖下游组件的 getTimestamp(String columnLabel) 或 getTimestamp(int columnIndex) 

  而 Mybatis 3.5.1 依赖下游组件的 getObject(int columnIndex, Class<T> type) 或 getObject(String columnLabel, Class<T> type) 

  2、升级 druid 版本

  升级到哪个版本,这个需要看 druid 从哪个版本开始支持 LocalDateTime 

  根据官方说明,从 1.1.18 开始支持

  我们来看下 1.1.18 相较于上一个版本( 1.1.16 ,没有 1.1.17 ),对 DruidPooledResultSet 调整了什么

  我们将 Mybatis-Plus 改回成 3.1.1 ( Mybatis 3.5.1 ),然后将 druid 升级到 1.1.18 

  再执行 com.qsl.OrderTest#orderListAllTest ,你会发现还是有异常!!!

  但是先别慌,该异常

  是不是很眼熟?

  不是在 都说了能不动就别动,非要去调整,出生产事故了吧 已经解决过了吗?

  那怎么修?

  有小伙伴会说:这个我会,将 mysql-connector-java 升级到 5.1.37 

  就不能一步到位升级到 5.1.42 ?, 5.1.37 有 NullPointerException 呀!!!

  此时各个组件的版本: Mybatis-Plus 3.1.1 (即 Mybatis 3.5.1 ), druid 1.1.18 , mysql-connector-java 5.1.42 

Hikari 为何没问题

  此刻相信大家会有一个问题:为何 hikari 没有 druid 的那个问题( SQLFeatureNotSupportedException )

  我们来分析下,hikari 版本是 3.4.5 

  它的 HikariProxyResultSet 有实现 getObject(int columnIndex, Class<T> type) 和 getObject(String columnLabel, Class<T> type) 

  具体实现交给了下游,也就是交给了 mysql-connector-java 

  那它是一开始就是这么实现的,还是在 3.4.5 或之前的某个版本调整成这样的了?

  因为 HikariProxyResultSet 是动态生成的,没有现成的源代码,我也想帮你们分析,可我暂时做不到呀

  等我了解了 HikariCP 动态代理实现机制,我再来给你们分析,暂时算我欠你们的!

  你们要实在是觉的不爽,来打我呀

 

总结

  遇到异常不要害怕,异常堆栈是很有用的信息

  遇到开源组件的问题, github 上搜它的相关 issue ,往往能事半功倍

  还是那句话:能不动就不要动,改好没绩效,改出问题要背锅,吃力不讨好,又不是不能跑

标签:java,事故,别动,ibatis,org,apache,非要,com,junit
From: https://www.cnblogs.com/youzhibing/p/18029435

相关文章

  • 都说了能不动就别动,非要去调整,出生产事故了吧
    开心一刻减肥是有技巧的,比如我朋友早上一杯水,中午一个鸡蛋,晚上一碗稀粥每天10公里,500个跳绳,200个蛙跳他以前180斤,现在连人带盒才5斤就问你们,这减肥效果是不是杠杠的?MyBatis替换成MyBatis-Plus背景介绍一个老项目,数据库用的是 MySQ......
  • 监控Celery不一定非要使用Flower
    运维平台中有许多的周期/定时/异步任务,例如证书扫描、数据备份、日志清理、线上作业等等,这些任务的执行都是借助于Celery来完成的。任务多了之后就会遇到一系列的问题,例如我之前写过的将任务分多队列来解决生产环境下的任务优先级问题,除此之外还要经常关注队列的状态以及任务的运......
  • 给你一颗“定心丸”——记一次由线上事故引发的Log4j2日志异步打印优化分析
    一、内容提要自知是人外有人,天外有天,相信对于Log4j2的异步日志打印早有老师或者同学已是熟稔于心,优化配置更是信手拈来,为了防止我在这里啰里八嗦的班门弄斧,我先将谜底在此公布:log4j2.asyncQueueFullPolicy=Discard&log4j2.discardThreshold=ERROR,这两个Log4j2配置在强依赖的RPC......
  • 线程池最佳实践!这几个坑使用不当直接生产事故!!
    拿来即用!这篇文章我会介绍我使用线程池的时候应该注意的坑以及一些优秀的实践。1、正确声明线程池线程池必须手动通过 ThreadPoolExecutor 的构造函数来声明,避免使用Executors 类创建线程池,会有OOM风险。Executors 返回线程池对象的弊端如下(后文会详细介绍到):FixedThreadPo......
  • Mybatis-Wrapper导致的生产事故
    近期遭遇了一次生产环境的严重告警,涉及慢接口和CPU过载。经过排查,发现问题根源在于一段使用MyBatis的查询代码。当传入空列表作为查询条件时,MyBatis会忽略该条件,导致全表扫描,进而引发系统资源耗尽和频繁的FullGC灾难回顾前两天晚上,正在收拾包准备下班,电脑刚放进包里,我的手机......
  • 【智慧煤矿】三令五申注重煤矿生产安全,为何事故仍然频发?
    12月21日,黑龙江一煤矿发生事故致12死13伤。就在昨天,令人痛心的新闻再次发生,黑龙江一地煤矿发生一起斜井跑车事故,造成了重大人员伤亡事故。对于煤矿产业,安全生产是重中之重,除了人为保障以外,智慧煤矿——智能监控安全监管仍然重要。一、方案概述TSINGSEE青犀通过物联网、移动互......
  • mq引发的线上事故【nested exception is org.springframework.amqp.AmqpIOException】
    背景:前一天晚上运维关闭了服务间公网访问,第二天早晨系统登录不进去了,重启进去也报错原因:mq连接不上,大量mq请求拖垮了系统;开发人员犯了两个错误(1.内部通信用了公网地址访问,2.mq发送请求没有做异步处理)ERROR-Servlet.service()forservlet[dispatcherServlet]incontex......
  • P0级事故频发后,这家公司终于醒悟了!
    近期,国内的互联网大厂接连爆发P0级事件,阿里云崩完滴滴崩,企业在追求效益的前提是业务的连续和稳定。如果发生故障不能快速恢复,引发业务中断,给企业带来的损失是巨大的,换言之,企业需要一套清晰、智能化的运维管理系统来帮助管理人员提高对整个IT系统的把控能力。运维的核心价值是保障业......
  • 多起问界事故背后 莫要捧杀华为
    文|AUTO芯球作者|李欣从来没想到,华为造汽车会像是在“造神”一样。这不,这两天,很多类似“车朗诵”的视频号主都发布了同一个视频,视频显示,自己朋友驾驶问界M5开启辅助驾驶后,和路上行车还是追尾了。由事故图片可知,安全气囊已经弹出车头损毁严重,华为官方已经把车辆拖走,表示会维修。......
  • P0级事故发生,留给运维的时间不多了?
    一、案例背景打工人的焦虑,已经延伸到在线文档了。近日,语雀P0级故障想必大家都有所体会,宕机近8小时,笔记、离线同步完全不可用。作为用户尤其担心我的文档资料是否会因此消失。这泼天的8小时,放眼互联网界也是相当炸裂的。从次日的故障处理通报可知,团队在收到运维监控系统报警后,定位故......