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

C++ OpenCV實(shí)現(xiàn)灰度圖蒙版GrayMask的示例代碼

 更新時(shí)間:2022年05月07日 11:39:22   作者:翟天保Steven  
這篇文章主要為大家詳細(xì)介紹了如何利用C++和OpenCV實(shí)現(xiàn)灰度圖蒙版GrayMask,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)或工作有一定參考價(jià)值,需要的可以參考一下

需求說(shuō)明

在對(duì)圖像進(jìn)行處理時(shí),經(jīng)常會(huì)有這類需求:想對(duì)感興趣區(qū)域進(jìn)行掩膜處理,只操作掩膜內(nèi)數(shù)據(jù),此時(shí)需要搭配掩膜繪制功能,并在繪制過(guò)程中希望能區(qū)分掩膜區(qū)和非掩膜區(qū);除了掩膜本身的線條以外,還希望掩膜內(nèi)圖像是原色,掩膜外圖像的顏色進(jìn)行一定調(diào)整;通常可以采用圖像透明化或者色彩單通道加深的方式實(shí)現(xiàn)。

比如對(duì)三通道的圖像,可以將掩膜外數(shù)據(jù)的紅通道數(shù)值提高,此時(shí)該部分圖像就會(huì)偏紅色,如下圖1所示。

圖1 顏色加深的蒙版效果

但是針對(duì)灰度圖,因?yàn)閳D像數(shù)據(jù)本身就是單通道的,就不能借用顏色通道或者透明通道了;有一個(gè)基礎(chǔ)的辦法是將非掩膜區(qū)數(shù)據(jù)同時(shí)加減某個(gè)數(shù)值,這種方法簡(jiǎn)單但有一個(gè)弊端,比如我加了50,那原圖200-255之間的數(shù)值都將變?yōu)?55,這樣就損壞了原圖的某些特征信息。為此,我們采用非線性的算法將其進(jìn)行較為合理的數(shù)值調(diào)整。

下面介紹具體實(shí)現(xiàn)流程。

具體流程

1)構(gòu)造非線性參數(shù)。其中n為函數(shù)輸入的參數(shù),范圍在-100到100;a、b、k為算法參數(shù),公式見(jiàn)下。

n = min(max(n, -100), 100);
int k = 4;
float b = n / 100.f / k;
float a = 1.0f + k * b;

2)參照上式,對(duì)掩膜外數(shù)據(jù)進(jìn)行如下處理。

float temp = pow(float(in[j]) / 255.f, 1.0f / a) * (1.0 / (1 + b));

3)temp的數(shù)值需要進(jìn)行校準(zhǔn),因?yàn)榛叶戎等绻莡char型,范圍在0-255之間,校準(zhǔn)后乘上255,即可得到算法計(jì)算的結(jié)果。

if (temp > 1.0f)
    temp = 1.0f;
if (temp < 0.0f)
    temp = 0.0f;
uchar utemp = uchar(255 * temp);
r[j] = utemp;

4)掩膜內(nèi)數(shù)值保持不變。

r[j] = in[j];

5)輸出圖像,完成。

功能函數(shù)

// 灰度圖蒙版
cv::Mat GrayMask(cv::Mat input, cv::Mat mask, int n)
{
    // 通道判斷
    if (input.channels() != 1)
    {
        return input;
    }
    
    // 非線性參數(shù)
    // 當(dāng)n=100時(shí),非掩膜區(qū)呈灰色;當(dāng)n=0時(shí),非掩膜區(qū)無(wú)變化;當(dāng)n=-100時(shí),非掩膜區(qū)黑色
    // n越小,非掩膜區(qū)越暗,n越大,非掩膜區(qū)越灰
    // 目的是區(qū)分非掩膜區(qū)和掩膜區(qū)
    n = min(max(n, -100), 100);
    int k = 4;
    float b = n / 100.f / k;
    float a = 1.0f + k * b;
 
    // 蒙版處理
    cv::Mat result = cv::Mat::zeros(input.size(), input.type());
    for (int i = 0; i < input.rows; ++i)
    {
        uchar *m = mask.ptr<uchar>(i);
        uchar *in = input.ptr<uchar>(i);
        uchar *r = result.ptr<uchar>(i);
        for (int j = 0; j < input.cols; ++j)
        {
            if (m[j] == 0)
            {
                float temp = pow(float(in[j]) / 255.f, 1.0f / a) * (1.0 / (1 + b));
                if (temp > 1.0f)
                    temp = 1.0f;
                if (temp < 0.0f)
                    temp = 0.0f;
                uchar utemp = uchar(255 * temp);
                r[j] = utemp;
            }
            else {
                r[j] = in[j];
            }
        }
    }
    return result;
}

C++測(cè)試代碼

#include <iostream>
#include <unordered_map>
#include <unordered_set>
#include <stdio.h>
#include <io.h>
#include <chrono> 
#include <time.h>
#include <stdio.h>
#include <direct.h>
#include <ctime>
#include <opencv2/core/core.hpp>
#include <opencv2/highgui.hpp>
#include <opencv2/imgproc.hpp>
 
using namespace std;
using namespace cv;
 
cv::Mat GrayMask(cv::Mat input, cv::Mat mask, int n);
 
int main()
{
    // 讀取灰度圖
    cv::Mat src = imread("1.png",0);
    // 繪制掩膜
    cv::Mat mask = cv::Mat::zeros(src.size(), CV_8UC1);
    cv::circle(mask, cv::Point(src.cols / 2, src.rows / 2), 100, cv::Scalar(255), -1);
    // 灰度圖蒙版
    cv::Mat result = GrayMask(src, mask, 100);
    // 結(jié)果顯示
    imshow("src", src);
    imshow("result", result);
    waitKey(0);
    system("pause");
    return 0;
}
 
// 灰度圖蒙版
cv::Mat GrayMask(cv::Mat input, cv::Mat mask, int n)
{
    // 通道判斷
    if (input.channels() != 1)
    {
        return input;
    }
    
    // 非線性參數(shù)
    // 當(dāng)n=100時(shí),非掩膜區(qū)呈灰色;當(dāng)n=0時(shí),非掩膜區(qū)無(wú)變化;當(dāng)n=-100時(shí),非掩膜區(qū)黑色
    // n越小,非掩膜區(qū)越暗,n越大,非掩膜區(qū)越灰
    // 目的是區(qū)分非掩膜區(qū)和掩膜區(qū)
    n = min(max(n, -100), 100);
    int k = 4;
    float b = n / 100.f / k;
    float a = 1.0f + k * b;
 
    // 蒙版處理
    cv::Mat result = cv::Mat::zeros(input.size(), input.type());
    for (int i = 0; i < input.rows; ++i)
    {
        uchar *m = mask.ptr<uchar>(i);
        uchar *in = input.ptr<uchar>(i);
        uchar *r = result.ptr<uchar>(i);
        for (int j = 0; j < input.cols; ++j)
        {
            if (m[j] == 0)
            {
                float temp = pow(float(in[j]) / 255.f, 1.0f / a) * (1.0 / (1 + b));
                if (temp > 1.0f)
                    temp = 1.0f;
                if (temp < 0.0f)
                    temp = 0.0f;
                uchar utemp = uchar(255 * temp);
                r[j] = utemp;
            }
            else {
                r[j] = in[j];
            }
        }
    }
    return result;
}

測(cè)試效果

圖2 原圖

圖3 灰度圖蒙版效果

如果函數(shù)有什么可以改進(jìn)完善的地方,非常歡迎大家指出,一同進(jìn)步何樂(lè)而不為呢~

到此這篇關(guān)于C++ OpenCV實(shí)現(xiàn)灰度圖蒙版GrayMask的示例代碼的文章就介紹到這了,更多相關(guān)C++ OpenCV灰度圖蒙版內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • C++11實(shí)現(xiàn)簡(jiǎn)易定時(shí)器的示例代碼

    C++11實(shí)現(xiàn)簡(jiǎn)易定時(shí)器的示例代碼

    這篇文章主要介紹了C++11實(shí)現(xiàn)簡(jiǎn)易定時(shí)器的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2020-04-04
  • C++實(shí)現(xiàn)掃雷、排雷小游戲

    C++實(shí)現(xiàn)掃雷、排雷小游戲

    這篇文章主要為大家詳細(xì)介紹了C++實(shí)現(xiàn)掃雷、排雷小游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-05-05
  • C++ vector的基本使用示例詳解

    C++ vector的基本使用示例詳解

    這篇文章主要介紹了C++ vector的基本使用,本文通過(guò)示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2023-03-03
  • VC++在TXT文件指定位置追加內(nèi)容的方法

    VC++在TXT文件指定位置追加內(nèi)容的方法

    這篇文章主要介紹了VC++在TXT文件指定位置追加內(nèi)容的方法,功能較為實(shí)用,需要的朋友可以參考下
    2014-08-08
  • C++ explicit關(guān)鍵字講解

    C++ explicit關(guān)鍵字講解

    這篇文章主要介紹了C++ explicit關(guān)鍵字講解,++提供了explicit關(guān)鍵字,相對(duì)于implicit而言,他默認(rèn)關(guān)閉了隱式類型轉(zhuǎn)換方法。至于兩者有什么區(qū)別,看下面文章內(nèi)容的介紹吧
    2021-12-12
  • 手把手教你如何優(yōu)化C語(yǔ)言程序

    手把手教你如何優(yōu)化C語(yǔ)言程序

    程序進(jìn)行優(yōu)化,通常是指優(yōu)化程序代碼或程序執(zhí)行速度。優(yōu)化代碼和優(yōu)化速度實(shí)際上是一個(gè)予盾的統(tǒng)一,一般是優(yōu)化了代碼的尺寸,就會(huì)帶來(lái)執(zhí)行時(shí)間的增加,如果優(yōu)化了程序的執(zhí)行速度,通常會(huì)帶來(lái)代碼增加的副作用,很難魚與熊掌兼得,只能在設(shè)計(jì)時(shí)掌握一個(gè)平衡點(diǎn)
    2013-07-07
  • 實(shí)例代碼分析c++動(dòng)態(tài)分配

    實(shí)例代碼分析c++動(dòng)態(tài)分配

    這篇文章主要介紹了c++動(dòng)態(tài)分配的的相關(guān)資料,文中代碼簡(jiǎn)單易懂,方便大家更好的學(xué)習(xí)參考,感興趣的朋友可以了解下
    2020-06-06
  • C語(yǔ)言實(shí)例梳理講解常用關(guān)鍵字的用法

    C語(yǔ)言實(shí)例梳理講解常用關(guān)鍵字的用法

    關(guān)鍵字是C語(yǔ)言非常重要的一部分,熟練的掌握和使用關(guān)鍵字有助于我們更加熟悉了解C語(yǔ)言,同時(shí)C語(yǔ)言的關(guān)鍵字也是面試筆試中??嫉膬?nèi)容。C語(yǔ)言的關(guān)鍵字共有32個(gè),但并不是每個(gè)關(guān)鍵字都有坑,本篇文章將通過(guò)理論聯(lián)系實(shí)際的方式為大家講解C語(yǔ)言中易混易錯(cuò)以及??嫉囊恍╆P(guān)鍵字
    2022-05-05
  • c++中ref的作用示例解析

    c++中ref的作用示例解析

    這篇文章主要為大家介紹了c++中ref的作用示例解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-04-04
  • C語(yǔ)言算法學(xué)習(xí)之雙向鏈表詳解

    C語(yǔ)言算法學(xué)習(xí)之雙向鏈表詳解

    雙向鏈表也叫雙鏈表,是鏈表的一種,它的每個(gè)數(shù)據(jù)結(jié)點(diǎn)中都有兩個(gè)指針,分別指向直接后繼和直接前驅(qū)。本文主要介紹了C語(yǔ)言算法中雙向鏈表的實(shí)現(xiàn),需要的可以參考一下
    2022-05-05

最新評(píng)論