读取图像,然后将$\text{RGB}$通道替换成$\text{BGR}$通道。
下面的代码用于提取图像的红色通道。
注意,cv2.imread()
的系数是按$\text{BGR}$顺序排列的!
其中的变量red
表示的是仅有原图像红通道的imori.jpg
。
import cv2
img = cv2.imread("imori.jpg")
red = img[:, :, 2].copy()
输入(imori.jpg) | 输出(answers_image/answer_1.jpg) |
---|---|
答案:
- Python >> answers_py/answer_1.py
- C++ >> answers_cpp/answer_1.cpp
将图像灰度化吧!
灰度是一种图像亮度的表示方法,通过下式计算: $$ Y = 0.2126\ R + 0.7152\ G + 0.0722\ B $$
输入(imori.jpg) | 输出(answers_image/answer_2.jpg) |
---|---|
答案:
Python >> answers_py/answer_2.py
C++ >> answers_cpp/answer_2.cpp
把图像进行二值化吧。
二值化是将图像使用黑和白两种颜色表示的方法。
我们将灰度的阈值设置为$128$来进行二值化,即: $$ y= \begin{cases} 0& (\text{if}\quad y < 128) \ 255& (\text{else}) \end{cases} $$
输入(imori.jpg) | 输出(answers_image/answer_3.jpg) |
---|---|
答案
- Python >> answers_py/answer_3.py
- C++ >> answers_cpp/answer_3.cpp
使用大津算法来二值化图像吧。
大津算法,也被称作最大类间方差法,是一种可以自动确定二值化中阈值的算法。
从类内方差和类间方差的比值计算得来:
- 小于阈值$t$的类记作$0$,大于阈值$t$的类记作$1$;
-
$w_0$ 和$w_1$是被阈值$t$分开的两个类中的像素数占总像素数的比率(满足$w_0+w_1=1$); -
${S_0}^2$ ,${S_1}^2$ 是这两个类中像素值的方差; -
$M_0$ ,$M_1$是这两个类的像素值的平均值;
即:
- 类内方差:${S_w}^2=w_0\ {S_0}^2+w_1\ {S_1}^2$
- 类间方差:${S_b}^2 = w_0 \ (M_0 - M_t)^2 + w_1\ (M_1 - M_t)^2 = w_0\ w_1\ (M_0 - M_1) ^2$
- 图像所有像素的方差:${S_t}^2 = {S_w}^2 + {S_b}^2 = \text{常数}$
根据以上的式子,我们用以下的式子计算分离度$X$:1
也就是说: $$ \arg\max\limits_{t}\ X=\arg\max\limits_{t}\ {S_b}^2 $$ 换言之,如果使${S_b}^2={w_0}\ {w_1}\ (M_0 - M_1)^2$最大,就可以得到最好的二值化阈值$t$。
输入(imori.jpg) | 输出 ( |
---|---|
答案 Python >> answers/answer_4.py C++ >> answers_cpp/answer_4.py
将使用$\text{HSV}$表示色彩的图像的色相反转吧!
-
色相:将颜色使用$0^{\circ}$到$360^{\circ}$表示,就是平常所说的颜色名称,如红色、蓝色。色相与数值按下表对应:
红 黄 绿 青色 蓝色 品红 红 $0^{\circ}$ $60^{\circ}$ $120^{\circ}$ $180^{\circ}$ $240^{\circ}$ $300^{\circ}$ $360^{\circ}$ -
饱和度:是指色彩的纯度,饱和度越低则颜色越黯淡($0\leq S < 1$);
-
明度:即颜色的明暗程度。数值越高越接近白色,数值越低越接近黑色($0\leq V < 1$);
从$\text{RGB}$色彩表示转换到$\text{HSV}$色彩表示通过以下方式计算:
输入(imori.jpg) | 输出(answers_image/answer_5.jpg) |
---|---|
答案
- Python >> answers_py/answer_5.py
- C++ >> answers_cpp/answer_5.cpp
问题六:减色处理2
我们将图像的值由$256^3$压缩至$4^3$,即将$\text{RGB}$的值只取${32, 96, 160, 224}$。这被称作色彩量化。色彩的值按照下面的方式定义: $$ \text{val}= \begin{cases} 32& (0 \leq \text{var} < 64)\ 96& (64\leq \text{var}<128)\ 160&(128\leq \text{var}<192)\ 224&(192\leq \text{var}<256) \end{cases} $$
输入(imori.jpg) | 输出(answers_image/answer_6.jpg) |
---|---|
答案
- Python >> answers_py/answer_6.py
- C++ >> answers_cpp/answer_6.cpp
将图片按照固定大小网格分割,网格内的像素值取网格内所有像素的平均值。
我们将这种把图片使用均等大小网格分割,并求网格内代表值的操作称为池化(Pooling)。
池化操作是**卷积神经网络(Convolutional Neural Network)**中重要的图像处理方式。平均池化按照下式定义:
$$
v=\frac{1}{|R|}\ \sum\limits_{i=1}^R\ v_i
$$
请把大小为$128\times128$的imori.jpg
使用$8\times8$的网格做平均池化。
输入(imori.jpg) | 输出(answers_image/answer_7.jpg) |
---|---|
答案
- Python >> answers_py/answer_7.py
- C++ >> answers_cpp/answer_7.cpp
网格内的值不取平均值,而是取网格内的最大值进行池化操作。
输入(imori.jpg) | 输出(answers_image/answer_8.jpg) |
---|---|
答案
- Python >> answers_py/answer_8.py
- C++ >> answers_cpp/answer_8.cpp
使用高斯滤波器($3\times3$大小,标准差$\sigma=1.3$)来对imori_noise.jpg
进行降噪处理吧!
高斯滤波器是一种可以使图像平滑的滤波器,用于去除噪声。可用于去除噪声的滤波器还有中值滤波器(参见问题十),平滑滤波器(参见问题十一)、LoG滤波器(参见问题十九)。
高斯滤波器将中心像素周围的像素按照高斯分布加权平均进行平滑化。这样的(二维)权值通常被称为卷积核(kernel)或者滤波器(filter)。
但是,由于图像的长宽可能不是滤波器大小的整数倍,因此我们需要在图像的边缘补$0$。这种方法称作Zero Padding。并且权值$g$(卷积核)要进行归一化操作($\sum\ g = 1$)。
按下面的高斯分布公式计算权值: $$ g(x,y,\sigma)=\frac{1}{2\ \pi\ \sigma^2}\ e^{-\frac{x^2+y^2}{2\ \sigma^2}} $$
标准差$\sigma=1.3$的$8-$近邻高斯滤波器如下: $$ K=\frac{1}{16}\ \left[ \begin{matrix} 1 & 2 & 1 \ 2 & 4 & 2 \ 1 & 2 & 1 \end{matrix} \right] $$
输入(imori_noise.jpg) | 输出(answers_image/answer_9.jpg) |
---|---|
答案
- Python >> answers_py/answer_9.py
- C++ >> answers_cpp/answer_9.cpp
使用中值滤波器($3\times3$大小)来对imori_noise.jpg
进行降噪处理吧!
中值滤波器是一种可以使图像平滑的滤波器。这种滤波器用滤波器范围内(在这里是$3\times3$)像素点的中值进行滤波,请在这里也采用Zero Padding。
输入(imori_noise.jpg) | 输出(answers_image/answer_10.jpg) |
---|---|
答案
- Python >> answers_py/answer_10.py
- C++ >> answers_cpp/answer_10.cpp