自定义类加载器使用geotools读取高程报 ImageRead: No OperationDescriptor is registered 问题

news2024/9/20 22:28:46

背景

项目中使用了 自定义classLoader ,,然后使用下面简化后的代码读取高程数据

public class Test{
	public static void main(String[] args) throwS Exception{
		CustomClassLoader cl = new CustomClassLoader();
		Class<?> clazz = cl.loadClass(“Test”);
		Test test =(Test)clazz.newInstance();
		test.readTiff(cl);
	}

	public void readTiff(){
		//读取基础dem文件
        		File file = new File(“a.tif”);
        		//tiff读取器
        		GeoTiffReader tifReader = new GeoTiffReader(file, tiffHints);
        		//读取tiff,获得基础GridCoverage2D 。
        		GridCoverage2D coverage = tifReader.read(null);
	}
}

此时会报如下错误 ImageRead: No OperationDescriptor is registered in the current operation registry under this name.

注意:项目中 geotools 的相关 jar 包都是由自定义classLoader加载的, AppClassLoader  中并没有加载 geotools 相关 jar 包,这也是问题产生的前提条件。

相关依赖如下所示

 <dependency>
            <groupId>org.geotools</groupId>
            <artifactId>gt-epsg-hsql</artifactId>
            <version>22.0</version>
        </dependency>
        <dependency>
            <groupId>org.geotools</groupId>
            <artifactId>gt-geotiff</artifactId>
            <version>22.0</version>
        </dependency>

解决方案:

  1. 用 AppClassLoader 加载  geotools 相关的jar
  2. 代码改为如下所示,和上面代码区别在于增加了 init 方法
    public class Test{
    	public static void main(String[] args) throwS Exception{
    		CustomClassLoader cl = new CustomClassLoader();
    		Class<?> clazz = cl.loadClass(“Test”);
    		Test test =(Test)clazz.newInstance();
    
    		test.init(cl);
    
    		test.readTiff(cl);
    	}
    
    	boolean firstInit =true;
    	/**
    	* 初始化进行JAI的注册
    	* 此方法会加载ImageReader
    	*/
    	public void init(ClassLoader cl){
    		if(firstInit){
                //先触发 JAI 静态代码块使用 系统类加载器加载render
                JAI.getDefaultInstance();
                //设置当前线程类加载器为自定义类加载器.
                Thread.currentThread().setContextClassLoader(cl);
                //主动调用注册render的方法
    			JAI.getDefaultInstance().getOperationRegistry().registerServices(cl);
    			firstInit=false;
    		}
    	}
    	public void readTiff(){
    		//读取基础dem文件
            		File file = new File(“a.tif”);
            		//tiff读取器
            		GeoTiffReader tifReader = new GeoTiffReader(file, tiffHints);
            		//读取tiff,获得基础GridCoverage2D 。
            		GridCoverage2D coverage = tifReader.read(null);
    	}
    }
    

 原因分析:

首先报错是在如下图所示的地方,此处 render 只加载了 97 个,而实际情况是应该加载 102 ,97中并不包含 ImageReader,于是 getDescriptor 的时候就报错了。调用链路为 GeoTiffReader.read -> JAI.create -> JAI.createNS

render 内容加载是在如下图所示的地方,而这个方法是由JAI中的的静态代码块触发的,没有提供相关图片。静态代码块只在第一次被类加载的时候执行一次

上图中两个序号标识的地方会以不同方式会创建 render

        序号①:这个地方地方会从 jai_core.jar 中的 javax.media.jai.registryFile.jai文件中加载 97个 render ,就是前面提到的 97render

        序号②:这个地方按我项目中正常情况是应该要加载5render的,其中就包含ImagerReader,而背景中的问题代码从这个地方却只加载了 0 个,所以问题就产生在序号②方法里面的逻辑。注意,此处传入的ClassLoader 是 null

 序号②方法里面的逻辑如下图所示

图中有两个方法,也是会以两种不同方式创建 render 

        第1个红框:

        上面提到过此方法传入进来的 classLoader null,于是会走第一个 if,会从系统的 classLoader AppClassLoader 中加载所有 jar 包中的  META-INF/registryFile.jai 文件中的 render ,但前面强调过了,geotools 的相关 jar 包都是通过 自定义classLoader 加 载的,而 META-INF/registryFile.jai  是在 geotools jar 包中,所以系统的 AppClassLoader 自然是加载不到任何的 render 的。

        所以如果要加载到 META-INF/registryFile.jai 中的 render 要么让 AppClassLoader  加载 geotools 的相关 jar 包,要么此处传入的 classLoader  为 自定义classLoader

        不过这个地方加载的 render  不包含  ImagerReader 。所以没加载对本项目中的错误不会有影响。

         第2个红框:

        此处会从当前线程的类加载器读取 spi 中的 render 进行加载。,和上面一样,由于包含 render spi 文件还是在 geotools 相关的 jar 包中,所以此处如果不是 自定义classLoader 依然会加载不到 render 。而如果没有特意设置当前线程的类加载器为 自定义classLoader,则使用的还是 AppClassLoader 

        此处加载的 render 包括 ImageReader ,于是就产生了本文章的错误。

解决思路  (其实就是开头的两种解决方案)

思路1:
         就是让 AppClassLoader  加载 geotools 相关的 jar 包。

思路2:
        设置当前线程的类加载器为
自定义classLoader,然后主动调用JAI.getDefaultInstance().getOperationRegistry().registerServices(自定义classLoader注册 render ,并传入自定义classLoader.这样上面第1个红框和第2个红框就都可以使用 自定义classLoader 加载 render
        需要注意的是在设置当前线程的类加载器为 自定义classLoader 之前,需要先触发 JAI 的类加载,从而执行 JAI 的静态代码块使用 AppClassLoader  进行一次 render 的注册。否则 JAI 的静态代码块注册 render 时,会使用 自定义classLoader ,注册一次 ImageReader,然后主动注册时 ImageReader 又会注册一次,就会报重复注册的错误。

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

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

相关文章

后端代码练习5--验证码案例

我们日常生活中&#xff0c;在进行应用程序注册或者登录的时候&#xff0c;出于安全性的考虑&#xff0c;我们都会被进行一项验证的操作&#xff0c;即通过网页给我们的图片进行一些列的操作&#xff0c;最终完成对我们身份的验证并给我们这些用户返回验证码&#xff0c;让我们…

C语言-有两个磁盘文件A和B,各存放一行字母,今要求把这两个文件的信息合并(按字母顺序排列),输出到一个新文件C中去-深度代码解析

1、题目要求 有两个磁盘文件A和B&#xff0c;各存放一行字母&#xff0c;今要求把这两个文件的信息合并&#xff08;按字母顺序排列&#xff09;&#xff0c;输出到一个新文件C中去 2、准备工作 问题1&#xff1a;为什么不需要手动创建C.txt文件&#xff1f; 答&#xff1a;根…

技术分享-商城篇-订单模块-取消/收货功能(十六)

前言 再上一篇文章技术分享-商城篇-用户订单管理&#xff08;十五) 中&#xff0c;订单模块用户操作含有&#xff1a;取消订单、去支付、确认收货、删除订单、查看详情、去退款、查看物流、再次购买等业务操作&#xff0c;以上的每一个操作&#xff0c;都是对应不同的业务和状…

AudioNotes -将音频内容转 markdown

文章目录 一、关于 AudioNotes效果展示音视频识别和整理与音视频内容对话 二、使用方法1、安装 Ollama2、拉取模型3、部署服务3.1 Docker部署&#xff08;推荐&#xff09;&#x1f433;3.2 本地部署 &#x1f4e6; 一、关于 AudioNotes AudioNotes 能够快速提取音视频的内容&…

贪心处理任务(华为od机考题)

一、题目 1.原题 在某个项目中有多个任务&#xff08;用 tasks 数组表示&#xff09;需要您进行处理&#xff0c; 其中 tasks[i] [si, ei]&#xff0c; 你可以在 si < day < ei 中的任意一天处理该任务。 请返回你可以处理的最大任务数。 注&#xff1a;一天可以完成一…

硬件面试经典 100 题(81~90)题

81、请问下图电路中二极管 D1、D2 有什么作用&#xff1f; 在 Vi 输入电压接近于零时&#xff0c;D1、D2 给三极管 T1、T2 提供偏置电压&#xff0c;使 T1、T2 维持导通&#xff0c;以消除交越失真。 陈氏解释 这道题参见&#xff1a;硬件面试经典 100 题&#xff08;51~70 题…

【学习笔记】STM32F407探索者HAL库开发(三)IO分配

【学习笔记】STM32F407探索者HAL库开发&#xff08;三&#xff09;IO分配 1 STM32F407 IO资源分配表2 STM32F407ZGT6 引脚定义3 IO分配的重要性3.1 硬件设计优化3.2 软件编程3.3 系统性能提升 4 F1/F7/H7芯片的IO分配差异4.1 引脚数量和分组4.2 功能模式4.2.1 输入模式4.2.2 输…

Kubernetes 外部 etcd 集群的快速 Docker Compose 部署指南

一、背景 在高可用 Kubernetes 部署中&#xff0c;需要单独部署外部 etcd 集群&#xff0c;而不是使用 kubeadm 默认在 master 节点上部署的 etcd。以下是关于这一配置场景的详细记录。 二、etcd简介 etcd 是一个高可用的分布式键值存储系统&#xff0c;主要用于存储和管理配…

使用Qt+Visual Stuidio写一个简单的音乐播放器(1)

1.使用QMediaPlayer播放音乐 第三步:在代码头部加上: #include <QtMultimedia/QMediaPlayer> // VS向.pro文件添加代码的方式 #pragma execution_character_set("utf-8") // qt支持显示中文 QMediaPlayer类是一个高级媒体播放类。它可以用来播放歌曲、电…

leetcode 893. Groups of Special-Equivalent Strings

原题链接 You are given an array of strings of the same length words. In one move, you can swap any two even indexed characters or any two odd indexed characters of a string words[i]. Two strings words[i] and words[j] are special-equivalent if after any …

力扣: 设计链表

文章目录 需求代码结尾 需求 你可以选择使用单链表或者双链表&#xff0c;设计并实现自己的链表。 单链表中的节点应该具备两个属性&#xff1a;val 和 next 。val 是当前节点的值&#xff0c;next 是指向下一个节点的指针/引用。 如果是双向链表&#xff0c;则还需要属性 p…

Java 的数组详解

数组的定义 数组是相同类型数据的有序集合 数组描述的是相同类型的若干个数据&#xff0c;按照一定的先后次序排列组合而成 其中&#xff0c;每一个数据称作一个数组元素&#xff0c;每个数组元素可以通过一个下标(编号、标记)来访问它&#xff0c;下标是从 0 开始的&#xf…

100套动画PPT模版分享

100套动画PPT模板 目录下载链接 目录 下载链接 「动画模板」链接&#xff1a;https://pan.quark.cn/s/73ea2523f198 点击下载

中小型企业如何管理文档?8款工具来帮你

文章介绍了以下几个工具&#xff1a;PingCode、Worktile、氚云、泛微、中通天鸿、Tower、知因智慧、SharePoint。 在中小型互联网企业中&#xff0c;文档知识库的管理常常让人头疼。团队成员散布在不同的地点&#xff0c;文档分散在各种工具中&#xff0c;查找信息变得异常困难…

Linux启动流程和Systemd特性

文章目录 内核设计流派linux启动流程1.硬件加电自检2.启动加载器bootloader3.加载kernel4.init初始化5.用户终端启动 systemdsystemd特性systemd的unitunit配置文件 systemctl管理系统服务service unit服务状态 service unit文件格式Unit段Service段Install段 内核设计流派 1.…

资源第二篇:bundle 的config.json 文件内容的解析

简介 本篇文章主要是对bundle包的核心文件config.json 的分析。config.json记录着整个bundle包的具体信息&#xff0c;并通过config.json 去解析整个bundle包。 bundle 目录下的文件结构 import 存放所有的json。场景、预制体、texture2D配置等jsonnative 存放所有的实际资源…

分子属性梯度引导的3D分子生成扩散模型 TAGMOL - 评测

TAGMoL 是一个基于分子属性条件引导扩散的 3D 分子生成模型&#xff0c;适合在给定靶标蛋白质的情况下&#xff0c;可以生成一系列满足目标特性&#xff08;分子属性&#xff0c;binding affinity&#xff09;的候选分子。 一、背景介绍 TAGMoL 来源于新德里 Molecule AI, 以及…

ESP32 出现 failed to load RF 报错

前言 个人邮箱&#xff1a;zhangyixu02gmail.com 正文 周五我测试程序没有问题&#xff0c;板子放在桌子上就没动过了。周一过来的时候&#xff0c;重新烧录程序&#xff0c;就发现出现如下报错。最终发现是电池过放导致电池损坏功率不够&#xff0c;因此 RF 无法启动&#…

探寻少儿自闭症的解决之道

自闭症&#xff0c;又称孤独症谱系障碍&#xff0c;是一种广泛性发育障碍&#xff0c;给无数家庭带来了沉重的负担。然而&#xff0c;随着科学技术的不断进步和人们对自闭症认识的逐步深入&#xff0c;越来越多的方法和途径正在被探索出来&#xff0c;为自闭症的解决带来了希望…

RocketMQ 与 Spring Cloud Stream之事务消息配置

1 引言 RocketMQ的事务消息设计是为了解决分布式系统中数据一致性的问题。在分布式系统中&#xff0c;由于数据可能分布在不同的服务或节点上&#xff0c;因此需要一种机制来确保数据的最终一致性。事务消息通过引入本地事务和消息状态的关联&#xff0c;确保了消息的发送与本…