记一次Kotlin Visibility Modifiers引发的问题

概述

测试环境爆出ERROR告警日志java.lang.IllegalStateException: Didn't find report for specified language,登录测试环境ELK查到如下具体的报错堆栈日志:

java.lang.IllegalStateException: Didn't find report for specified language
at com.aba.report.service.biz.AssessmentReportService.getReportDTOByLanguage(AssessmentReportService.kt:116)
at com.aba.report.service.biz.AssessmentReportService.getAssessmentReport(AssessmentReportService.kt:60)
at com.aba.report.provider.biz.ReportBiz.getAssessmentResultTesting(ReportBiz.java:211)
at com.aba.report.provider.biz.ReportBiz.syncReportDataToXuhui(ReportBiz.java:171)
at com.aba.report.provider.controller.ReportController.lambda$saveReport$0(ReportController.java:143)
at org.apache.skywalking.apm.toolkit.trace.SupplierWrapper.get$original$K9vf71bp(SupplierWrapper.java:37)
at org.apache.skywalking.apm.toolkit.trace.SupplierWrapper.get$original$K9vf71bp$accessor$gH2x29d6(SupplierWrapper.java)
at org.apache.skywalking.apm.toolkit.trace.SupplierWrapper$auxiliary$KujBxtqH.call(Unknown Source)
at org.apache.skywalking.apm.agent.core.plugin.interceptor.enhance.InstMethodsInter.intercept(InstMethodsInter.java:86)
at org.apache.skywalking.apm.toolkit.trace.SupplierWrapper.get(SupplierWrapper.java)
at java.base/java.util.concurrent.CompletableFuture$AsyncSupply.run(CompletableFuture.java:1700)
at java.base/java.lang.Thread.run(Thread.java:834)

排查

这个报错就显得很是莫名其妙,没有一点点熟悉的感觉,像NPE、数组越界啥的。只能去看源码:

private fun getReportDTOByLanguage(assessmentReport: AssessmentReport): TranslatedAssessmentReportDTO {val language = assessmentReport.languageval creationDate = assessmentReport.creationDateval translatedAssessmentReportDTO = assessmentReport.translatedReports[Locale.CHINA.toLanguageTag()]?: assessmentReport.translatedReports[language]?: throw IllegalStateException("Didn't find report for specified language")translatedAssessmentReportDTO.creationDate = SimpleDateFormat("yyyy/MM/dd").format(creationDate.toEpochMilli())return translatedAssessmentReportDTO
}

至此还是一脸迷惑。

getAssessmentResult

这个方法的调用入参AssessmentReport,是从数据库查询到的数据。此时并没有第一时间怀疑数据库数据有问题,因为一直以来都是好的。

既然问题代码还没发布到生产环境,报错是在测试环境产生,那问题复现肯定比生产环境好复现。ELK里记录错误日志调用链TraceId,根据ELK的搜索结果以及代码,可以很快知道是哪个接口报错(仅仅看代码的话,遇到工程特别复杂,方法调用及被调用关系太多,并不能第一时间知道是哪个地方调用),也能查询到接口的requestBody。
在这里插入图片描述
启动应用,postman模拟请求getAssessmentResult接口,断点调试,本地可以复现问题。

经过反复分析,定位到问题根源在于数据库查询到的数据为空。如下图,Map为空,size=0
在这里插入图片描述
报错代码见下面截图,此处代码提交者,某某前辈简直是【呕心沥血】啊,在绝大多数报错响应信息都是中文的编码环境(将近30多个微服务工程)里,给你整个英文errMsg。其实这尼玛不就是一个空指针异常吗?!!
在这里插入图片描述
问题回到NPE。这个地方是在获取报告时出错,那我们去看看数据库的报告是啥样的?
在这里插入图片描述
如上图所示,下面一条MongoDB数据库记录对应的报告是正常的,translatedReports字段不为空,即不是空的JSON Object。上面的数据是有问题的。

那为啥保存到数据库的数据变成空呢?找到保存报告数据的方法,源码如下:

fun createReport(case: caseDTO): AssessmentReport {val translatedReports = localesConfig.supportedLanguages().map { createResult(case, it) }.associateBy { it.language }val assessmentReport = AssessmentReport(case.key, now(), Constants.LOCALE,false, translatedReports)return assessmentReportRepository.save(assessmentReport)
}

saveReport

还是上面提到的,在ELK里找到saveReport这个controller接口的requestBody日志,把数据复制到postman,点击send,断点调试。这就要看对代码的熟悉程度,如果熟悉业务代码逻辑,可以较快定位到问题。

下面截图显示localesConfig.supportedLanguages()是一个空集合,导致translatedReports是一个空对象,自然而然地,assessmentReport也就是一个空对象,即空的JSON Object,保存到数据库的数据自然就是空对象。
在这里插入图片描述
那就去看看LocalesConfig.kt源代码:

@Configuration
@ConfigurationProperties("locales")
open class LocalesConfig {private var supported = mutableListOf<LanguageConf>()fun supportedLanguages(): Set<Locale> {return supported.map { Locale.forLanguageTag(it.languageKey) }.toSet()}
}open class LanguageConf {lateinit var languageKey: Stringprivate lateinit var translatorKey: Stringfun langLocale() = Locale.forLanguageTag(this.languageKey)fun translatorLocale() = Locale.forLanguageTag(this.translatorKey)
}

是一个Spring @Configuration配置类,对应的配置文件类:

locales:supported:- languageKey: zh-CNtranslatorKey: zh-CN

这个配置类,最近把LocalesConfig里的var supported变量和LanguageConflateinit var translatorKey各增加一个private关键词。


ok, fine…
内心戏:这尼玛逗我呢。


去掉这两个关键词,则可以看到在这里插入图片描述
问题解决,保存到数据库的数据正常。

反思

Why

为啥要动这个配置类呢?

如下截图所示,IDEA给出提示:
在这里插入图片描述
注:

  1. IDEA版本号:
IntelliJ IDEA 2022.1.4 (Ultimate Edition)
Build #IU-221.6008.13, built on July 19, 2022
  1. 通过maven引入的Kotlin stdlib包版本号:1.3.72
<dependency><groupId>org.jetbrains.kotlin</groupId><artifactId>kotlin-stdlib</artifactId><version>1.3.72</version>
</dependency>
  1. Kotlin插件版本号: 221-1.6.21-release-337-IJ6008.13

private

根据文末给出的参考链接:

类、对象、接口、构造函数、方法与属性及其setter都可以有可见性修饰符。getter总是与属性有着相同的可见性。
在Kotlin中有这四个可见性修饰符:private、protected、internal和 public。 默认可见性是public。
声明为private,它只会在声明它的文件内可见。

!!!getter总是与属性有着相同的可见性!!!
!!!声明为private,它只会在声明它的文件内可见!!!

这和Java不一样吧。

我们在写Java实体类时,一般都会使用lombok来简化getter和setter,getter和setter当然是public的。

这也是为何在Java与Kotlin混合编程里,经常会遇到如下编译失败问题:

 Failed to execute goal org.jetbrains.kotlin:kotlin-maven-plugin:1.3.72:compile (kotlin-compile) on project dialog-service: Compilation failure
[ERROR] /Users/johnny/code/arch-biz/dialog/dialog-service/src/main/java/com/aba/dialog/service/domain/assessment/dialog/itemfactory/EvidenceQuestionDialogFactory.kt:[53,100] Cannot access 'state': it is private in 'OptionDTO'

对应的实体类源码:

@Data
public class OptionDTO implements Serializable {private String state;public String name;
}

那Kotlin语言里如何访问Java里的private方法或属性呢?

参考:kotlin-extension-function-access-java-private-field

open

在Java中允许创建任意的子类并重写方法任意的方法,除非显式使用final关键字进行标注。

但在Kotlin中所有的类默认都是final的,意味着不能被继承,而且在类中所有的方法默认也是final,那么就是Kotlin的方法默认也不能被重写。如果想在Kotlin中继承父类应该怎么做呢?

  • 为类增加open关键词,则类class可以被继承
open class Person {
}
  • 为方法增加open关键词,则方法method可以被重写

如果一个类没有增加open关键词,却为方法增加open关键词,会怎样呢?

简单分析之后,不难得出,如果要定义一个方法为open方法,则必须使方法所在的类也是open类;反之,则不亦然。写个demo验证一下:

class Person {open fun eat(food: String) {}
}

IDEA给出如下提示:
在这里插入图片描述

lateinit

参考

  • 可见性修饰符

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/172969.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

数组栈的实现

1.栈的概念及结构 栈&#xff1a;一种特殊的线性表&#xff0c;其只允许在固定的一端进行插入和删除元素操作 进行数据插入和删除操作的一端称为栈顶&#xff0c;另一端称为栈底 栈中的数据元素遵守后进先出LIFO,&#xff08;Last In First Out&#xff09;的原则 压栈&…

如何将 Python 运用到实际的测试工作中

1、自动化测试脚本编写&#xff1a; Python广泛用于编写自动化测试脚本&#xff0c;以执行各种测试任务。可以使用Selenium、Appium或PyTest等库来辅助测试脚本的编写。 下面是一个示例&#xff1a; from selenium import webdriver import unittestclass LoginTest(unittes…

MIT 6.824 -- MapReduce Lab

MIT 6.824 -- MapReduce Lab 环境准备实验背景实验要求测试说明流程说明 实验实现GoLand 配置代码实现对象介绍协调器启动工作线程启动Map阶段分配任务执行任务 Reduce 阶段分配任务执行任务 终止阶段 崩溃恢复 注意事项并发安全文件转换golang 知识点 测试 环境准备 从官方gi…

鸿蒙开发-ArkTS 语言-状态管理

鸿蒙开发-ArkTS 语言-基础语法 3. 状态管理 变量必须被装饰器装饰才能成为状态变量&#xff0c;状态变量的改变才能导致 UI 界面重新渲染 概念描述状态变量被状态装饰器装饰的变量&#xff0c;改变会引起UI的渲染更新。常规变量没有状态的变量&#xff0c;通常应用于辅助计算…

案例026:基于微信的原创音乐小程序的设计与实现

文末获取源码 开发语言&#xff1a;Java 框架&#xff1a;SSM JDK版本&#xff1a;JDK1.8 数据库&#xff1a;mysql 5.7 开发软件&#xff1a;eclipse/myeclipse/idea Maven包&#xff1a;Maven3.5.4 小程序框架&#xff1a;uniapp 小程序开发软件&#xff1a;HBuilder X 小程序…

单片机学习7——定时器/计数器编程

#include<reg52.h>unsigned char a, num; sbit LED1 P1^0;void main() {num0;EA1;ET01;//IT00;//设置TMOD的工作模式TMOD0x01;//给定时器装初值&#xff0c;50000,50ms中断20次&#xff0c;就得到1sTH0(65536-50000)/256;TL0(65536-50000)%256;TR01; // 定时器/计数器启…

从0开始学习JavaScript--JavaScript中的对象原型

JavaScript中的对象原型是理解该语言核心概念的关键之一。本文将深入探讨JavaScript对象原型的作用、使用方法以及与继承相关的重要概念。通过详细的示例代码和全面的讲解&#xff0c;将能够更好地理解和运用JavaScript对象原型&#xff0c;提高代码的可维护性和扩展性。 Java…

wangeditor实时预览

<template><div><!--挂载富文本编辑器--><div style"width: 45%;float: left;margin-left: 2%"><p>编辑内容</p><div id"editor" style"height: 100%"></div></div><div style"w…

日本运营商启动先进边缘云技术研发

摘要&#xff1a;日本运营商乐天移动最近启动了为 5G 之后的下一个通信标准开发边缘平台功能的研发工作。 乐天移动&#xff08;Rakuten Mobile&#xff09;表示&#xff0c;其面向下一代通信的先进边缘云技术研发&#xff08;R&D&#xff09;项目已被日本国家信息通信技术…

【Linux】信号

Linux 信号 1.信号介绍2.core dump3.发送信号3.1.kill3.2.send3.3.abort 4.信号产生4.1.软件条件产生信号4.1.1.SIGPIPE4.1.2.SIGALRM 4.2.硬件异常产生信号 5.信号处理6.可重入函数 & volatile7.SIGCHLD 1.信号介绍 信号本质是一种通知机制。 而进程要处理信号&#xff0…

windows通过regsvr32注册dll文件失败

1、注册dll文件失败 最近在研究中文输入法&#xff0c;下载SampleIME源码后编译得到SampleIME.dll&#xff0c;最后只需要将输入法安装&#xff08;即注册&#xff09;就可以使用了。 但是通过命令&#xff1a; regsvr32 C:\Windows\System32\SampleIME.dll 注册时却提示错…

6.2 Windows驱动开发:内核枚举SSSDT表基址

在Windows内核中&#xff0c;SSSDT&#xff08;System Service Shadow Descriptor Table&#xff09;是SSDT&#xff08;System Service Descriptor Table&#xff09;的一种变种&#xff0c;其主要用途是提供Windows系统对系统服务调用的阴影拷贝。SSSDT表存储了系统调用的函数…

光线追踪-Peter Shirley的RayTracing In One Weekend系列教程(book1-book3)代码分章节整理

自己码完了一遍了&#xff0c;把代码分章节整理了一下&#xff0c;可以按章节独立编译&#xff0c;运行, 也可以直接下载编译好的release版本直接运行。 项目地址&#xff1a; Github: https://github.com/disini/RayTracingInOneWeekendChaptByChapt ​ ​ ​ ​

transformers pipeline出现ConnectionResetError的解决方案

大家好,我是爱编程的喵喵。双985硕士毕业,现担任全栈工程师一职,热衷于将数据思维应用到工作与生活中。从事机器学习以及相关的前后端开发工作。曾在阿里云、科大讯飞、CCF等比赛获得多次Top名次。现为CSDN博客专家、人工智能领域优质创作者。喜欢通过博客创作的方式对所学的…

如果客户端同时有ipv4和ipv6,浏览器是如何选择用哪种ip

在互联网协议&#xff08;IP&#xff09;的发展历程中&#xff0c;IPv4和IPv6是两种主要的版本。对于一个客户端来说&#xff0c;同时拥有IPv4和IPv6的能力是常见的情况。那么&#xff0c;当一个客户端同时具有IPv4和IPv6的能力时&#xff0c;浏览器是如何选择使用哪种IP进行通…

linux复习笔记05(小滴课堂)

hell脚本与crontab定时器的运用 查看状态&#xff1a; 关闭服务&#xff1a; 开启服务&#xff1a; 重启服务&#xff1a; crontab定时器的使用&#xff1a; 我们可以看到没有任何任务。 编辑&#xff1a; 我们可以看到这个任务了。 删除所有任务&#xff1a; 这代表着每分钟…

5.一维数组——输入一行字符,统计其中各个大写字母出现的次数。

文章目录 前言一、题目描述 二、题目分析 三、解题 程序运行代码 四、举一反三一、题目描述 二、题目分析 三、解题 程序运行代码 前言 本系列为一维数组编程题&#xff0c;点滴成长&#xff0c;一起逆袭。 一、题目描述 输入一行字符&#xff0c;统计其中各个大写字母出现的…

Redis序列化操作

目录 1.protostuff 的 Maven 依赖 2.定义实体类 3.序列化工具类 ProtostuffSerializer 提供了序列化和反序列化方法 4.测试 利用 Jedis 提供的字节数组参数方法&#xff0c;如&#xff1a; public String set(String key, String value) public String set(byte[] key…

CentOS7部署FTP服务器

首先准备一台centos7虚拟机&#xff0c;作为服务器IP地址必须是固定的。 vim /etc/sysconfig/network-scripts/ifcfg-ens33配置内容如下&#xff1a; TYPE"Ethernet" PROXY_METHOD"none" BROWSER_ONLY"no" BOOTPROTO"static" DEFROU…

【Vue3+Vite】解决build后空白页的问题

目录 Hash 模式 HTML5 模式&#xff08;历史模式&#xff09; 配置Nginx 配置Spring Boot Hash 模式 build后空白页的问题可能是使用的是历史模式&#xff0c;因为Vue是一个单页的客户端应用&#xff0c;如果没有适当的服务器配置&#xff0c;访问会得到一个 404 错误…