opencv python num_labels, labels, stats, centroids = cv2.connectedComponentsWithStats(thresh) 如何将二值图像中的非连通域的区域变为黑色
时间: 2024-02-17 13:01:08 浏览: 84
可以使用以下代码将二值图像中的非连通域的区域变为黑色:
```python
import cv2
# 读取二值图像
img = cv2.imread('binary_image.png', 0)
# 进行连通域分析
num_labels, labels, stats, centroids = cv2.connectedComponentsWithStats(img)
# 获取连通域的面积
area = stats[:, cv2.CC_STAT_AREA]
# 找到最大的连通域
max_area_label = area.argmax()
# 将非最大连通域的区域变为黑色
for i in range(num_labels):
if i != max_area_label:
img[labels == i] = 0
# 显示结果
cv2.imshow('result', img)
cv2.waitKey(0)
cv2.destroyAllWindows()
```
在上述代码中,我们首先读取二值图像并进行连通域分析,然后获取每个连通域的面积。接着,我们找到最大的连通域,并将非最大连通域的区域变为黑色。具体来说,我们遍历每个连通域的标签,如果该标签不是最大连通域的标签,就将对应的像素点设置为黑色。
相关问题
Python中labels, stats, centroids = cv2.connectedComponentsWithStats(img)的详细用法
在Python的OpenCV(cv2)库中,`connectedComponentsWithStats`函数是一个用于图像处理的功能,它主要用于检测并统计连续区域(contiguous components)。这个函数会将输入图像分割成独立的区域,并返回每个区域的标签(labels)、大小、位置等信息。
当你调用`labels, stats, centroids = cv2.connectedComponentsWithStats(img)`时,参数解释如下:
1. `img`: 这是你需要分析的灰度或二值图像(通常是numpy数组),通常表示为8位或32位无符号整数。
```python
img = cv2.imread('image.png', cv2.IMREAD_GRAYSCALE)
```
2. `labels`: 返回的是一个uint8类型的二维数组,每个元素代表原图中的一个连通组件,值从0开始编号。
3. `stats`: 这是一个结构化的数组,包含每个连通区域的信息。每一行对应于`labels`数组的一个元素,列包括:
- label:该区域的标签ID。
- area:区域的像素总数。
- x,y,width,height:构成该区域的矩形左上角坐标以及宽度和高度。
4. `centroids`: 如果第三个输出参数被包含(例如上面的例子),它将是一个同样大小的uint8数组,包含了每个区域的中心点(x, y)坐标。如果没有提供此参数,它将不会被计算和返回。
使用这个函数的一般步骤如下:
```python
# 加载图片并转换为二值化
binary_img = cv2.threshold(img, threshold, 255, cv2.THRESH_BINARY)[1]
# 找到连接的组件及其统计信息
labels, stats, _ = cv2.connectedComponentsWithStats(binary_img)
# 然后你可以遍历stats,对每个区域做进一步的处理
for i in range(1, len(stats)):
# 根据label处理每个区域
```
import numpy as np import cv2 as cv # 加载图片 img = cv.imread('4.jpg') # 灰度化 img_gray = cv.cvtColor(img, cv.COLOR_BGR2GRAY) # 二值化 ret, thresh = cv.threshold(img_gray, 127, 255, cv.THRESH_BINARY) # 寻找连通域 num_labels, labels, stats, centroids = cv.connectedComponentsWithStats(thresh, connectivity=8) # 计算平均面积 areas = list() for i in range(num_labels): areas.append(stats[i][-1]) print("轮廓%d的面积:%d" % (i, stats[i][-1])) area_avg = np.average(areas[1:-1]) print("轮廓平均面积:", area_avg) # 筛选超过平均面积的连通域 image_filtered = np.zeros_like(img) for (i, label) in enumerate(np.unique(labels)): # 如果是背景,忽略 if label == 0: continue if stats[i][-1] < area_avg : image_filtered[labels == i] = 255 #cv.imshow("image_filtered", image_filtered) #cv.imshow("img", img) cv.imwrite('4.jpg',image_filtered ) cv.waitKey() cv.destroyAllWindows(),将上述代码转化为c++形式
以下是将上述Python代码转化为C++形式的示例:
```cpp
#include <iostream>
#include <opencv2/opencv.hpp>
using namespace std;
using namespace cv;
int main() {
// 加载图片
Mat img = imread("4.jpg");
// 灰度化
Mat img_gray;
cvtColor(img, img_gray, COLOR_BGR2GRAY);
// 二值化
Mat thresh;
threshold(img_gray, thresh, 127, 255, THRESH_BINARY);
// 寻找连通域
Mat labels, stats, centroids;
int num_labels = connectedComponentsWithStats(thresh, labels, stats, centroids, 8);
// 计算平均面积
vector<int> areas;
for (int i = 0; i < num_labels; i++) {
areas.push_back(stats.at<int>(i, CC_STAT_AREA));
cout << "轮廓" << i << "的面积:" << stats.at<int>(i, CC_STAT_AREA) << endl;
}
double area_avg = mean(Mat(areas))[0];
cout << "轮廓平均面积:" << area_avg << endl;
// 筛选超过平均面积的连通域
Mat image_filtered = Mat::zeros(img.size(), CV_8UC1);
for (int i = 1; i < num_labels; i++) {
if (stats.at<int>(i, CC_STAT_AREA) < area_avg) {
labels.setTo(0, labels == i);
} else {
image_filtered.setTo(255, labels == i);
}
}
// 显示结果
imshow("image_filtered", image_filtered);
imshow("img", img);
waitKey();
destroyAllWindows();
// 保存结果
imwrite("4.jpg", image_filtered);
return 0;
}
```
需要注意的是,C++中没有Python中的`list`类型,需要使用`vector`替代。同时,C++中没有Python中的`np.average()`方法,需要使用OpenCV的`mean()`方法计算平均值。此外,C++中的Mat类需要使用`.at()`方法访问元素,而不是Python中的下标索引。
阅读全文