Opencv 学习笔记:手动绘制彩色图像的 RGB 通道直方图

直方图是分析图像像素分布的核心工具,本文通过自定义函数实现彩色图像 BGR 三通道直方图的手动绘制,新手可直观理解直方图的计算与可视化逻辑。

核心代码实现

python 复制代码
import cv2 as cv
import numpy as np

# 定义计算并绘制直方图的函数
def calcanddrawhist(src, color):
    # 1. 计算直方图:[图像], [通道索引], 掩膜, 直方图条数, 像素值范围
    hist = cv.calcHist([src], [0], None, [256], [0.0, 255.0])
    
    # 2. 获取直方图最大值(用于归一化)
    min_val, max_val, min_loc, max_loc = cv.minMaxLoc(hist)
    
    # 3. 创建256×256的空白画布(用于绘制直方图)
    hist_img = np.zeros([256, 256, 3], np.uint8)
    hpt = int(0.9 * 256)  # 限制直方图高度(预留10%边距)
    
    # 4. 遍历每个像素值,绘制直方图线条
    for h in range(256):
        # 归一化直方图数值到画布高度范围
        intensity = int(hist[h] * hpt / max_val)
        # 绘制垂直线:从底部向上,颜色为指定通道色
        cv.line(hist_img, (h, 256), (h, 256-intensity), color)
    
    return hist_img

# 主程序
src = cv.imread('.\image\1.bmp')
if src is None:
    print('could not load image')
    exit()
cv.imshow('src', src)

# 拆分BGR通道
b, g, r = cv.split(src)

# 分别计算并绘制各通道直方图
hist_imgb = calcanddrawhist(b, [255, 0, 0])  # 蓝色通道(B)
hist_imgg = calcanddrawhist(g, [0, 255, 0])  # 绿色通道(G)
hist_imgr = calcanddrawhist(r, [0, 0, 255])  # 红色通道(R)

# 显示各通道直方图
cv.imshow('his_b', hist_imgb)
cv.imshow('his_g', hist_imgg)
cv.imshow('his_r', hist_imgr)

cv.waitKeyEx(0)
cv.destroyAllWindows()

关键知识点解析

1. 直方图绘制核心流程

步骤 核心 API / 操作 作用说明
计算直方图 cv.calcHist() 统计每个像素值(0-255)的出现次数
归一化 hist[h] * hpt/max_val 将直方图数值缩放到画布高度范围,避免超出画布
绘制线条 cv.line() 以像素值为 x 轴,出现次数为 y 轴,绘制垂直线条

2. 关键参数说明

  • cv.calcHist([src], [0], None, [256], [0.0, 255.0])
    • [src]:输入图像(需为列表格式);
    • [0]:通道索引(单通道处理时固定为 0);
    • None:无掩膜(分析整幅图像);
    • [256]:直方图条数(对应 0-255 像素值);
    • [0.0, 255.0]:像素值范围。
  • hpt = int(0.9 * 256):预留 10% 画布边距,避免直方图顶边溢出。

3. 优化与扩展技巧

  • 合并显示 :将三个通道直方图拼接为一张图,更便于对比:

    python

    运行

    python 复制代码
    hist_merge = cv.hconcat([hist_imgb, hist_imgg, hist_imgr])
    cv.imshow('hist_merge', hist_merge)
  • 灰度图直方图 :无需拆分通道,直接传入灰度图即可:

    python

    运行

    python 复制代码
    gray = cv.cvtColor(src, cv.COLOR_BGR2GRAY)
    hist_gray = calcanddrawhist(gray, [255, 255, 255])  # 白色线条
  • 效率优化:循环绘制可改用 numpy 向量化操作,提升大图像处理速度。

总结

  1. 直方图绘制核心是cv.calcHist()统计像素分布,再通过归一化 + 画线实现可视化;
  2. 彩色图像需拆分 BGR 通道分别绘制,通道颜色与线条颜色对应更易区分;
  3. 归一化步骤是关键,可避免直方图数值超出画布范围导致显示异常。
相关推荐
杨云龙UP7 小时前
从0到1快速学会Linux操作系统(基础),这一篇就够了!
linux·运维·服务器·学习·ubuntu·centos·ssh
头疼的程序员8 小时前
计算机网络:自顶向下方法(第七版)第八章 学习分享(三)
网络·学习·计算机网络
minglie18 小时前
zynq环境用opencv测摄像头
人工智能·opencv·计算机视觉
_李小白8 小时前
【OSG学习笔记】Day 37: NodeVisitor(顶点访问器)
笔记·学习
Fleshy数模9 小时前
基于MediaPipe实现人体姿态与脸部关键点检测
python·opencv·计算机视觉
程序员雷欧9 小时前
大模型应用开发学习第八天
大数据·人工智能·学习
SccTsAxR9 小时前
算法基石:手撕离散化、递归与分治
c++·经验分享·笔记·算法
晓晓hh9 小时前
JavaSE学习——set集合和Map映射
学习
輕華9 小时前
OpenCV答题卡识别:从图像预处理到自动评分
人工智能·opencv·计算机视觉
西梅汁10 小时前
C++ 观察者模式
笔记