IntelliJ IDE 插件开发 | (七)PSI 入门及实战(实现 MyBatis 插件的跳转功能)

news/2024/5/13 13:36:02

系列文章

  • IntelliJ IDE 插件开发 |(一)快速入门
  • IntelliJ IDE 插件开发 |(二)UI 界面与数据持久化
  • IntelliJ IDE 插件开发 |(三)消息通知与事件监听
  • IntelliJ IDE 插件开发 |(四)来查收你的 IDEA 使用报告吧
  • IntelliJ IDE 插件开发 |(五)VFS 与编辑器
  • IntelliJ IDE 插件开发 |(六)内部模式的使用
  • IntelliJ IDE 插件开发 |(七)PSI 入门及实战(实现 MyBatis 插件的跳转功能)

前言

所谓 PSI(Program Structure Interface),直译过来是程序结构接口,其实就是 IntelliJ 平台给我们提供用来解析代码文件,简化对各类编程语言(Java、Kotlin、XML)操作的接口。大部分针对编程语言或者框架的便利插件其实就与此相关,本文则会先介绍关于 PSI 的一些基础知识,然后再以一些 Mybatis 插件提供的 Java 方法 和 Mapper XML 文件互相跳转的例子来说明 PSI 的实际应用,最终实现效果如下图,本文涉及的到的完整代码文件也已上传到GitHub。

动画

PSI file(PSI 文件)

在本系列的第五篇文章中介绍了Virtual Files 和 Documents 用于处理文件的 API,而 PSI file 也是用于处理文件的 API,不过也有一些不同,具体如下:

类别层面范围
VF、Document文本文件应用级
PSI编程语言语法树项目级

获取文件的 PSI file 对象的方式主要有以下几种(来自官网):

ContextAPI
ActionAnActionEvent.getData(CommonDataKeys.PSI_FILE)
DocumentPsiDocumentManager.getPsiFile()
PSI ElementPsiElement.getContainingFile()
Virtual FilePsiManager.findFile(), PsiUtilCore.toPsiFiles()
File NameFilenameIndex.getVirtualFilesByName()

最后一种方式FilenameIndex.getVirtualFilesByName()得到的结果是Virtual File 对象,需要再通过倒数第二行的方式再获取到对应的 PIS file 对象。

不过通过以上方式获取到的 PsiFile 只是顶层的接口,针对不同的编程语言,我们会使用相应的实现类。例如 Java 是 PsiJavaFile,XML 是 XMLFile。

下面通过实际使用来进行介绍,首先是 PsiJavaFile,当然在使用之前需要确保build.gradle文件中将 Java 添加到插件配置中(XML 内置无需添加):

intellij {// 用到的插件plugins.set(listOf("com.intellij.java"))
}

然后在plugin.xml中加入以下配置:

<depends>com.intellij.modules.java</depends>

如果是 XML 则需要添加如下配置:

<depends>com.intellij.modules.xml</depends>

经过以上配置后,我们就可以使用 PsiJavaFile 和 XMLFile 的相关 API 了。

例如以下代码可以得到 java 文件的所属表名和类中 所有的方法名,然后展示出来:

class PsiJavaAction: AnAction() {override fun actionPerformed(e: AnActionEvent) {// 获取 PsiFile 对象val psiFile = e.getData(CommonDataKeys.PSI_FILE)// 转换为 PsiJavaFileval psiJavaFile = psiFile as PsiJavaFile// 获取类所属包Utils.info("当前类所属包:${psiJavaFile.packageName}")// 遍历获取所有的方法名psiJavaFile.accept(object: JavaRecursiveElementVisitor() {override fun visitMethod(method: PsiMethod) {Utils.info("查找到方法:${method.name}")}})}}

image-20240324153748554

在以上代码中需要注意psiJavaFile.accept()方法,其中 accept 方法是 PsiFile 所提供的方法,方法签名为void accept(@NotNull PsiElementVisitor visitor),用于遍历 PSI 文件中的各类元素,可以看到上面我们在传参时传递的是JavaRecursiveElementVisitor,这是用于遍历 Java 中各类元素(字段、方法、注解等)的一个实现类,只需要重写对应的方法即可,在上面我们重写了visitMethod方法,其实内部提供了很多方法,大家可以自行尝试,通过方法名也可以看到这里还支持遍历 break 语句,断言语句等等:

image-20240324155416824

下面再说明如何遍历 XML 文件中的元素:

class PsiXMLAction: AnAction() {override fun actionPerformed(e: AnActionEvent) {// 获取 PsiFile 对象val psiFile = e.getData(CommonDataKeys.PSI_FILE)// 转换为 XmlFileval xmlFile = psiFile as XmlFile// 获取根标签名称Utils.info("根标签名称:${xmlFile.rootTag?.name}")// 遍历获取所有的元素信息xmlFile.accept(object: XmlRecursiveElementVisitor() {override fun visitXmlAttribute(attribute: XmlAttribute) {Utils.info("属性名称:${attribute.name}, 属性值:${attribute.value}")}})}}

image-20240324160702331

可以看到这里遍历使用的是XmlRecursiveElementVisitor,是 XML 对于PsiElementVisitor 的一个实现类,用于遍历 XML 文件中的各种元素:

image-20240324161835384

PSI Element(PSI 元素)

在上面介绍 PSI 文件的时候多次提到元素的概念,PSI 文件则正是由一系列的 PSI Element 所组成。和 PSI file 类似,PSI Element 也属于一个顶层接口,针对不同的编程语言,会有多种 PSI 元素。以 Java 为例,有 PsiClass、PSIMethod、PsiField 等对应 Java 语法的各类元素。而 XML 中也有 XmlTag、XmlAttribute 等概念。那我们如何快速知道一个文件中有哪些 PSI 元素?如何快速知道一个我们不熟悉的编程语言中的 PSI 元素?别慌,IntelliJ平台给我提供了工具:

image-20240324163733415

通过 IntelliJ 平台的工具,我们可以很方便地查看当前或者任意一种文件的 PSI 结构,下面分别以 Java 和 XML 文件为例,首先是 Java 文件:

image-20240324164120824

同时点击左下的元素节点,上方还会自动对应到元素位置:

image-20240324164235622

然后是 XML 文件:

image-20240324164626925

当然,除了 Java 和 XML,IntelliJ 支持的编程语言远不止这些,这里展示一部分,剩下的大家可以自行探索:

image-20240324164741314

上面介绍了如何快速查看 PSI 文件中的元素,下面再介绍如何去获取 PSI 元素,以下来自官网:

ContextAPI
ActionAnActionEvent.getData(CommonDataKeys.PSI_ELEMENT)Note: If an editor is currently open and the element under caret is a reference, this will return the result of resolving the reference.
PSI FilePsiFile.findElementAt(offset)
ReferencePsiReference.resolve()

可以看到总共有三种方式:第一种是直接获取当前光标位置的 PSI 元素;第二种是可以自己指定偏移量(如果不熟悉偏移量的概念,可以看本系列第五篇文章中讲解 CaretModel 的部分),获取指定文件指定位置的 PSI 元素;最后一种引用则使用的较少,这里不再展开介绍,大家可以查看官方文档进行了解。

除了获取某个位置的 PSI 元素,我们还可以获取其所属父元素或者子元素,下面以 Java 文件为例讲解如何使用:

class PsiJavaAction: AnAction() {override fun actionPerformed(e: AnActionEvent) {val psiFile = e.getData(CommonDataKeys.PSI_FILE)// 获取光标处 PSI 元素,假定该元素在方法内部val psiElement = e.getData(PlatformDataKeys.EDITOR)?.caretModel?.let { psiFile?.findElementAt(it.offset) }// 获取该元素所属的方法名val psiMethod = PsiTreeUtil.getParentOfType(psiElement, PsiMethod::class.java)// 获取该元素所属的类名val psiClass = PsiTreeUtil.getParentOfType(psiElement, PsiClass::class.java)Utils.info("所属方法名:${psiMethod?.name}")Utils.info("所属类名:${psiClass?.name}")}}

可以看到上面我们使用PsiTreeUtil::getParentOfType可以获取到一个元素的父元素,同时支持跨层级获取,既可以获取元素所属的方法,也可以获取元素所属的类。

效果如下:

image-20240324172523728

相应地我们也可以通过PsiTreeUtil::getChildrenOfTypeAsList去获取某个元素的所有子元素:

class PsiJavaAction: AnAction() {override fun actionPerformed(e: AnActionEvent) {val psiFile = e.getData(CommonDataKeys.PSI_FILE)// 先获取光标所在处元素所属的类val psiElement = e.getData(PlatformDataKeys.EDITOR)?.caretModel?.let { psiFile?.findElementAt(it.offset) }val psiClass = PsiTreeUtil.getParentOfType(psiElement, PsiClass::class.java)// 获取类中所有的方法val psiMethods = PsiTreeUtil.getChildrenOfTypeAsList(psiClass, PsiMethod::class.java)Utils.info("包含的方法:${psiMethods.joinToString(",") { it.name }}")}}

image-20240324174920319

实战

在正式实现前,先介绍一下整体的实现思路,这里只说明如何从 Java 方法跳转到 Mapper XML 文件中的节点,反向参考代码也很好理解,思路如下:

  1. 左侧图标行标记符通过实现RelatedItemLineMarkerProvider并重写collectNavigationMarkers方法设置。
  2. 判断代码行的元素类型为 PsiMethod 才进行设置,同时文件类名以 Mapper 结尾。
  3. 根据类名在项目查找同名的 Mapper XML 文件。
  4. 通过 accept 方法遍历 XML 文件所有的属性,将 id 值为对应方法名的标签所对应的元素保存到可跳转的目标。

设置行标记符号,平台给我们提供了 RelatedItemLineMarkerProvider 类进行设置,只需要自定义了自己的行标记类,然后在 plugin.xml 中添加 如下配置即可:

<codeInsight.lineMarkerProvider language="JAVA"implementationClass="cn.butterfly.psi.provider.JavaMapperLineMarkerProvider"/>

代码实现如下:

class JavaMapperLineMarkerProvider: RelatedItemLineMarkerProvider() {override fun collectNavigationMarkers(element: PsiElement,result: MutableCollection<in RelatedItemLineMarkerInfo<*>>) {// 查找类名后缀为 Mapper 内的所有方法if (element !is PsiMethod) {return}val psiClass = PsiTreeUtil.getParentOfType(element, PsiClass::class.java) ?: returnval className = psiClass.name ?: returnif (!className.endsWith("Mapper")) {return}// 查找同名 XML 文件对应的 PSI 文件对象val virtualFile = FileTypeIndex.getFiles(XmlFileType.INSTANCE, GlobalSearchScope.allScope(element.project)).first { it.name.startsWith(className) }val psiFile = PsiManager.getInstance(element.project).findFile(virtualFile)// 遍历 XML 文件中标签 id 节点值等于 Java 方法名的元素, 然后添加可跳转的行标记符psiFile?.accept(object : XmlRecursiveElementVisitor() {override fun visitXmlAttribute(attribute: XmlAttribute) {if (attribute.name == "id" && attribute.value == element.name) {// NavigationGutterIconBuilder 用于创建标识符result.add(NavigationGutterIconBuilder.create(PluginIcons.MAPPER_ICON).setTargets(setOf(attribute.navigationElement)).setTooltipText("Navigation to target in mapper xml").createLineMarkerInfo(element))}}})}}

总结

本文简单介绍了关于 PSI 文件和元素的基础知识,最后以一个 Mybatis 文件跳转的例子去演示了如何去实际运用 PSI,在下一篇文章则会介绍关于 PSI 的进阶知识,敬请期待~~


http://www.mrgr.cn/p/16868458

相关文章

Django之图形验证码

【1】生成图片验证码依赖于pillow模块 pip install pillow 使用pillow模块在导入时使用import PIL,而不是pillow【1.1】Pillow图像生成模块 from PIL import Image, ImageDraw, ImageFont # Image : 生成图片对象 # ImageDraw : 生成画笔对象 # ImageFont : 控制字体样式# 图片…

BBS项目创作流程

BBS项目创作流程 【零】完整文件gitee仓库BBS/BBS1.0/BlogBasedSystem Lea4ning/DjangoObject - 码云 - 开源中国 (gitee.com)【一】项目基本配置 【1】所需模块 asgiref==3.7.2 beautifulsoup4==4.12.3 certifi==2024.2.2 charset-normalizer==3.3.2 Django==3.2.12 fake-use…

Python shp矢量提取多波段栅格图像的均值、提取采样点数值

目录 一、背景介绍 二、实现 1、取样本点数值 2、取区域均值 一、背景介绍 一般我们在利用Arcgis进行按shp提取多波段的均值或者采样点的数值&#xff0c;这是非常麻烦的。如果图像非常大&#xff0c;当shp的采样点非常多的时候&#xff0c;根本出不了结果。因此&#xff…

BLE --- GAT/GATT

GAT GAT 定义了数据交互的协议(PDU、各种命令),和存储在 server 的各种 attribute。 PDU 格式 命令类型具体命令 具体的命令由 Attribute Opcode 指定atttributeGATTGATT 使用 GAT 进行数据交互,对存储在 server 设备的 attribute 操作

网络安全:Kali Linux 进行SQL注入与XSS漏洞利用

目录 一、实验 1.环境 2.Kali Linux 进行SQL注入 3.Kali Linux 进行XSS漏洞利用 二、问题 1.XSS分类 2.如何修改beef-xss的密码 3.beef-xss 服务如何管理 4.运行beef报错 5.beef 命令的颜色有哪些区别 6.owasp-top-10 有哪些变化 一、实验 1.环境 &#xff08;1&a…

Python 全栈系列236 rabbit_agent搭建

说明 通过rabbit_agent, 以接口方式实现对队列的标准操作&#xff0c;将pika包在微服务内&#xff0c;而不必在太多地方重复的去写。至少在服务端发布消息时&#xff0c;不必再去考虑这些问题。 在分布式任务的情况下&#xff0c;客户端本身会启动一个持续监听队列的客户端服…

Oracle重做日志文件clear logfile与clear unarchived logfile浅析

首先,从v$log动态视图中观察到ARC和STATUS两个字段STATUS:分为CURRENT、ACTIVE和INACTIVE三种,当数据库进程DBWn进行一次写入,脏数据从内存刷写到redo logfile中,这时承载数据写入的redo logfile状态即为CURRENT;而数据从redo logfile拷贝到归档目录下时处于ACTIVE状态,…

第十篇【传奇开心果系列】Python自动化办公库技术点案例示例:深度解读Python自动化操作Excel

传奇开心果博文系列 系列博文目录Python自动化办公库技术点案例示例系列博文目录 前言一、重要作用解说二、Python操作Excel的常用库介绍三、数据处理和分析示例代码四、自动化报表生成示例代码五、数据导入和导出示例代码六、数据可视化示例代码八、数据校验和清洗示例代码九、…

Qt实现简易的多线程TCP服务器(附源码)

目录 一.UI界面的设计 二.服务器的启动 三.实现自定义的TcpServer类 1.在widget中声明自定义TcpServer类的成员变量 2.在TcpServer的构造函数中对于我们声明的m_widget进行初始化&#xff0c;m_widget我们用于后续的显示消息等&#xff0c;说白了就是主界面的更新显示等 …

实验二 数据描述

c语言程序设计——实验报告二 实验项目名称:实验二 数据描述 实验项目类型:验证性 实验日期:2023年3月21日 一、实验目的 1、掌握C语言数据类型,熟悉如何定义一个整型、字符型和实型的变量,以及对它们赋值的方法。 2、掌握不同数据类型之间赋值的规律。 3、学会使用C的有…

三级数据库技术考点(详解!!)

1、 答疑:【解析】分布式数据库系统按不同层次提供的分布透明性有:分片透明性;②位置透明性;③局部映像透明性&#xff0c;位置透明性是指数据分片的分配位置对用户是透明的&#xff0c;用户编写程序时只需 要考虑数据分片情况&#xff0c;不需要了解各分片在各个场地的分配情…

实例、构造函数、原型、原型对象、prototype、__proto__、原型链……

学习原型链和原型对象&#xff0c;不需要说太多话&#xff0c;只需要给你看看几张图&#xff0c;你自然就懂了。 prototype 表示原型对象__proto__ 表示原型 实例、构造函数和原型对象 以 error 举例 图中的 error 表示 axios 抛出的一个错误对象&#xff08;实例&#xff0…

hiveserver2拒绝连接

一、报错内容 二、解决办法 基本都是core-site.xml文件中没做好代理导致的。 在文件中添加如下配置<property><name>hadoop.proxyuser.xxx.hosts</name><value>*</value></property><property><name>hadoop.proxyuser.xxx.gro…

SpringCloud微服务集成Dubbo

1、Dubbo介绍 Apache Dubbo 是一款易用、高性能的 WEB 和 RPC 框架,同时为构建企业级微服务提供服务发现、流量治理、可观测、认证鉴权等能力、工具与最佳实践。用于解决微服务架构下的服务治理与通信问题,官方提供了 Java、Golang 等多语言 SDK 实现。使用 Dubbo 开发的微服…

修改 RabbitMQ 默认超时时间

MQ客户端正常运行&#xff0c;突然就报连接错误&#xff0c; 错误信息写的很明确&#xff0c;是客户端连接超时。 不过很疑虑&#xff0c;为什么会出现连接超时呢&#xff1f;代码没动过&#xff0c;网络也ok&#xff0c;也设置了心跳和重连机制。 最终在官网中找到了答案&am…

vs2022安装和使用教程(详细)

vs2022和vs2019一样强大&#xff0c;C/C&#xff0c;Python&#xff0c;F#&#xff0c;ios&#xff0c;Android&#xff0c;Web&#xff0c;Node.js&#xff0c;Azure&#xff0c;Unity&#xff0c;HTML&#xff0c;JavaScript等开发都可以执行&#xff0c;大家快来使用它吧~ 如…

《自动机理论、语言和计算导论》阅读笔记:p49-p67

《自动机理论、语言和计算导论》学习第4天,p49-p67总结,总计19页。 一、技术总结 1.Deterministic Finite Automata(DFA) vs Nondeterministic Finite Automata(NFA) (1)DFA定义(2)NFA定义A "nonedeterministic" finite automata has the power to be in several s…

HTTPS:原理、使用方法及安全威胁

文章目录 一、HTTPS技术原理1.1 主要技术原理1.2 HTTPS的工作过程1.2.1 握手阶段1.2.2 数据传输阶段 1.3 CA证书的签发流程1.4 HTTPS的安全性 二、HTTPS使用方法三、HTTPS安全威胁四、总结 HTTPS&#xff08;全称&#xff1a;Hyper Text Transfer Protocol over Secure Socket …

如何开始定制你自己的大型语言模型

2023年的大型语言模型领域经历了许多快速的发展和创新,发展出了更大的模型规模并且获得了更好的性能,那么我们普通用户是否可以定制我们需要的大型语言模型呢? 首先你需要有硬件的资源,对于硬件来说有2个路径可以选。高性能和低性能,这里的区别就是是功率,因为精度和消息…

探索跨海大桥新境界:3D可视化技术的魔力

跨海大桥3D可视化,不仅仅是一场技术的革新,更是桥梁建设领域的一次划时代飞跃。跨海大桥3D可视化,不仅仅是一场技术的革新,更是桥梁建设领域的一次划时代飞跃。让我们先来想象一个场景:站在海岸边,望着眼前辽阔的海面,一座雄伟的跨海大桥如巨龙般蜿蜒伸展,连接着两岸。…