记一次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.language
  val creationDate = assessmentReport.creationDate
  val 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: String
  private lateinit var translatorKey: String

  fun langLocale() = Locale.forLanguageTag(this.languageKey)
  fun translatorLocale() = Locale.forLanguageTag(this.translatorKey)
}

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

locales:
  supported:
    - languageKey: zh-CN
      translatorKey: 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

参考

  • 可见性修饰符

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

7000字详解 动态代理(JDK动态代理 CGLIB动态代理)与静态代理

代理模式 1. 代理模式 概念2. 静态代理3. 动态代理3.1.JDK动态代理3.2.CGLIB动态代理3.3. JDK动态代理和CGLIB动态代理区别 4.静态代理和动态代理区别5.篇末 1. 代理模式 概念 代理模式是一种设计模式。 使用代理对象来替代真实对象&#xff0c;用代理对象去访问目标对象。这样…

笔记(三)maxflow push relabel与图像分割

笔记&#xff08;三&#xff09;maxflow push relabel与图像分割 1. Push-Relabel算法思想2.Push-Relabel算法原理示意图3.Push-Relabel算法具体实例4. push relabel与图割 1. Push-Relabel算法思想 对于一个网络流图: 该算法直观可以这样理解&#xff0c;先在源节点处加入充足…

WordPress无需插件禁用WP生成1536×1536和2048×2048尺寸图片

我们在使用WordPress上传图片媒体文件的时候&#xff0c;是不是看到媒体库中有15361536和20482048的图片文件&#xff0c;当然这么大的文件会占用我们的服务器空间&#xff0c;如何禁止掉呢&#xff1f; function remove_default_image_sizes( $sizes) {unset( $sizes[1536x15…

人力资源管理后台 === 组织架构

目录 1.组织架构-树组件应用 2.组织架构-树组件自定义结构 3.组织架构-获取组织架构数据 4.组织架构-递归转化树形结构 5.组织架构-添加子部门-新建弹层组件 6.组织架构-添加子部门-表单结构 7.组织架构-添加子部门-表单基本校验 8.组织架构-添加子部门-表单业务校验 9…

ATK-ESP8266 WIFI模块串口通信通用实现方案

ATK-ESP8266 WIFI模块是一种常用的无线模块&#xff0c;它可以通过串口与外部设备进行通信&#xff0c;实现数据的收发和控制。本文将介绍一种通用的实现方案&#xff0c;帮助您在项目中使用ATK-ESP8266 WIFI模块进行串口通信。 【方案概述】 这个通用实现方案涵盖了ATK-ESP82…

VMWare虚拟机ubuntu克隆打不开

ubuntu克隆打不开 复制的存有ubuntu克隆的文件夹&#xff0c;导入vmware打不开 说找不到这个文件&#xff0c;那就到目录把它的删掉 的删掉 换000001.vmdk后缀的

Android 10.0 mtp模式下连接pc后显示的文件夹禁止删除copy重命名功能实现

1.前言 在10.0的系统开发中,usb连接pc端的时候有好几种模式,在做otg连接pc端的时候,改成mtp模式的时候,在pc端可以看到产品设备 的显示的文件夹的内容,对于产品设备里面的文件在pc端禁止做删除重命名拷贝等操作功能的实现 2.mtp模式下连接pc后显示的文件夹禁止删除copy重命…

在我国干独立游戏开发有多难?

游戏独立开发在中国&#xff0c;一直以来都是一条充满挑战的道路。尽管有着无限的激情和创意&#xff0c;但面对市场、资金、政策等多方面的困难&#xff0c;许多独立开发者在这条路上艰难前行。 首先&#xff0c;市场竞争激烈是中国游戏独立开发者面临的首要挑战。随着游戏产…

Unsupervised MVS论文笔记(2019年)

Unsupervised MVS论文笔记&#xff08;2019年&#xff09; 摘要1 引言2 相关工作3 实现方法3.1 网络架构3.2 通过光度一致性学习3.3 MVS的鲁棒光度一致性3.4 学习设置和实施的细节3.5.预测每幅图像的深度图 4 实验4.1 在DTU上的结果4.2 消融实验4.3 在ETH3D数据集上的微调4.4 在…

BUUCTF [MRCTF2020]Ez_bypass 1

题目环境&#xff1a;F12查看源代码 I put something in F12 for you include flag.php; $flagMRCTF{xxxxxxxxxxxxxxxxxxxxxxxxx}; if(isset($_GET[gg])&&isset($_GET[id])) { $id$_GET[id]; $gg$_GET[gg]; if (md5($id) md5($gg) && $id ! $gg) { …

显示Excel功能区或工具栏的方法不少,其中快捷方式最快

Microsoft Excel是Office套件中最复杂的工具之一&#xff0c;它提供了大量功能&#xff0c;其中大部分都是使用工具栏操作的。缺少工具栏使Excel很难完成工作。 如果Excel中没有这些关键元素&#xff0c;你将无法快速完成工作&#xff0c;因此&#xff0c;可以理解的是&#x…

算法基础之合并集合

合并集合 核心思想:并查集: 1.将两个集合合并2.询问两个元素是否在一个集合当中 基本原理:每个集合用一棵树表示 树根的编号就是整个集合的编号 每个节点存储其父节点&#xff0c;p[x]表示x的父节点 #include<iostream>using namespace std;const int N100010;int p[N];…

RESTful API 架构快速入门 Flask实现

RESTful 简介 1.1 为什么要使用 RESTful 架构&#xff1f; Representational State Transfer&#xff08;REST&#xff09;是一种面向资源的架构风格&#xff0c;广泛应用于网络服务的设计和开发。使用RESTful架构有以下几个优点&#xff1a; 简单性和可扩展性&#xff1a; RE…

springboot 拦截器中使用@Value注解为null

拦截器中获取配置参数为null 代码如下&#xff1a; 解决方式一&#xff1a; 检查你的WebMvcConfigurer实现类&#xff0c;比如我的是CCBWebMvcConfig 将拦截器以bean的形式注入&#xff1a; 我之前的写法是new 一个放进去的&#xff0c;这种会导致Value为null AutowiredJSCCB…

图像分割模型及架构选型介绍(MMSegmentation|sssegmentation等)

参考&#xff1a; https://zhuanlan.zhihu.com/p/618226513 0. 图像分割概述 图像分割通过给出图像中每个像素点的标签&#xff0c;将图像分割成若干带类别标签的区块&#xff0c;可以看作对每个像素进行分类。图像分割是图像处理的重要组成部分&#xff0c;也是难点之一。随…

二次创作Z01语言

目录 一&#xff0c;字符集 二&#xff0c;编译分词 三&#xff0c;token含义 四&#xff0c;Z01翻译成C 五&#xff0c;执行翻译后的代码 六&#xff0c;打印Hello World! 一&#xff0c;字符集 假设有门语言叫Z01语言&#xff0c;代码中只有0和1这两种字符。 二&#…

使用花生壳外网远程ssh访问内网主机 亲测有效

经常会遇到远程访问其他电脑的需求&#xff0c;一般首选向日葵软件&#xff0c;傻瓜式的连接远程桌面控制&#xff0c;非常方便。但是仅限于远程桌面远程协助这种。 对于程序员来说最佳的登录方式是ssh&#xff0c;同时远程桌面连过来的时候分辨率比较低&#xff0c;图形效果相…

上海交通大学生存手册

强烈推荐所有大学生去阅读《上海交通大学生存手册》。虽然它可能有些冗长&#xff0c;但非常重要&#xff0c;因为它道出了大学教育的本质。 如果几年前我能够看到这本书&#xff0c;也许我的大学生活会有所不同。现在我将向正在上大学或者将要上大学的你推荐这本书。 无论你…

Less的函数的介绍

文章目录 前言描述style.less输出后言 前言 hello world欢迎来到前端的新世界 &#x1f61c;当前文章系列专栏&#xff1a;Sass和Less &#x1f431;‍&#x1f453;博主在前端领域还有很多知识和技术需要掌握&#xff0c;正在不断努力填补技术短板。(如果出现错误&#xff0c;…

python--获取每张切片的不同PEF区间值的百分比

在全直径数字岩心中&#xff0c;如何获取每张切片的不同PEF区间值的百分比&#xff1f; import os import datetime from PIL import Image import numpy as np import csv import easygui as gclass Table(object):def __init__(self, table_data_path):self.table_data_path…