数字图像处理(2):图像反相(分别用C语言、OpenCV、Matlab实现)

1. 图像格式

1.1 灰度图像

灰度数字图像是每个像素只有一个采样颜色的图像。这类图像通常显示为从最暗黑色到最亮的白色的灰度。灰度图像与黑白图像不同,在计算机图像领域中黑白图像只有黑白两种颜色,灰度图像在黑色与白色之间还有许多级的颜色深度。

1.2 RGB888图像

RGB色彩模式是工业界的一种颜色标准,是通过对红®、绿(G)、蓝(B)三个颜色通道的变化以及它们相互之间的叠加来得到各式各样的颜色的,RGB即是代表红、绿、蓝三个通道的颜色,这个标准几乎包括了人类视力所能感知的所有颜色,是目前运用最广的颜色系统之一。

256级的RGB色彩总共能组合出约1678万种色彩,即256×256×256=16777216。通常也被简称为1600万色或千万色。也称为24位色(2的24次方)。


2. 图像反相

“反相”即为图像的颜色色相反转。以前照相机的底片就是打印后的照片的反相。比如黑变白,蓝变黄、红变绿。

对灰度级范围为[0 , L-1]的一幅图像进行反转的操作为:s = L - 1 - r

其中r和s分别代表处理前后的像素值。使用这种方式反转一幅图像的灰度级,可得到等效的照片底片。

这种类型的处理特别适用于增强嵌入在一幅图像的暗区域中的白色和灰色细节,特别是当黑色面积在尺寸上占主导地位时。


3. 代码&运行

3.1 平台:VS2015 + C语言

界面截图:

在这里插入图片描述

流程:

开始 -> 读取RAW格式图片 -> 保存至二维数组 -> 数组每个元素进行“255-”操作 -> 将二维数组生成RAW格式文件 -> 用PS查看图片操作结果

结果:

在工程目录下生成了output.raw文件,用PS打开显示结果如下

在这里插入图片描述

对彩色进行操作:

在这里插入图片描述

彩色图像为三通道,故而代码中修改宽度乘以3

在这里插入图片描述

遇到问题:

修改代码后直接运行报错如下——

在这里插入图片描述

经查找原因得知图像较大,静态数组不够。

解决方法:

修改为动态数组进行图片的操作,解决问题

在这里插入图片描述


代码:

#include <stdio.h>  
#include <stdlib.h>  
#define height  600  
#define width   800 
typedef unsigned char  BYTE;    // 定义BYTE类型,占1个字节  
int main(void)
{
    FILE *fp = NULL;
    BYTE Pic[height][width];
    BYTE *ptr;
    int i, j;
    fp = fopen("weiminglake.raw", "rb");       
    ptr = (BYTE*)malloc(width * height * sizeof(BYTE));//创建内存
    for (i = 0; i < height; i++)
    {
        for (j = 0; j < width; j++)
        {
            fread(ptr, 1, 1, fp);
            Pic[i][j] = *ptr;  // 把图像输入到2维数组中,变成矩阵型式  
            ptr++;
        }
    }
    fclose(fp);
    for (i = 0; i < height; i++)
    {
        for (j = 0; j < width; j++)
        {
            Pic[i][j] = 255 - Pic[i][j];  // 图像反相  
        }
    }
    fp = fopen("output.raw", "wb");
    for (i = 0; i < height; i++)
    {
        for (j = 0; j < width; j++)
        {
            fwrite(&Pic[i][j], 1, 1, fp);
        }
    }
    fclose(fp);
    return 0;
}


3.2 平台:VS2015+OpenCv3.4.1

界面截图:

在这里插入图片描述

运行结果:

灰度图像反相

在这里插入图片描述


彩色图像反相

在这里插入图片描述


代码:

#include <iostream>  
#include <opencv2/core/core.hpp>  
#include <opencv2/highgui/highgui.hpp>  
using namespace cv;
int main() 
{   
    Mat src;
    int height, width;
    int i,j;
    src = imread("weiminglake_huidu.jpg");     //载入图片
    height = src.rows;                       //获取图像信息
    width = src.cols* src.channels();          // 列项要乘通道数
    namedWindow("src", CV_WINDOW_AUTOSIZE);     //创建窗口
    namedWindow("dst", CV_WINDOW_AUTOSIZE);
    imshow("src", src);                       //显示原图
    //图像反转
    for (i = 0; i< height; i++)
    {
        for (j = 0; j< width; j++)
        {
            src.at<uchar>(i, j) = 255 - src.at<uchar>(i, j);   // 每一个像素反转
        }
    }
    imshow("dst", src);                       //显示反相图片
    waitKey(0);
    return 0;
}

当然直接用现成的函数也可以:)

3.3 平台:Matlab

界面截图:

在这里插入图片描述


运行结果:

在这里插入图片描述


分别为:
彩色图像,灰度图像
彩色反相图像,灰度反相图像

代码:

a=imread('weiminglake.jpg');
b=rgb2gray(a);
c = 255 - a;
%c=imcomplement(a); 
%c = intmax(class(a)) - a;
d = 255 - b;
%d=imcomplement(b);
subplot(2,2,1);
imshow(a);
subplot(2,2,2);
imshow(b);
subplot(2,2,3);
imshow(c)
subplot(2,2,4);
imshow(d)

本工程源码已更新至github,欢迎star,欢迎PR:)

相关文章

学习编程是顺着互联网的发展潮流,是一件好事。新手如何学习...
IT行业是什么工作做什么?IT行业的工作有:产品策划类、页面...
女生学Java好就业吗?女生适合学Java编程吗?目前有不少女生...
Can’t connect to local MySQL server through socket \'/v...
oracle基本命令 一、登录操作 1.管理员登录 # 管理员登录 ...
一、背景 因为项目中需要通北京网络,所以需要连vpn,但是服...