位图与位运算的深度联系:从图像处理到高效数据结构的C++实现与优化

ops/2025/2/13 22:40:01/

        在学习优选算法课程的时候,博主学习位运算了解到位运算的这个概念,之前没有接触过,就查找了相关的资料,丰富一下自身,当作课外知识来了解一下。

位图(Bitmap):

在计算机科学中有两种常见含义:

  1. 图像位图:由像素矩阵组成的图像数据结构(如BMP文件)。

  2. 位数组(Bit Array):用单个位(bit)表示布尔值的高效数据结构(如集合的紧凑表示)。

        位运算(Bitwise Operations)是直接操作二进制位的底层技术,在位图的存储、处理和优化中具有核心作用。


目录

一、图像位图与位运算

1. 颜色通道的位操作

示例:32位ARGB颜色值的操作

位运算优势:

2. 图像混合与透明度计算

优化技巧:

3. 位图压缩与掩码操作

示例:读取1位位图数据

关键点:

二、位数组(Bit Array)与位运算

1. 位数组的实现

内存优化:

2. 位运算的集合操作

性能优势:

3. 应用场景示例:布隆过滤器

核心优势:

三、性能优化与注意事项

1. 位运算的编译器优化

2. 平台相关性问题

3. 替代方案

四、总结


 

一、图像位图与位运算

1. 颜色通道的位操作

        图像位图中的每个像素通常由多个颜色通道(如RGB)组成,颜色值以整数形式存储。位运算可用于快速提取、合并或修改颜色通道。

示例:32位ARGB颜色值的操作

// 定义32位ARGB颜色结构(0xAARRGGBB)
using ARGB = uint32_t;// 提取颜色通道(通过位移和掩码)
constexpr ARGB getAlpha(ARGB color) { return (color >> 24) & 0xFF; }
constexpr ARGB getRed(ARGB color)   { return (color >> 16) & 0xFF; }
constexpr ARGB getGreen(ARGB color) { return (color >> 8)  & 0xFF; }
constexpr ARGB getBlue(ARGB color)  { return color & 0xFF; }// 合并颜色通道(通过位或和位移)
constexpr ARGB makeARGB(uint8_t a, uint8_t r, uint8_t g, uint8_t b) {return (a << 24) | (r << 16) | (g << 8) | b;
}

位运算优势:

  • 高效性:无需浮点运算,单周期完成操作。

  • 内存紧凑:32位整型存储一个像素,比结构体更节省空间。


2. 图像混合与透明度计算

使用位运算实现图像的Alpha混合(透明度混合),公式为:

result = (src * alpha) + (dst * (1 - alpha))

通过位运算优化计算过程:

// 快速Alpha混合(假设alpha范围为0-255)
ARGB blendARGB(ARGB src, ARGB dst) {uint8_t alpha = getAlpha(src);uint8_t invAlpha = 255 - alpha;uint8_t r = (getRed(src) * alpha + getRed(dst) * invAlpha) >> 8;uint8_t g = (getGreen(src) * alpha + getGreen(dst) * invAlpha) >> 8;uint8_t b = (getBlue(src) * alpha + getBlue(dst) * invAlpha) >> 8;return makeARGB(255, r, g, b);
}

优化技巧:

  • 预移位乘法:将alphainvAlpha左移8位,用整数乘法的溢出特性优化除法(>> 8等价于除以256)。


3. 位图压缩与掩码操作

        在位图文件(如BMP)中,像素数据可能按位压缩存储。例如,1位位图(黑白图像)中,每个像素用一个位表示。

示例:读取1位位图数据

void read1BitBMP(uint8_t* data, int width, int height) {int rowSize = (width + 31) / 32 * 4; // BMP每行对齐到4字节for (int y = 0; y < height; y++) {for (int x = 0; x < width; x++) {int byteIndex = y * rowSize + x / 8;int bitIndex = 7 - (x % 8); // BMP位顺序从高位到低位bool isBlack = (data[byteIndex] >> bitIndex) & 1;// 处理像素...}}
}

关键点:

  • 位对齐:BMP文件每行数据按4字节对齐,需计算rowSize

  • 位顺序:BMP中每个字节的最高位(MSB)对应最左侧像素。


二、位数组(Bit Array)与位运算

        位数组是一种用单个位表示布尔值的数据结构,常用于高效存储大量标志(如布隆过滤器、权限系统)。

1. 位数组的实现

在C++中,通常使用std::bitset或原生数组实现位数组。以下为原生实现:

class BitArray {
public:BitArray(size_t size) : size(size) {data = new uint32_t[(size + 31) / 32]; // 每32位存储一个整数}~BitArray() { delete[] data; }// 设置指定位为1void set(size_t index) {data[index / 32] |= (1U << (index % 32));}// 清除指定位为0void reset(size_t index) {data[index / 32] &= ~(1U << (index % 32));}// 检查位是否为1bool test(size_t index) const {return (data[index / 32] >> (index % 32)) & 1U;}private:uint32_t* data;size_t size;
};

内存优化:

  • 空间效率:存储N个标志仅需N/8字节,比bool数组(每个元素占1字节)节省87.5%内存。


2. 位运算的集合操作

位数组支持高效的集合运算(如并集、交集),通过位运算批量处理。

// 求两个位数组的并集
BitArray union(const BitArray& a, const BitArray& b) {BitArray result(a.size);for (size_t i = 0; i < a.blockCount(); i++) {result.data[i] = a.data[i] | b.data[i];}return result;
}// 求两个位数组的交集
BitArray intersection(const BitArray& a, const BitArray& b) {BitArray result(a.size);for (size_t i = 0; i < a.blockCount(); i++) {result.data[i] = a.data[i] & b.data[i];}return result;
}

性能优势:

  • 并行处理:单个位运算指令可同时处理32位(或64位)数据。


3. 应用场景示例:布隆过滤器

布隆过滤器(Bloom Filter)利用位数组和多个哈希函数,高效判断元素是否存在于集合中。

class BloomFilter {
public:BloomFilter(size_t size, size_t numHashes) : bitArray(size), numHashes(numHashes) {}void insert(const std::string& key) {for (size_t i = 0; i < numHashes; i++) {size_t hash = std::hash<std::string>{}(key + std::to_string(i));bitArray.set(hash % bitArray.size());}}bool contains(const std::string& key) const {for (size_t i = 0; i < numHashes; i++) {size_t hash = std::hash<std::string>{}(key + std::to_string(i));if (!bitArray.test(hash % bitArray.size())) return false;}return true;}private:BitArray bitArray;size_t numHashes;
};

核心优势:

  • 低内存占用:存储百万级数据仅需几MB内存。

  • 常数时间查询:查询时间复杂度为O(k),k为哈希函数数量。


三、性能优化与注意事项

1. 位运算的编译器优化

  • 内联函数:使用constexprinline关键字提示编译器内联小型位操作函数。

  • 循环展开:编译器可能自动展开涉及位运算的循环(如-O3优化)。

2. 平台相关性问题

  • 字节序(Endianness):位顺序在不同CPU架构(如x86 vs ARM)可能不同,影响跨平台数据解析。

  • 位移操作符行为:右移有符号数时,C++标准未规定填充位(符号扩展或补零),需显式使用无符号类型。

3. 替代方案

  • SIMD指令集:如AVX-512,可同时对512位数据进行位运算,进一步提升性能。

  • 标准库工具:优先使用std::bitset(固定大小)或std::vector<bool>(动态大小,可能压缩存储)。


四、总结

位图与位运算的联系体现在两个层面:

  1. 图像位图:位运算用于颜色通道操作、图像混合、压缩存储,提升处理效率。

  2. 位数组:位运算实现紧凑存储和高效集合操作,应用于布隆过滤器、权限系统等场景。

        在C++中,通过合理使用位掩码、位移和逻辑操作,可显著优化内存占用和计算性能。实际开发中需注意平台差异,并结合标准库或SIMD指令进一步优化。

 

 


http://www.ppmy.cn/ops/158157.html

相关文章

DeepSeek和ChatGPT的对比

最近DeepSeek大放异彩&#xff0c;两者之间有什么差异呢&#xff1f;根据了解到的信息&#xff0c;简单做了一个对比。 DeepSeek 和 ChatGPT 是两种不同的自然语言处理&#xff08;NLP&#xff09;模型架构&#xff0c;尽管它们都基于 Transformer 架构&#xff0c;但在设计目标…

Python----Python高级(网络编程:网络基础:发展历程,IP地址,MAC地址,域名,端口,子网掩码,网关,URL,DHCP,交换机)

一、网络 早期的计算机程序都是在本机上运行的&#xff0c;数据存储和处理都在同一台机器上完成。随着技术的发展&#xff0c;人 们开始有了让计算机之间相互通信的需求。例如安装在个人计算机上的计算器或记事本应用&#xff0c;其运行环 境仅限于个人计算机内部。这种设置虽然…

5G无线网络技术深度解析

5G无线网络技术深度解析 一、5G物理层核心技术 灵活参数集&#xff08;Numerology&#xff09; 子载波间隔&#xff08;SCS&#xff09;&#xff1a;5G NR定义了5种子载波间隔&#xff08;15/30/60/120/240 kHz&#xff09;&#xff0c;SCS与频段强相关&#xff1a; FR1&#x…

go 语言设置 商城首页

1&#xff1a;前端传递的数据结构: {"page_type": 10,"page_name": "商城首页","page_data": {"page": {"params": {"name": "商城首页","title": "萤火商城2.0","…

ubuntu20.04+RTX4060Ti大模型环境安装

装显卡驱动 这里是重点&#xff0c;因为我是跑深度学习的&#xff0c;要用CUDA&#xff0c;所以必须得装官方的驱动&#xff0c;Ubuntu的附件驱动可能不太行. 进入官网https://www.nvidia.cn/geforce/drivers/&#xff0c;选择类型&#xff0c;最新版本下载。 挨个运行&#…

HTML之JavaScript对象声明

HTML之JavaScript对象声明 常用&#xff1a;方式1&#xff1a;new Object() 创建一个空对象方式2&#xff1a;{属性名:属性值,属性名:属性值,...函数名:function(){}} 创建一个对象<!DOCTYPE html> <html lang"en"> <head><meta charset&quo…

JVM的性能优化

1.方法内联 方法内联,是指 JVM在运行时将调用次数达到一定阈值的方法调用替换为方法体本身 ,从而消除调用成本,并为接下来进一步的代码性能优化提供基础,是JVM的一个重要优化手段之一。 注: C++的inline属于编译后内联,但是java是运行时内联 简单通俗的讲就是把方法内部调…

MindStudio制作MindSpore TBE算子(四)算子测试(ST测试-Ascend910B/ModelArts)--失败尝试

上一节&#xff0c;MindStudio制作MindSpore TBE算子&#xff08;三&#xff09;算子测试&#xff08;ST测试&#xff09;&#xff0c;因此缺乏对应的硬件环境导致无法进行ST测试&#xff0c;导致难以自安&#xff0c;今天搞来Ascend910B服务器来填坑&#xff0c;看看是否是硬件…