」工欲善其事,必先利其器。「—孔子《論語.錄靈公》
首頁 > 程式設計 > OpenCV 影像壓縮完整指南

OpenCV 影像壓縮完整指南

發佈於2024-11-08
瀏覽:380

图像压缩是计算机视觉中的一项关键技术,它使我们能够更有效地存储和传输图像,同时保持视觉质量。理想情况下,我们希望拥有最佳质量的小文件。然而,我们必须做出权衡并决定哪个更重要。

本教程将教授使用 OpenCV 进行图像压缩,涵盖理论和实际应用。最后,您将了解如何为计算机视觉项目(或您可能拥有的任何其他项目)成功压缩照片。

什么是图像压缩?

图像压缩正在减小图像的文件大小,同时保持可接受的视觉质量水平。压缩主要有两种类型:

  1. 无损压缩:保留所有原始数据,允许精确的图像重建。
  2. 有损压缩: 丢弃一些数据以获得更小的文件大小,可能会降低图像质量。

为什么要压缩图像?

如果正如我们经常听到的那样“磁盘空间很便宜”,那么为什么还要压缩图像呢?在小范围内,图像压缩并不重要,但在大范围内,它至关重要。

例如,如果您的硬盘上有一些图像,您可以压缩它们并保存几兆字节的数据。当硬盘驱动器以 TB 为单位时,这不会产生太大影响。但如果您的硬盘上有 100,000 张图像怎么办?一些基本的压缩可以节省实时时间和金钱。从性能的角度来看,是一样的。如果您的网站包含大量图像,并且每天有 10,000 人访问您的网站,那么压缩就很重要。

这就是我们这样做的原因:

  • 减少存储要求:在同一空间中存储更多图像
  • 更快的传输:非常适合Web应用程序和带宽受限的场景
  • 提高处理速度:较小的图像加载和处理速度更快

图像压缩背后的理论

图像压缩技术利用两种类型的冗余:

  1. 空间冗余:相邻像素之间的相关性
  2. 颜色冗余:相邻区域颜色值的相似度

空间冗余利用了相邻像素在大多数自然图像中往往具有相似值的事实。这会产生平滑的过渡。许多照片“看起来很真实”,因为从一个区域到另一个区域有一种自然的流动。当相邻像素具有截然不同的值时,您会得到“嘈杂”的图像。像素发生了变化,通过将像素分组为单一颜色,使这些过渡变得不那么“平滑”,从而使图像更小。

The Complete Guide to Image Compression with OpenCV

另一方面,

颜色冗余重点关注图像中的相邻区域如何经常共享相似的颜色。想象一下蓝天或绿地——图像的大部分可能具有非常相似的颜色值。它们也可以组合在一起并制成单一颜色以节省空间。

The Complete Guide to Image Compression with OpenCV

OpenCV 提供了用于处理这些想法的可靠工具。例如,OpenCV 的 cv2.inpaint() 函数利用空间冗余,使用附近像素的信息填充图片中缺失或损坏的区域。 OpenCV 允许开发人员使用 cv2.cvtColor() 在多个关于颜色冗余的颜色空间之间转换图像。这作为许多压缩技术中的预处理步骤可能会有所帮助,因为某些颜色空间在编码特定类型的图像时比其他颜色空间更有效。

我们现在将测试这个理论的一些内容。我们来玩一下吧。

动手实践图像压缩

让我们探索如何使用 OpenCV 的 Python 绑定来压缩图像。写出此代码或复制它:

您也可以在这里获取源代码

import cv2
import numpy as np

def compress_image(image_path, quality=90):
    # Read the image
 img = cv2.imread(image_path)
    
    # Encode the image with JPEG compression
 encode_param = [int(cv2.IMWRITE_JPEG_QUALITY), quality]
 _, encoded_img = cv2.imencode('.jpg', img, encode_param)
    
    # Decode the compressed image
 decoded_img = cv2.imdecode(encoded_img, cv2.IMREAD_COLOR)
    
    return decoded_img

# Example usage
original_img = cv2.imread('original_image.jpg')
compressed_img = compress_image('original_image.jpg', quality=50)

# Display results
cv2.imshow('Original', original_img)
cv2.imshow('Compressed', compressed_img)
cv2.waitKey(0)
cv2.destroyAllWindows()

# Calculate compression ratio
original_size = original_img.nbytes
compressed_size = compressed_img.nbytes
compression_ratio = original_size / compressed_size
print(f"Compression ratio: {compression_ratio:.2f}")

此示例包含一个 compress_image 函数,该函数采用两个参数:

  • 图片路径(图片所在的位置)
  • 质量(所需图像的质量)

然后,我们将原始图像加载到original_img中。然后,我们将同一图像压缩 50% 并将其加载到新实例压缩图像中。

然后我们将显示原始图像和压缩图像,以便您可以并排查看它们。

然后我们计算并显示压缩比。

此示例演示如何在 OpenCV 中使用 JPEG 压缩来压缩图像。质量参数控制文件大小和图像质量的权衡。

让我们运行它:

The Complete Guide to Image Compression with OpenCV

最初查看图像时,您发现几乎没有什么区别。但是,放大后您会看到质量的差异:

The Complete Guide to Image Compression with OpenCV

关闭窗口并查看文件后,我们可以看到文件的大小急剧减小:

The Complete Guide to Image Compression with OpenCV

另外,如果我们进一步降低,我们可以将质量更改为10%

compressed_img = compress_image('sampleimage.jpg', quality=10)

结果更加剧烈:

The Complete Guide to Image Compression with OpenCV

文件大小结果也更加剧烈:

The Complete Guide to Image Compression with OpenCV

您可以非常轻松地调整这些参数,并在质量和文件大小之间实现所需的平衡。

评估压缩质量

为了评估压缩的影响,我们可以使用以下指标:

  1. 均方误差 (MSE)

均方误差 (MSE) 衡量两幅图像之间的差异程度。当您压缩图像时,MSE 可以帮助您确定压缩图像与原始图像相比发生了多少变化。

它通过对两个图像中相应像素的颜色之间的差异进行采样、对这些差异进行平方并取平均值来实现此目的。结果是一个数字:较低的 MSE 意味着压缩图像更接近原始图像。相比之下,更高的 MSE 意味着质量损失更明显。

这里有一些Python代码来衡量:

def calculate_mse(img1, img2):
    return np.mean((img1 - img2) ** 2)

mse = calculate_mse(original_img, compressed_img)
print(f"Mean Squared Error: {mse:.2f}")

这是我们的演示图像压缩的样子:

The Complete Guide to Image Compression with OpenCV

  1. 峰值信噪比 (PSNR)

峰值信噪比 (PSNR) 是一种衡量图像质量在压缩后下降程度的指标。这通常是肉眼可见的,但它指定了一个设定值。它将原始图像与压缩图像进行比较,并将差异表示为比率。

PSNR 值越高,意味着压缩后的图像质量更接近原始图像,表明质量损失更少。 PSNR 越低意味着退化越明显。 PSNR 通常与 MSE 一起使用,PSNR 提供了一个更易于解释的量表,其中越高越好。

下面是一些用于测量的 Python 代码:

def calculate_psnr(img1, img2):
 mse = calculate_mse(img1, img2)
    if mse == 0:
        return float('inf')
 max_pixel = 255.0
    return 20 * np.log10(max_pixel / np.sqrt(mse))

psnr = calculate_psnr(original_img, compressed_img)
print(f"PSNR: {psnr:.2f} dB")

这是我们的演示图像压缩的样子:

The Complete Guide to Image Compression with OpenCV

压缩后“观察”图像以确定质量是否良好;然而,在大规模上,让脚本执行此操作是设置标准并确保图像遵循这些标准的更简单的方法。

让我们看看其他一些技巧:

先进的压缩技术

对于更高级的压缩,OpenCV 支持各种算法:

  1. PNG 压缩:

您可以将图像转换为PNG格式,它有很多优点。使用以下代码行,您可以根据需要将压缩设置为 0 到 9。 0 表示不进行任何压缩,9 表示最大。请记住,PNG 是一种“无损”格式,因此即使在最大压缩下,图像也应保持完整。最大的权衡是文件大小和压缩时间。

以下是使用 OpenCV 进行 PNG 压缩的代码:

cv2.imwrite('compressed.png', img, [cv2.IMWRITE_PNG_COMPRESSION, 9])

这是我们的结果:

The Complete Guide to Image Compression with OpenCV

注意:有时您可能会注意到 PNG 文件实际上尺寸更大,如本例所示。这取决于图像的内容。

  1. WebP 压缩:

您还可以将图像转换为 .webp 格式。这是一种越来越流行的新型压缩方法。多年来我一直在博客上的图像上使用这种压缩。

在下面的代码中,我们可以将图像写入 webp 文件并将压缩级别设置为 0 到 100。这与 PNG 的比例相反,因为 0,因为我们设置的是 quality 而不是压缩。这个微小的区别很重要,因为设置为 0 是可能的最低质量,文件大小较小且损失很大。 100是最高质量,这意味着大文件具有最好的图像质量。

下面是实现这一点的 Python 代码:

cv2.imwrite('compressed.webp', img, [cv2.IMWRITE_WEBP_QUALITY, 80])

这是我们的结果:

The Complete Guide to Image Compression with OpenCV

这两种技术非常适合压缩大量数据。您可以编写脚本来自动压缩数千或数十万张图像。

结论

图像压缩非常棒。它在很多方面对于计算机视觉任务都是至关重要的,特别是在节省空间或提高处理速度时。当您想要减少硬盘空间或节省带宽时,计算机视觉之外还有许多用例。图像压缩有很大帮助。

通过理解其背后的理论并应用它,您可以在您的项目中做一些强大的事情。

请记住,有效压缩的关键是找到减小文件大小和保持应用程序可接受的视觉质量之间的最佳平衡点。

感谢您的阅读,如果您有任何意见或问题,请随时与我们联系!

版本聲明 本文轉載於:https://dev.to/jeremycmorgan/the-complete-guide-to-image-compression-with-opencv-2bld?1如有侵犯,請聯絡[email protected]刪除
最新教學 更多>
  • 解決MySQL插入Emoji時出現的\\"字符串值錯誤\\"異常
    解決MySQL插入Emoji時出現的\\"字符串值錯誤\\"異常
    Resolving Incorrect String Value Exception When Inserting EmojiWhen attempting to insert a string containing emoji characters into a MySQL database us...
    程式設計 發佈於2025-04-29
  • 在Pandas中如何將年份和季度列合併為一個週期列?
    在Pandas中如何將年份和季度列合併為一個週期列?
    pandas data frame thing commans date lay neal and pree pree'和pree pree pree”,季度 2000 q2 這個目標是通過組合“年度”和“季度”列來創建一個新列,以獲取以下結果: [python中的concate...
    程式設計 發佈於2025-04-29
  • 如何檢查對像是否具有Python中的特定屬性?
    如何檢查對像是否具有Python中的特定屬性?
    方法來確定對象屬性存在尋求一種方法來驗證對像中特定屬性的存在。考慮以下示例,其中嘗試訪問不確定屬性會引起錯誤: >>> a = someClass() >>> A.property Trackback(最近的最新電話): 文件“ ”,第1行, AttributeError: SomeClass...
    程式設計 發佈於2025-04-29
  • C++中如何將獨占指針作為函數或構造函數參數傳遞?
    C++中如何將獨占指針作為函數或構造函數參數傳遞?
    在構造函數和函數中將唯一的指數管理為參數 unique pointers( unique_ptr [2啟示。通過值: base(std :: simelor_ptr n) :next(std :: move(n)){} 此方法將唯一指針的所有權轉移到函數/對象。指針的內容被移至功能中,在操作...
    程式設計 發佈於2025-04-29
  • 如何實時捕獲和流媒體以進行聊天機器人命令執行?
    如何實時捕獲和流媒體以進行聊天機器人命令執行?
    在開發能夠執行命令的chatbots的領域中,實時從命令執行實時捕獲Stdout,一個常見的需求是能夠檢索和顯示標準輸出(stdout)在cath cath cant cant cant cant cant cant cant cant interfaces in Chate cant inter...
    程式設計 發佈於2025-04-29
  • 圖片在Chrome中為何仍有邊框? `border: none;`無效解決方案
    圖片在Chrome中為何仍有邊框? `border: none;`無效解決方案
    在chrome 中刪除一個頻繁的問題時,在與Chrome and IE9中的圖像一起工作時,遇到了一個頻繁的問題。和“邊境:無;”在CSS中。要解決此問題,請考慮以下方法: Chrome具有忽略“ border:none; none;”的已知錯誤,風格。要解決此問題,請使用以下CSS ID塊創建帶...
    程式設計 發佈於2025-04-29
  • Java 8中如何有效處理Lambda和Stream中的Checked Exception?
    Java 8中如何有效處理Lambda和Stream中的Checked Exception?
    [java 8 lambdas and streams and streams 在利用Java 8 lambdas and stream時處理檢查的異常。但是,與期望相反:的限制異常:A Potential Solution (Missed Opportunity):A more appropr...
    程式設計 發佈於2025-04-29
  • 如何將MySQL數據庫添加到Visual Studio 2012中的數據源對話框中?
    如何將MySQL數據庫添加到Visual Studio 2012中的數據源對話框中?
    在Visual Studio 2012 儘管已安裝了MySQL Connector v.6.5.4,但無法將MySQL數據庫添加到實體框架的“ DataSource對話框”中。為了解決這一問題,至關重要的是要了解MySQL連接器v.6.5.5及以後的6.6.x版本將提供MySQL的官方Visual...
    程式設計 發佈於2025-04-29
  • Python中何時用"try"而非"if"檢測變量值?
    Python中何時用"try"而非"if"檢測變量值?
    使用“ try“ vs.” if”來測試python 在python中的變量值,在某些情況下,您可能需要在處理之前檢查變量是否具有值。在使用“如果”或“ try”構建體之間決定。 “ if” constructs result = function() 如果結果: 對於結果: ...
    程式設計 發佈於2025-04-29
  • 使用JavaScript訪問MVC模型屬性方法
    使用JavaScript訪問MVC模型屬性方法
    在JavaScript中訪問MVC模型屬性 問題: 如何在JavaScript代碼中訪問綁定到視圖模型的數據?例如,如何在JavaScript中訪問FloorPlanSettingsModel的屬性? 最初嘗試: var floorplanSettings = "@Model.FloorPlan...
    程式設計 發佈於2025-04-29
  • 為什麼我的CSS背景圖像出現?
    為什麼我的CSS背景圖像出現?
    故障排除:CSS背景圖像未出現 ,您的背景圖像儘管遵循教程說明,但您的背景圖像仍未加載。圖像和样式表位於相同的目錄中,但背景仍然是空白的白色帆布。 而不是不棄用的,您已經使用了CSS樣式: bockent {背景:封閉圖像文件名:背景圖:url(nickcage.jpg); 如果您的html,cs...
    程式設計 發佈於2025-04-29
  • 大批
    大批
    [2 數組是對象,因此它們在JS中也具有方法。 切片(開始):在新數組中提取部分數組,而無需突變原始數組。 令ARR = ['a','b','c','d','e']; // USECASE:提取直到索引作...
    程式設計 發佈於2025-04-29
  • Java數組中元素位置查找技巧
    Java數組中元素位置查找技巧
    在Java數組中檢索元素的位置 利用Java的反射API將數組轉換為列表中,允許您使用indexof方法。 (primitives)(鏈接到Mishax的解決方案) 用於排序陣列的數組此方法此方法返回元素的索引,如果發現了元素的索引,或一個負值,指示應放置元素的插入點。
    程式設計 發佈於2025-04-29
  • 為什麼不````''{margin:0; }`始終刪除CSS中的最高邊距?
    為什麼不````''{margin:0; }`始終刪除CSS中的最高邊距?
    在CSS 問題:不正確的代碼: 全球範圍將所有餘量重置為零,如提供的代碼所建議的,可能會導致意外的副作用。解決特定的保證金問題是更建議的。 例如,在提供的示例中,將以下代碼添加到CSS中,將解決餘量問題: body H1 { 保證金頂:-40px; } 此方法更精確,避免了由全局保證金重置...
    程式設計 發佈於2025-04-29
  • 表單刷新後如何防止重複提交?
    表單刷新後如何防止重複提交?
    在Web開發中預防重複提交 在表格提交後刷新頁面時,遇到重複提交的問題是常見的。要解決這個問題,請考慮以下方法: 想像一下具有這樣的代碼段,看起來像這樣的代碼段:)){ //數據庫操作... 迴聲“操作完成”; 死(); } ? > ...
    程式設計 發佈於2025-04-29

免責聲明: 提供的所有資源部分來自互聯網,如果有侵犯您的版權或其他權益,請說明詳細緣由並提供版權或權益證明然後發到郵箱:[email protected] 我們會在第一時間內為您處理。

Copyright© 2022 湘ICP备2022001581号-3