Java从Tif中抽取最大的那张图进行裁剪成x*y份
之前我有一篇帖子《kfb格式文件转jpg格式》讲述到 kfb ===> tif ===> jpg,但是针对于超大tif中的大图是无法顺利提取的,就算是能顺利提取,试想一下,2G的tif文件,如果能提取处理最大的那张图,并且在不压缩的情况下,jpg大图大小高达1G,前端也是无法顺利加载展示的。
这里我为大家带来了解决方案:将tif中最大的那张图切割成 x * y 份,之后让前端去对每一份进行渲染加载。经过了两周时间,百度或是谷歌的雷都被我踩完了(搜索结果都不怎么靠谱)。多次实验后,总算是成功了。
针对与tif中的最大的那张图的切割,我总结出了切割实现方法,仅供参考。
利用jai-imageio进行处理
package com.lonzh.utils;import javax.imageio.IIOException;
import javax.imageio.ImageIO;
import javax.imageio.ImageReadParam;
import javax.imageio.ImageReader;
import javax.imageio.stream.FileImageInputStream;
import java.awt.*;
import java.awt.image.BufferedImage;
import java.io.File;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;/*** 图片切割** @author DaiHaijiao*/
public class ImgCutUtil {public static void main(String[] args) throws Exception {cut("D:\\病理切片\\教学切片(新)\\BL-01-11 心肌梗死.tif");}/*** @param imageWidth 图片宽度* @param imageHeight 图片高度* @param bisectionNumWidth 宽度方向等分数* @param bisectionNumHeight 高度方向等分数*/private static List<ImgCutUtil.Position> listPosition(int imageWidth, int imageHeight, int bisectionNumWidth, int bisectionNumHeight) {List<ImgCutUtil.Position> list = new ArrayList<>();//每个部分的宽度int partWidth = imageWidth / bisectionNumWidth;//宽方向的余数int remainderWidth = imageWidth % bisectionNumWidth;//每个部分的高度int partHeight = imageHeight / bisectionNumHeight;//高方向的余数int remainderHeight = imageHeight % bisectionNumHeight;for (int i = 0; i < bisectionNumWidth; i++) {for (int j = 0; j < bisectionNumHeight; j++) {//左下角 x 坐标int left = i * partWidth;//左下角 y 坐标int bottom = j * partHeight;//右上角 x 坐标int right = i == bisectionNumWidth - 1 ? (i + 1) * partWidth + remainderWidth : (i + 1) * partWidth;//右上角 y 坐标int top = j == bisectionNumWidth - 1 ? (j + 1) * partHeight + remainderHeight : (j + 1) * partHeight;
// System.out.println("第 " + (i * bisectionNumWidth + j + 1) + " 等分:");
// System.out.println("左下角坐标:(" + left + ", " + bottom + ")");
// System.out.println("右上角坐标:(" + right + ", " + top + ")");list.add(new ImgCutUtil.Position(left, bottom, right, top));}}return list;}public static List<File> cut(String tifPath) {List<File> jpgFileList = new ArrayList<>();File tifFile = new File(tifPath);String outPutDirPath = tifPath.substring(0, tifPath.lastIndexOf(".")) + "_large_img" + File.separator;File outDirPath = new File(outPutDirPath);if (!outDirPath.exists()) {outDirPath.mkdir();}try (FileImageInputStream fis = new FileImageInputStream(tifFile)) {Iterator<ImageReader> readers = ImageIO.getImageReadersByFormatName("TIFF");if (!readers.hasNext()) {throw new IIOException("No suitable image reader found!");}ImageReader reader = readers.next();reader.setInput(fis, true);int w = reader.getWidth(0);int h = reader.getHeight(0);int bisectionNumWidth = 5, bisectionNumHeight = 5;List<Position> positions = listPosition(w, h, bisectionNumWidth, bisectionNumHeight);for (int i = 0; i < positions.size(); i++) {ImageReadParam imageReadParam = new ImageReadParam();Position position = positions.get(i);System.out.println(position.left + "---" + position.bottom + "---" + (position.right - position.left) + "---" + (position.top - position.bottom));imageReadParam.setSourceRegion(new Rectangle(position.left, position.bottom, position.right - position.left, position.top - position.bottom));BufferedImage image = reader.read(0, imageReadParam);File file = new File(outPutDirPath + i + ".jpg");ImageIO.write(image, "jpg", ImageIO.createImageOutputStream(file));jpgFileList.add(file);}} catch (IOException e) {e.printStackTrace();}return jpgFileList;}static class Position {//左下角 x 坐标private Integer left;//左下角 y 坐标private Integer bottom;//右上角 x 坐标private Integer right;//右上角 y 坐标private Integer top;public Position(Integer left, Integer bottom, Integer right, Integer top) {this.left = left;this.bottom = bottom;this.right = right;this.top = top;}@Overridepublic String toString() {return "Position{" +"left=" + left +", bottom=" + bottom +", right=" + right +", top=" + top +'}';}}}
上述代码说明:
1. 代码运行需要引用maven依赖
<dependency><groupId>javax.media.jai</groupId><artifactId>jai-imageio-core</artifactId><version>1.4.0</version></dependency>
有C币的朋友可直接移步jar文件下载传送门(jar引入方式见:我的另一篇文章《本地Maven仓库导入外部jar》)
2. main方法中的测试文件较大,高达1.64G

3. 切割比较吃内存,电脑配置太低时谨慎运行,避免电脑卡死

4. 代码中我是将该tif切割成了 5 * 5 份,切割完后会在该tif文件所在文件内产生一个新的文件夹(里面就是所有切割后的小图)

裁剪后的图片图号如下:

图序如下:

不管是几乘几的切割,图序排列规则都是一样的模式,如下图规则:

如果你对图序感兴趣,或是我的这种图序觉得前端不好使用,你可以再次研究修改listPositioin方法以达到你所需的图序。
5. 计算每一份的点坐标位置时注意:tif的宽和高未必能被要切割成的宽份数和高的份数整除,所以注意余数问题。我的那个listPosition方法中已经对于余数进行了处理,详见那块代码。
6. 代码执行超大tif文件裁剪会报错,报错地方如图:

报错原因就是tif的宽*高的值大于了Integer的最大值。解决办法很简单,报错的这个class文件位于rt.jar下面的javax.imgio下的ImageReader.class,而解决办法就在此。
我们在我们的项目中按照这个包路径创建一个包,并在该包下创建这个类,类里面的内容直接将ImageReader.class的内容拷贝过来,之后修改报错的地方:

修改完后重新编译项目,就会在target下面生产一个ImageReader.class文件。我们只需要替换jar包中的这个class即可。
替换前先关闭idea,找到项目所用到的rt.jar

如果打不开,自己下载个7z就能打开了,之后把我们编译后的那个class拖进去即可完成替换

最后重新打开idea运行代码即可
相关文章:
Java从Tif中抽取最大的那张图进行裁剪成x*y份
之前我有一篇帖子《kfb格式文件转jpg格式》讲述到 kfb > tif > jpg,但是针对于超大tif中的大图是无法顺利提取的,就算是能顺利提取,试想一下,2G的tif文件,如果能提取处理最大的那张图,并且在不压缩的…...
人工智能AI界的龙头企业,炸裂的“英伟达”时代能走多远
原创 | 文 BFT机器人 1、AI芯片的竞争格局已趋白热化 尽管各类具有不同功能和定位的AI芯片在一定程度上可实现互补,但同时也在机遇与挑战并存中持续调整定位。在AI训练端,英伟达的GPU凭着高算力的门槛,一直都是训练端的首选。 只有少数芯片能…...
【实战】H5 页面同时适配 PC 移动端 —— 旋转横屏
文章目录 一、场景二、方案三、书单推荐01 《深入实践Kotlin元编程》02 《Spring Boot学习指南》03 《Kotlin编程实战》 一、场景 一个做数据监控的单页面,页面主要内容是一个整体必须是宽屏才能正常展示,这时就不能用传统的适配方案了,需要…...
使用凌鲨进行聚合搜索
作为研发人员,我们经常需要在多个来源之间查找信息,以便进行研发工作。除了常用的搜索引擎如百度和必应之外,我们还需要查阅各种代码文档和依赖包等资源。这些资源通常分散在各个网站和文档库中,需要花费一定的时间和精力才能找到…...
程序设计之——手把手教你如何从Excel文件中读取学生信息
在当今信息化时代,计算机技术已经深入到各个领域,而程序设计则成为推动信息化建设的关键技术之一。在众多领域中,学生信息管理系统无疑是其中一个重要的应用。本文将从学生信息管理系统的开发入手,探讨开如何高效且保证质量的完成…...
Docker容器化技术(从零学会Docker)
文章目录 前言一、初识Docker1.初识Docker-Docker概述2.初识Docker-安装Docker3.初识Docker-Docker架构4.初识Docker-配置镜像加速器 二、Docker命令1.Docker命令-服务相关命令2.Docker命令-镜像相关命令3.Docker命令-容器相关命令 三、Docker容器的数据卷1.Docker容器数据卷-数…...
【新版】系统架构设计师 - 案例分析 - 总览
个人总结,仅供参考,欢迎加好友一起讨论 架构 - 案例分析 - 总览 新旧大纲对应 旧版新版系统规划软件架构设计设计模式系统设计系统建模分布式系统设计嵌入式系统设计系统的可靠性分析与设计系统的安全性和保密性设计系统计划信息系统架构的设计理论和实…...
【Git】02-Git常见应用
文章目录 1. 删除不需要分支2. 修改最新Commit的Message3. 修改之前Commit的Message4. 连续多个Commit整理为一个5. 不连续的Commit整理为一个6. 比较暂存区和HEAD中文件差异7. 比较工作区和暂存区中文件差异8. 将暂存区恢复为HEAD相同9. 工作区文件恢复和暂存区相同10. 取消暂…...
YOLO物体检测-系列教程2:YOLOV2整体解读
🎈🎈🎈YOLO 系列教程 总目录 YOLOV1整体解读 YOLOV2整体解读 YOLOV2提出论文:YOLO9000: Better, Faster, Stronger 1、YOLOV1 优点:快速,简单!问题1:每个Cell只预测一个类别&…...
u盘传输数据的时候拔出会怎么样?小心这些危害
U盘是我们日常生活和工作中常使用的一种便携式存储设备。然而,在使用U盘传输数据时,有时我们会不小心将它拔出,而这个看似微不足道的行为实际上可能会带来严重的后果。本文将向您介绍U盘在传输数据时突然拔出可能导致的各种危害,其…...
【踩坑纪实】URL 特殊字符 400 异常
URL 特殊字符 400 异常 笔者之前在写后端或者前端时,在处理表单时,经常有对特殊字符的检验处理,但自己也不清楚为什么要这么做,浅浅地以为可能是特殊字符不好看或者存取可能会造成异常?不过一直没遇到过问题ÿ…...
Contents:帮助公司为营销目的创建内容
【产品介绍】 名称 Contents上线时间 2017年5月 具体描述 Contents是一家提供基于人工智能的内容生成平台的企业,可以帮助用户在各种网站和工具中使用最先进的机器学习模型,实现视频编辑、图像生成、3D建模等内容创作。【团队介绍…...
1397: 图的遍历——广度优先搜索
题目描述 广度优先搜索遍历类似于树的按层次遍历的过程。其过程为:假设从图中的某顶点v出发,在访问了v之后依次访问v的各个未曾被访问过的邻接点,然后分别从这些邻接点出发依次访问它们的邻接点,并使“先被访问的顶点的邻接点”先…...
Java 华为真题-选修课
需求: 现有两门选修课,每门选修课都有一部分学生选修,每个学生都有选修课的成绩,需要你找出同时选修了两门选修课的学生,先按照班级进行划分,班级编号小的先输出,每个班级按照两门选修课成绩和的…...
Invalid access token: Invalid header string: ‘utf-8‘ codec can‘t decode byte
报错:在运行一个txt文档时报Invalid access token: Invalid header string: ‘utf-8’ codec can’t decode byte 原因:文档编码方式的原因,电脑默认的是UFT-8格式的编码 解决方法:用notepad改一下文档编码就好...
Java 中将多个 PDF 文件合并为一个 PDF
一.前言 我们将从以下两个方面向您展示如何将多个PDF文件合并为一个PDF: 1. 将文件中的多个 PDF 合并为单个 PDF 2. 将流中的多个 PDF 合并为单个 PDF 1. 了解 Spire.PDF 库 要在 Java 中合并 PDF 文件,我们将使用Spire.PDF 库。Spire.PDF for Java 是…...
python经典百题之水仙花数
题目:打印出所有的“水仙花数”,所谓“水仙花数”是指一个三位数,其各位数字立方和等于该数 本身。例如:153是一个“水仙花数”,因为1531的三次方+5的三次方+3的三次方。 方法一:暴…...
jvm的调优工具
1. jps 查看进程信息 2. jstack 查看进程的线程 59560为进程id 产生了死锁就可以jstack查看了 详细用途可以看用途 3. jmap 如何使用dump文件看下 查看 4.jstat 空间占用和次数 5. jconsole可视化工具 各种使用情况,以及死锁检测 6. visualvm可视化工具…...
C语言--字符串旋转笔试题
C语言–字符串旋转笔试题 文章目录 C语言--字符串旋转笔试题一、字符串左旋1.1 思路11.2 思路1代码1.3 思路21.4 思路2代码 二、字符串旋转结果判断2.1 思路12.2 思路2 一、字符串左旋 实现一个函数,可以左旋字符串中的k个字符。 例如: ABCD左旋一个字…...
IntelliJ IDEA使用_常规设置
文章目录 版本说明主题设置取消检查更新依赖自动导入禁止import xxx.*、允许import内部类显示行号、方法分割线、空格代码提示(匹配所有字母)自定义注释颜色添加头部注释自定义字体设置字符编码关联本地GitJDK编译版本Maven配置Tomcat配置代码注释设置头…...
eNSP-Cloud(实现本地电脑与eNSP内设备之间通信)
说明: 想象一下,你正在用eNSP搭建一个虚拟的网络世界,里面有虚拟的路由器、交换机、电脑(PC)等等。这些设备都在你的电脑里面“运行”,它们之间可以互相通信,就像一个封闭的小王国。 但是&#…...
linux arm系统烧录
1、打开瑞芯微程序 2、按住linux arm 的 recover按键 插入电源 3、当瑞芯微检测到有设备 4、松开recover按键 5、选择升级固件 6、点击固件选择本地刷机的linux arm 镜像 7、点击升级 (忘了有没有这步了 估计有) 刷机程序 和 镜像 就不提供了。要刷的时…...
Qwen3-Embedding-0.6B深度解析:多语言语义检索的轻量级利器
第一章 引言:语义表示的新时代挑战与Qwen3的破局之路 1.1 文本嵌入的核心价值与技术演进 在人工智能领域,文本嵌入技术如同连接自然语言与机器理解的“神经突触”——它将人类语言转化为计算机可计算的语义向量,支撑着搜索引擎、推荐系统、…...
C++ 基础特性深度解析
目录 引言 一、命名空间(namespace) C 中的命名空间 与 C 语言的对比 二、缺省参数 C 中的缺省参数 与 C 语言的对比 三、引用(reference) C 中的引用 与 C 语言的对比 四、inline(内联函数…...
【C++从零实现Json-Rpc框架】第六弹 —— 服务端模块划分
一、项目背景回顾 前五弹完成了Json-Rpc协议解析、请求处理、客户端调用等基础模块搭建。 本弹重点聚焦于服务端的模块划分与架构设计,提升代码结构的可维护性与扩展性。 二、服务端模块设计目标 高内聚低耦合:各模块职责清晰,便于独立开发…...
大学生职业发展与就业创业指导教学评价
这里是引用 作为软工2203/2204班的学生,我们非常感谢您在《大学生职业发展与就业创业指导》课程中的悉心教导。这门课程对我们即将面临实习和就业的工科学生来说至关重要,而您认真负责的教学态度,让课程的每一部分都充满了实用价值。 尤其让我…...
OPenCV CUDA模块图像处理-----对图像执行 均值漂移滤波(Mean Shift Filtering)函数meanShiftFiltering()
操作系统:ubuntu22.04 OpenCV版本:OpenCV4.9 IDE:Visual Studio Code 编程语言:C11 算法描述 在 GPU 上对图像执行 均值漂移滤波(Mean Shift Filtering),用于图像分割或平滑处理。 该函数将输入图像中的…...
七、数据库的完整性
七、数据库的完整性 主要内容 7.1 数据库的完整性概述 7.2 实体完整性 7.3 参照完整性 7.4 用户定义的完整性 7.5 触发器 7.6 SQL Server中数据库完整性的实现 7.7 小结 7.1 数据库的完整性概述 数据库完整性的含义 正确性 指数据的合法性 有效性 指数据是否属于所定…...
2025年渗透测试面试题总结-腾讯[实习]科恩实验室-安全工程师(题目+回答)
安全领域各种资源,学习文档,以及工具分享、前沿信息分享、POC、EXP分享。不定期分享各种好玩的项目及好用的工具,欢迎关注。 目录 腾讯[实习]科恩实验室-安全工程师 一、网络与协议 1. TCP三次握手 2. SYN扫描原理 3. HTTPS证书机制 二…...
Webpack性能优化:构建速度与体积优化策略
一、构建速度优化 1、升级Webpack和Node.js 优化效果:Webpack 4比Webpack 3构建时间降低60%-98%。原因: V8引擎优化(for of替代forEach、Map/Set替代Object)。默认使用更快的md4哈希算法。AST直接从Loa…...
