美文网首页CV
模拟椒盐噪声 salt and pepper

模拟椒盐噪声 salt and pepper

作者: 谢小帅 | 来源:发表于2017-07-16 16:57 被阅读33次

椒盐噪声

  • 椒盐噪声是由图像传感器,传输信道,解码处理等产生的 黑白相间的亮暗点噪声
  • 椒盐噪声是指两种噪声,一种是盐噪声(salt noise)盐 = 白色(255),另一种是胡椒噪声(pepper noise)椒 = 黑色(0)。前者是高灰度噪声,后者属于低灰度噪声。一般两种噪声同时出现,呈现在图像上就是黑白杂点。
  • 对于彩色图像,也有可能表现为在单个像素BGR三个通道同时随机出现的255或0,也是模拟的原理。

源代码

#include <cv.h>
#include <highgui.h>

using namespace cv;
using namespace std;

// n: 噪点个数
// 可以通过image的像素点个数设置一定比例数目的噪点
void salt(Mat image, int n) {
    int i, j;
    for (int k = 0; k < n / 2; ++k) {

        // 随机选取原图的行列
        i = rand() % image.rows;
        j = rand() % image.cols;

        if (image.type() == CV_8UC1) { // 单通道灰度图
            image.at<uchar>(i, j) = 255;
        } else if (image.type() == CV_8UC3) { // color image 彩色图
            image.at<Vec3b>(i, j) = {255, 255, 255};
        }
    }
}

// 椒噪声
// 255 -> 0 即可
void pepper(Mat image, int n) {
    int i, j;
    for (int k = 0; k < n / 2; ++k) {

        // 随机选取原图的行列
        i = rand() % image.rows;
        j = rand() % image.cols;

        if (image.type() == CV_8UC1) { // 单通道灰度图
            image.at<uchar>(i, j) = 0;
        } else if (image.type() == CV_8UC3) { // color image 彩色图
            image.at<Vec3b>(i, j) = {0, 0, 0};
        }
    }
}


int main() {

    Mat src = imread("../pictures/bear.jpeg");

    // 克隆一个一样的Mat,方便后面对同样的数据源添加椒噪声
    // 不能用 Mat src1(src),这样是引用,src与src1的数据一样
    Mat src1 = src.clone();

    // 原图
    namedWindow("src");
    imshow("src", src);

    // 盐噪声
    salt(src, 1000);
    namedWindow("salt");
    imshow("salt", src);

    // 椒噪声
    pepper(src1, 1000);
    namedWindow("pepper");
    imshow("pepper", src1);

    // 椒盐噪声
    pepper(src, 1000);
    namedWindow("salt & pepper");
    imshow("salt & pepper", src);

    waitKey(0);
    return 0;
}
  • 原图的克隆而不是引用 src.clone()
  • 灰度图:image.at<uchar>(i, j) = 255;
  • 彩色图:image.at<Vec3b>(i, j) = {255, 255, 255};
  • 盐噪声:255,椒噪声:0

添加噪声的图像


椒盐噪声

可调噪点大小的 pepper 函数

  • 噪点太小了,可以通过选择随机像素点周边区域的办法放大噪点。
  • 如果选择了 (a,b) ,那么 以 (a,b) 为原点,width为边长的 右下方的正方形区域 都选中,并通过取模运算避免超出图像边界。
  • 之所以不以 (a,b)为中心,避免复杂运算。
void pepper(Mat image, int n) {
    int i, j;
    for (int k = 0; k < n / 2; ++k) {

        // 随机选取原图的行列
        i = rand() % image.rows;
        j = rand() % image.cols;

        int width = 4;

        if (image.type() == CV_8UC1) { // 单通道灰度图
//            image.at<uchar>(i, j) = 0;
            // 核心代码
            // 4*4 的噪点
            for (int row = 0; row < width; ++row) {
                for (int col = 0; col < width; ++col) {
                    image.at<uchar>((i + row) % image.rows, (j + col) % image.cols) = 0;
                }
            }
        } else if (image.type() == CV_8UC3) { // color image 彩色图
//            image.at<Vec3b>(i, j) = {0, 0, 0};
            for (int row = 0; row < width; ++row) {
                for (int col = 0; col < width; ++col) {
                    image.at<Vec3b>((i + row) % image.rows, (j + col) % image.cols) = {0, 0, 0};
                }
            }
        }
    }
}

放大了的椒噪声


相关文章

网友评论

    本文标题:模拟椒盐噪声 salt and pepper

    本文链接:https://www.haomeiwen.com/subject/flatkxtx.html