欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java實現(xiàn)銳化圖片并保存功能(附源碼)

 更新時間:2025年05月15日 08:50:51   作者:Katie。  
在圖像處理領(lǐng)域,銳化(Sharpening)?是一種常見的操作,用于增強圖像中邊緣和細節(jié),使圖像看起來更清晰,下面小編就來介紹一下如何使用Java?SE?原生?API實現(xiàn)對圖像的銳化處理并保存為常見格式文件吧

一、項目背景詳細介紹

在圖像處理領(lǐng)域,銳化(Sharpening) 是一種常見的操作,用于增強圖像中邊緣和細節(jié),使圖像看起來更清晰、更有層次感。與模糊(Blur)相反,銳化通過加強局部像素的對比度來突出紋理和輪廓,常用于攝影后期、醫(yī)學(xué)圖像增強、工業(yè)檢測等場景。

攝影與后期:對照片進行微調(diào),增強主體邊緣,提升視覺沖擊力。

醫(yī)學(xué)影像:突出組織邊界,輔助診斷。

工業(yè)檢測:加深裂紋或瑕疵輪廓,便于自動識別。

教學(xué)與演示:理解圖像卷積和濾波器的原理。

本項目將使用 Java SE 原生 API(BufferedImage、ConvolveOp、Kernel、ImageIO),實現(xiàn)對圖像的銳化處理,并保存為常見格式文件。無需任何第三方依賴,跨平臺、易集成。

二、項目需求詳細介紹

2.1 功能需求

1.讀取圖像

  • 支持 JPEG、PNG、BMP、GIF 等常見格式;
  • 文件不存在或無法解析時給出清晰提示。

2.圖像銳化

  • 采用經(jīng)典的拉普拉斯算子、**Unsharp Mask(反銳化掩模)**或 高通濾波 方法;
  • 參數(shù)化控制:銳化強度(卷積核權(quán)重或掩模比例)。

3.保存輸出

  • 將銳化后的 BufferedImage 寫出為指定格式(PNG/JPEG),支持 JPEG 質(zhì)量參數(shù);
  • 自動創(chuàng)建輸出目錄。

4.批量處理

  • 支持對目錄內(nèi)所有圖片批量銳化;
  • 保持相對目錄結(jié)構(gòu)輸出;
  • 并行執(zhí)行提升效率。

5.用戶交互(可選)

  • 命令行模式:簡單參數(shù)調(diào)用;
  • Swing 界面:預(yù)覽原圖與處理結(jié)果。

6.日志與錯誤處理

  • 對每張圖像的處理結(jié)果記錄日志;
  • 捕獲并提示 I/O、處理異常。

2.2 非功能需求

零依賴:僅用 Java 標準庫。

易用性:封裝成工具類,命令行或 GUI 均可調(diào)用。

性能:對大圖或批量場景采用多線程或分塊處理。

可擴展:易于替換銳化算法或添加新濾鏡。

三、相關(guān)技術(shù)詳細介紹

1.BufferedImage

用于在內(nèi)存中表示可讀寫圖像,支持多種像素格式。

2.Kernel & ConvolveOp

  • Kernel 表示卷積核矩陣;
  • ConvolveOp 實現(xiàn)對 BufferedImage 的卷積操作,用于各種濾波。

3.拉普拉斯算子(Laplacian)

常見 3×3 核:

0  -1   0  
-1  5  -1  
0  -1   0

中心權(quán)重>1,周圍為負值,增強邊緣。

4.Unsharp Mask(反銳化掩模)

先對原圖做 Gaussian Blur 得到平滑圖,

再用原圖與平滑圖的差值增強細節(jié):

sharpen = original + amount × (original - blurred)。

5.ImageIO & ImageWriter

讀取寫出多種格式;

JPEG 支持質(zhì)量設(shè)置。

6.ExecutorService

并行批量處理提升效率。

7.Swing(可選)

JFileChooser、JSlider 等用于參數(shù)調(diào)整和實時預(yù)覽。

四、實現(xiàn)思路詳細介紹

讀取源圖

BufferedImage src = ImageIO.read(new File(srcPath));

選擇銳化算法

  • 拉普拉斯卷積:直接用 ConvolveOp;
  • Unsharp Mask:自定義 Gaussian Blur + 差值運算。

拉普拉斯卷積實現(xiàn)

float[] lap = {0,-1,0, -1,5,-1, 0,-1,0};
Kernel kernel = new Kernel(3,3,lap);
ConvolveOp op = new ConvolveOp(kernel, ConvolveOp.EDGE_NO_OP, null);
BufferedImage sharp = op.filter(src, null);

Unsharp Mask 實現(xiàn)

// 1. Gaussian Blur
float[] gauss = createGaussianKernel(radius);
Kernel gk = new Kernel(k, k, gauss);
BufferedImage blurred = new ConvolveOp(gk).filter(src, null);
// 2. 差值并增強
for each pixel:
  int orig = src.getRGB(x,y);
  int b = blurred.getRGB(x,y);
  int diff = orig - b;
  int result = clamp(orig + amount * diff);

保存輸出

  • 自動根據(jù)擴展名選 PNG 或 JPEG;
  • JPEG 可設(shè)質(zhì)量:
ImageWriter writer = ImageIO.getImageWritersByFormatName("jpg").next();
ImageWriteParam param = writer.getDefaultWriteParam();
param.setCompressionMode(MODE_EXPLICIT);
param.setCompressionQuality(q);
writer.write(null,new IIOImage(sharp,null,null),param);

批量與并發(fā)

  • 遍歷目錄收集所有圖片;
  • 使用 ExecutorService 并行調(diào)用上述流程;
  • 保持相對路徑輸出。

GUI 預(yù)覽

  • 用 JSlider 控制強度參數(shù);
  • 實時 BufferedImage 更新到 JLabel。

五、完整實現(xiàn)代碼

import java.awt.image.*;
import java.io.*;
import java.nio.file.*;
import java.util.*;
import java.util.concurrent.*;
import javax.imageio.*;
import javax.imageio.stream.ImageOutputStream;
 
/**
 * ImageSharpenUtil:圖像銳化工具
 */
public class ImageSharpenUtil {
    // 默認參數(shù)
    private static final double UNSHARP_AMOUNT = 1.0;   // 反銳化掩模強度
    private static final int GAUSS_RADIUS = 3;         // Gaussian 模糊半徑
    private static final float JPEG_QUALITY = 0.9f;    // JPEG 輸出質(zhì)量
    private static final int THREADS = Runtime.getRuntime().availableProcessors();
 
    public static void main(String[] args) throws Exception {
        if (args.length < 2) {
            System.out.println("用法: java ImageSharpenUtil <src> <dest> [laplacian|unsharp]");
            return;
        }
        Path src = Paths.get(args[0]), dest = Paths.get(args[1]);
        String mode = args.length >= 3 ? args[2] : "laplacian";
        if (Files.isDirectory(src)) {
            batchProcess(src, dest, mode);
        } else {
            processSingle(src.toFile(), dest.toFile(), mode);
        }
        System.out.println("完成。");
    }
 
    // 單圖處理
    public static void processSingle(File in, File out, String mode) throws IOException {
        BufferedImage src = ImageIO.read(in);
        if (src == null) throw new IOException("無法讀取: " + in);
        BufferedImage result = "unsharp".equals(mode)
            ? unsharpMask(src, GAUSS_RADIUS, UNSHARP_AMOUNT)
            : laplacianSharpen(src);
        writeImage(result, out);
        System.out.println("處理: " + in);
    }
 
    // 批量處理
    public static void batchProcess(Path srcDir, Path destDir, String mode) throws IOException, InterruptedException {
        List<Path> list = new ArrayList<>();
        try (Stream<Path> s = Files.walk(srcDir)) {
            s.filter(Files::isRegularFile)
             .filter(p->p.toString().matches(".*\\.(?i)(png|jpe?g|bmp|gif)$"))
             .forEach(list::add);
        }
        ExecutorService exec = Executors.newFixedThreadPool(THREADS);
        for (Path p : list) {
            exec.submit(() -> {
                try {
                    Path rel = srcDir.relativize(p);
                    File out = destDir.resolve(rel).toFile();
                    out.getParentFile().mkdirs();
                    processSingle(p.toFile(), out, mode);
                } catch (Exception e) {
                    System.err.println("失敗: " + p + " " + e.getMessage());
                }
            });
        }
        exec.shutdown();
        exec.awaitTermination(1, TimeUnit.HOURS);
    }
 
    // 拉普拉斯銳化
    public static BufferedImage laplacianSharpen(BufferedImage src) {
        float[] lap = {0,-1,0, -1,5,-1, 0,-1,0};
        Kernel kernel = new Kernel(3,3,lap);
        return new ConvolveOp(kernel, ConvolveOp.EDGE_NO_OP, null).filter(src, null);
    }
 
    // 反銳化掩模
    public static BufferedImage unsharpMask(BufferedImage src, int radius, double amount) {
        int size = radius * 2 + 1;
        float[] matrix = createGaussianKernel(radius);
        Kernel gk = new Kernel(size, size, matrix);
        BufferedImage blurred = new ConvolveOp(gk, ConvolveOp.EDGE_NO_OP, null).filter(src, null);
        BufferedImage dest = new BufferedImage(src.getWidth(), src.getHeight(), src.getType());
        for (int y=0; y<src.getHeight(); y++){
            for (int x=0; x<src.getWidth(); x++){
                int rgb1 = src.getRGB(x,y), rgb2 = blurred.getRGB(x,y);
                int a = (rgb1>>24)&0xFF;
                int r = clamp(((rgb1>>16&0xFF) - (rgb2>>16&0xFF)) * amount + (rgb1>>16&0xFF));
                int g = clamp(((rgb1>>8&0xFF) - (rgb2>>8&0xFF)) * amount + (rgb1>>8&0xFF));
                int b = clamp(((rgb1&0xFF) - (rgb2&0xFF)) * amount + (rgb1&0xFF));
                dest.setRGB(x,y, (a<<24)|(r<<16)|(g<<8)|b );
            }
        }
        return dest;
    }
 
    // 生成 Gaussian 核
    private static float[] createGaussianKernel(int radius) {
        int size = radius*2+1;
        float[] data = new float[size*size];
        double sigma = radius/3.0;
        double twoSigmaSq = 2*sigma*sigma;
        double piSigma = Math.PI * twoSigmaSq;
        double sum = 0;
        for(int y=-radius; y<=radius; y++){
            for(int x=-radius; x<=radius; x++){
                double v = Math.exp(-(x*x+y*y)/twoSigmaSq)/piSigma;
                data[(y+radius)*size + (x+radius)] = (float)v;
                sum += v;
            }
        }
        for(int i=0;i<data.length;i++) data[i] /= sum;
        return data;
    }
 
    // 寫出圖片
    public static void writeImage(BufferedImage img, File out) throws IOException {
        String ext = getExt(out.getName());
        out.getParentFile().mkdirs();
        if ("jpg".equalsIgnoreCase(ext)||"jpeg".equalsIgnoreCase(ext)) {
            Iterator<ImageWriter> it = ImageIO.getImageWritersByFormatName("jpg");
            ImageWriter w = it.next();
            try (ImageOutputStream ios = ImageIO.createImageOutputStream(out)) {
                w.setOutput(ios);
                ImageWriteParam p = w.getDefaultWriteParam();
                if (p.canWriteCompressed()) {
                    p.setCompressionMode(ImageWriteParam.MODE_EXPLICIT);
                    p.setCompressionQuality(JPEG_QUALITY);
                }
                w.write(null, new IIOImage(img,null,null),p);
            } finally { w.dispose(); }
        } else {
            ImageIO.write(img, ext, out);
        }
    }
 
    private static int clamp(double v) {
        return v<0?0:(v>255?255:(int)v);
    }
    private static String getExt(String name) {
        int i=name.lastIndexOf('.'); return i<0?"png":name.substring(i+1);
    }
}

六、代碼詳細解讀

1.main:解析命令行參數(shù),判斷單圖或批量模式,并選擇算法模式(拉普拉斯或反銳化掩模)。

2.laplacianSharpen:使用拉普拉斯 3×3 卷積核,通過 ConvolveOp 實現(xiàn)一行代碼銳化。

3.unsharpMask:

  • 調(diào)用 createGaussianKernel 生成高斯模糊核,模糊原圖。
  • 逐像素計算原圖與模糊圖之差,按 amount 放大后與原圖疊加,增強細節(jié)。

4.createGaussianKernel:根據(jù)給定半徑和 σ 值計算高斯分布核并歸一化。

5.writeImage:根據(jù)輸出文件擴展名自動選擇 PNG 或 JPEG;后者設(shè)置壓縮質(zhì)量。

6.batchProcess:使用 ExecutorService 并行遍歷目錄并處理,保持相對路徑輸出。

七、項目詳細總結(jié)

本項目實現(xiàn)了兩種經(jīng)典銳化算法:拉普拉斯算子 與 Unsharp Mask。核心特點:

純 Java SE:無需額外庫,方便集成。

多模式:算法可選,批量與單圖靈活。

參數(shù)化:可配置高斯半徑、掩模強度、JPEG 質(zhì)量。

高效并行:多線程批量處理。

可作為圖像處理入門示例,也可用于生產(chǎn)環(huán)境的自動化銳化管道。

八、項目常見問題及解答

Q1:拉普拉斯核會產(chǎn)生噪點或過度銳化?

A:中心權(quán)重過大或圖像噪聲明顯時,建議先做輕度高斯模糊再銳化,或使用 Unsharp Mask 控制更細膩。

Q2:反銳化掩模處理速度慢?

A:可降低模糊半徑或使用分塊并行;也可使用更小的 kernel。

Q3:輸出 JPEG 質(zhì)量差異大?

A:JPEG 為有損壓縮,建議將 JPEG_QUALITY 設(shè)置在 0.8–0.95 之間,或使用 PNG。

Q4:批量處理時內(nèi)存增長?

A:確保及時釋放 BufferedImage 引用,并合理設(shè)置線程數(shù),避免同時加載過多大圖。

九、擴展方向與性能優(yōu)化

GPU 加速:使用 OpenCL(JOCL)或 CUDA 版濾波器庫,大幅提升大圖處理速度。

分塊渲染:對超大圖像切塊處理,降低內(nèi)存峰值。

更高級算子:集成 Canny 邊緣檢測 + 導(dǎo)向中值濾波,實現(xiàn)更精準銳化。

實時 GUI:基于 JavaFX 或 Swing 加入滑桿、預(yù)覽、批量任務(wù)管理。

流水線處理:結(jié)合其他濾鏡(去噪、色彩校正)構(gòu)建完整圖像管道。

以上就是Java實現(xiàn)銳化圖片并保存功能(附源碼)的詳細內(nèi)容,更多關(guān)于Java銳化圖片的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Spring Boot實現(xiàn)郵件服務(wù)(附:常見郵箱的配置)

    Spring Boot實現(xiàn)郵件服務(wù)(附:常見郵箱的配置)

    這篇文章主要給大家介紹了關(guān)于Spring Boot實現(xiàn)郵件服務(wù)的相關(guān)資料,文中還附上了常見郵箱的配置,通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2018-12-12
  • Spring Boot 搭建 ELK正確看日志的配置流程

    Spring Boot 搭建 ELK正確看日志的配置流程

    這篇文章主要介紹了Spring Boot 搭建 ELK正確看日志的配置流程,本文給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-03-03
  • Java對象類型的判斷詳解

    Java對象類型的判斷詳解

    這篇文章主要介紹了Java對象類型的判斷,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-04-04
  • 搭建MyBatis開發(fā)環(huán)境及基本的CURD介紹

    搭建MyBatis開發(fā)環(huán)境及基本的CURD介紹

    這篇文章主要介紹了搭建MyBatis開發(fā)環(huán)境及基本的CURD,本文給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-08-08
  • SpringBoot?整合?Spring-Session?實現(xiàn)分布式會話項目實戰(zhàn)

    SpringBoot?整合?Spring-Session?實現(xiàn)分布式會話項目實戰(zhàn)

    本文主要介紹了SpringBoot?整合?Spring-Session?實現(xiàn)分布式會話項目實戰(zhàn),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-07-07
  • 淺談@RequestBody和@RequestParam可以同時使用

    淺談@RequestBody和@RequestParam可以同時使用

    這篇文章主要介紹了@RequestBody和@RequestParam可以同時使用,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-03-03
  • Java判斷字符串是否含有亂碼實例代碼

    Java判斷字符串是否含有亂碼實例代碼

    本文通過實例代碼給大家介紹了Java判斷字符串是否含有亂碼的方法,代碼簡單易懂,非常不錯,具有一定的參考借鑒價值,需要的朋友參考下吧
    2018-11-11
  • IDEA中的.iml文件和.idea文件夾

    IDEA中的.iml文件和.idea文件夾

    這篇文章主要介紹了IDEA中的.iml文件和.idea文件夾,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-10-10
  • Java深入淺出掌握SpringBoot之MVC自動配置原理篇

    Java深入淺出掌握SpringBoot之MVC自動配置原理篇

    在進行項目編寫前,我們還需要知道一個東西,就是SpringBoot對我們的SpringMVC還做了哪些配置,包括如何擴展,如何定制,只有把這些都搞清楚了,我們在之后使用才會更加得心應(yīng)手
    2021-10-10
  • java中的Io(input與output)操作總結(jié)(四)

    java中的Io(input與output)操作總結(jié)(四)

    前面已經(jīng)把java io的主要操作講完了,這一節(jié)我們來說說關(guān)于java io的其他內(nèi)容:Serializable序列化/DataOutputStream和DataInputStream類/管道流等等,感興趣的朋友可以了解下
    2013-01-01

最新評論