六、OpenCV中的图像读写

文章目录

一、基本用法

OpenCV 里的图像读写操作是最常用的功能之一,主要依赖 imread 和 imwrite 这两个函数(以及少量配套函数)。

1.读取图像 imread

cpp 复制代码
cv::Mat image = cv::imread(const std::string& filename, int flags = IMREAD_COLOR);

参数:

  • filename:图像文件路径,可以是 PNG、JPEG、BMP、TIFF 等常见格式。
  • flags:读取模式,常用选项:
    • IMREAD_COLOR(默认):以彩色方式读取,返回 3 通道 BGR。
    • IMREAD_GRAYSCALE:以灰度图方式读取,返回单通道。
    • IMREAD_UNCHANGED:原样读取(包括 Alpha 通道)。
    • IMREAD_ANYDEPTH:读入原始位深(例如 16 位图像不会被强制转成 8 位)。
    • IMREAD_ANYCOLOR:不管图像什么格式,都尽量转成彩色。

示例:

cpp 复制代码
cv::Mat color = cv::imread("image.png");                      // 彩色
cv::Mat gray = cv::imread("image.png", cv::IMREAD_GRAYSCALE); // 灰度
cv::Mat withAlpha = cv::imread("image.png", cv::IMREAD_UNCHANGED); // 包含 Alpha

2.保存图像 imwrite

cpp 复制代码
bool success = cv::imwrite(const std::string& filename, const cv::Mat& image, const std::vector<int>& params = {});

参数:

  • filename:保存路径,文件扩展名决定编码格式(如 .jpg、.png、.tiff)。
  • image:要保存的 Mat。
  • params:编码参数,不同格式有不同参数。
    • JPEG:IMWRITE_JPEG_QUALITY(默认 95,范围 0--100)。
    • PNG:IMWRITE_PNG_COMPRESSION(0--9,数值越大压缩越强,文件小但耗时更长)。
    • TIFF:支持多页存储(可以传 vector<Mat>)。
    • WebP:IMWRITE_WEBP_QUALITY(0--100)。

示例:

cpp 复制代码
cv::imwrite("gray.jpg", gray);  // 保存为 JPEG(有损)

std::vector<int> params;
params.push_back(cv::IMWRITE_PNG_COMPRESSION);
params.push_back(9);  // 最强压缩
cv::imwrite("alpha.png", withAlpha, params);

3.判断读写是否成功

  • imread 如果失败(文件不存在、路径错误、格式不支持),会返回一个 空 Mat(mat.empty() == true)。
  • imwrite 返回 true/false,保存失败时返回 false 或抛异常。

4.显示图像(配合 imshow)

通常读写图像后会想看效果,可以用:

cpp 复制代码
cv::imshow("window", image);
cv::waitKey(0); // 等待键盘输入,否则窗口一闪而过

5.多页读写(TIFF/多帧图像)

读取多页:

cpp 复制代码
std::vector<cv::Mat> pages;
cv::imreadmulti("test.tiff", pages, cv::IMREAD_ANYCOLOR);

保存多页:

cpp 复制代码
std::vector<cv::Mat> imgs = {mat1, mat2, mat3};
cv::imwrite("test.tiff", imgs);

二、示例代码

完整的 OpenCV 图像读写示例代码,涵盖了常见场景:读取、灰度化、保存 PNG、保存多页 TIFF。

cpp 复制代码
#include <opencv2/opencv.hpp>
#include <iostream>
using namespace cv;
using namespace std;

int main()
{
    // 1. 读取彩色图像
    Mat color = imread("input.jpg", IMREAD_COLOR);
    if (color.empty()) {
        cerr << "无法读取图像,请检查路径!" << endl;
        return -1;
    }
    imshow("原图", color);

    // 2. 转换为灰度图
    Mat gray;
    cvtColor(color, gray, COLOR_BGR2GRAY);
    imshow("灰度图", gray);

    // 3. 保存为 PNG(带压缩参数)
    vector<int> png_params;
    png_params.push_back(IMWRITE_PNG_COMPRESSION);
    png_params.push_back(9); // 0 = 无压缩,9 = 最大压缩
    if (!imwrite("output.png", color, png_params)) {
        cerr << "保存 PNG 失败!" << endl;
    } else {
        cout << "保存 output.png 成功!" << endl;
    }

    // 4. 保存为 JPEG(设置质量)
    vector<int> jpg_params;
    jpg_params.push_back(IMWRITE_JPEG_QUALITY);
    jpg_params.push_back(95); // 默认质量 95
    if (!imwrite("output.jpg", gray, jpg_params)) {
        cerr << "保存 JPEG 失败!" << endl;
    } else {
        cout << "保存 output.jpg 成功!" << endl;
    }

    // 5. 保存多页 TIFF(原图 + 灰度图)
    vector<Mat> imgs;
    imgs.push_back(color);
    imgs.push_back(gray);
    if (!imwrite("output.tiff", imgs)) {
        cerr << "保存 TIFF 失败!" << endl;
    } else {
        cout << "保存多页 output.tiff 成功!" << endl;
    }

    // 6. 读取多页 TIFF
    vector<Mat> pages;
    if (imreadmulti("output.tiff", pages, IMREAD_ANYCOLOR)) {
        cout << "读取 TIFF 成功,页数: " << pages.size() << endl;
        for (size_t i = 0; i < pages.size(); i++) {
            imshow("Page " + to_string(i), pages[i]);
        }
    } else {
        cerr << "读取 TIFF 失败!" << endl;
    }

    waitKey(0);
    return 0;
}

三、只针对 Alpha 通道的读写示例

图像的Alpha 通道的作用?

在图像处理中,Alpha 通道就是透明度通道(Opacity/Transparency Channel),它在 RGB 颜色通道之外,额外增加了一层,用来描述每个像素的"透明程度"。

通道概念

  • 普通彩色图像:3 通道 → R(红)、G(绿)、B(蓝)
  • 带 Alpha 的彩色图像:4 通道 → R、G、B + A(透明度)

Alpha 通道的取值

  • 8 位图像中,范围是 0 ~ 255
  • 0 表示完全透明
  • 255 表示完全不透明
  • 中间值表示半透明(常用于渐变、阴影、反射等效果)

Alpha 通道的作用
透明背景:

  • 允许图像在叠加时背景透过去。
  • 例如:PNG 图标放在网页上时,背景透明,不会出现难看的白底或黑底。

图像合成(混合)

  • 在图像 A 上叠加图像 B,利用 Alpha 通道进行"加权混合"。
  • OpenCV 中可用 addWeighted 或 cv::Mat 的逐像素运算。

混合公式(常用 Alpha Blending):

cpp 复制代码
C=α⋅F+(1−α)⋅B
  • C:最终像素
  • F:前景像素
  • B:背景像素
  • α:前景的 Alpha 值(0~1 之间)

图像遮罩(Mask)

  • Alpha 通道可以当作"掩码",决定哪些区域可见,哪些区域不可见。
  • 常用于抠图、虚化、贴图。

示例代码:

cpp 复制代码
#include <opencv2/opencv.hpp>
#include <iostream>
using namespace cv;
using namespace std;

int main()
{
    // 1. 读取带 Alpha 通道的 PNG
    Mat img = imread("input.png", IMREAD_UNCHANGED);
    if (img.empty()) {
        cerr << "无法读取 input.png,请检查路径!" << endl;
        return -1;
    }

    // 确认是否有 4 通道
    if (img.channels() != 4) {
        cerr << "图像没有 Alpha 通道!" << endl;
        return -1;
    }

    cout << "读取到 " << img.cols << "x" << img.rows 
         << " 的图像,通道数: " << img.channels() << endl;

    // 2. 分离 BGR 和 Alpha
    vector<Mat> channels;
    split(img, channels);  // channels[0]=B, [1]=G, [2]=R, [3]=A
    Mat b = channels[0], g = channels[1], r = channels[2], alpha = channels[3];

    // 显示 Alpha 通道
    imshow("Alpha 通道", alpha);

    // 3. 对 Alpha 通道做处理(示例:取反)
    Mat alpha_inv;
    bitwise_not(alpha, alpha_inv);  // 透明变不透明,反之亦然
    imshow("Alpha 取反", alpha_inv);

    // 4. 合并 BGR 和新 Alpha
    channels[3] = alpha_inv;
    Mat output;
    merge(channels, output);

    // 5. 保存 PNG(带 Alpha)
    vector<int> params;
    params.push_back(IMWRITE_PNG_COMPRESSION);
    params.push_back(9);
    if (!imwrite("output.png", output, params)) {
        cerr << "保存 output.png 失败!" << endl;
    } else {
        cout << "保存带 Alpha 通道的 output.png 成功!" << endl;
    }

    waitKey(0);
    return 0;
}

四、两张图像的融合(叠加合成)

基本原理(Alpha 混合公式)

对于前景图 F 和背景图 B,融合后的图像 C 的像素计算公式是:

cpp 复制代码
C=α⋅F+(1−α)⋅B

α:融合比例(0.0 ~ 1.0)

  • 1.0 → 完全显示前景
  • 0.0 → 完全显示背景
  • 0.5 → 前景和背景各占一半

OpenCV 提供了现成的函数 cv::addWeighted:

cpp 复制代码
void addWeighted(InputArray src1, double alpha,
                 InputArray src2, double beta,
                 double gamma, OutputArray dst);
  • src1、src2:输入图像(大小和类型需一致)
  • alpha、beta:权重(一般 alpha + beta = 1)
  • gamma:加到结果上的常数偏移量(通常为 0)
  • dst:输出图像

示例代码:

cpp 复制代码
#include <opencv2/opencv.hpp>
#include <iostream>
using namespace cv;
using namespace std;

int main()
{
    // 1. 读取两张图片
    Mat img1 = imread("opencv_demo_1.png");
    Mat img2 = imread("opencv_demo_2.png");

    if (img1.empty() || img2.empty()) {
        cerr << "无法读取图像!" << endl;
        return -1;
    }

    // 2. 调整大小(确保两张图尺寸相同)
    resize(img2, img2, img1.size());

    // 3. 融合
    Mat blended;
    double alpha = 0.8;  // 前景权重
    double beta = 1.0 - alpha; // 背景权重
    addWeighted(img1, alpha, img2, beta, 0.0, blended);

    // 4. 显示和保存
    imshow("图像1", img1);
    imshow("图像2", img2);
    imshow("融合结果", blended);

    imwrite("blended.jpg", blended);

    waitKey(0);
    return 0;
}
相关推荐
苏世-顾长歌6 小时前
Android studio导入OpenCV报“Unresolved reference: android“
android·opencv·android studio
荼蘼7 小时前
OpenCV 人脸检测、微笑检测 原理及案例解析
人工智能·opencv·计算机视觉
算法打盹中10 小时前
计算机视觉:基于YOLOv11 实例分割与OpenCV 在 Java 中的实现图像实例分割
opencv·yolo·计算机视觉·图像分割·实例分割·yolo11
CLTHREE11 小时前
处理视频抽帧并转换成json
python·opencv·计算机视觉
Monkey的自我迭代13 小时前
光流估计(可用于目标跟踪)
人工智能·opencv·计算机视觉
WeiJingYu.19 小时前
O3.6opencv
人工智能·opencv·计算机视觉
_nirvana_w_21 小时前
PyQt6+OpenCV 实战:打造功能完备的数字图像处理 GUI 系统
人工智能·python·qt·opencv·计算机视觉
华法林的小助手1 天前
windows c++环境 使用VScdoe配置opencv
c++·windows·opencv
何以解忧唯有撸码1 天前
OpenCvSharp基于颜色反差规避FBA面单贴标2
opencv