在学习优选算法课程的时候,博主学习位运算了解到位运算的这个概念,之前没有接触过,就查找了相关的资料,丰富一下自身,当作课外知识来了解一下。
位图(Bitmap):在计算机科学中有两种常见含义:
图像位图:由像素矩阵组成的图像数据结构(如BMP文件)。
位数组(Bit Array):用单个位(bit)表示布尔值的高效数据结构(如集合的紧凑表示)。
位运算(Bitwise Operations)是直接操作二进制位的底层技术,在位图的存储、处理和优化中具有核心作用。
目录
[1. 颜色通道的位操作](#1. 颜色通道的位操作)
[2. 图像混合与透明度计算](#2. 图像混合与透明度计算)
[3. 位图压缩与掩码操作](#3. 位图压缩与掩码操作)
[二、位数组(Bit Array)与位运算](#二、位数组(Bit Array)与位运算)
[1. 位数组的实现](#1. 位数组的实现)
[2. 位运算的集合操作](#2. 位运算的集合操作)
[3. 应用场景示例:布隆过滤器](#3. 应用场景示例:布隆过滤器)
[1. 位运算的编译器优化](#1. 位运算的编译器优化)
[2. 平台相关性问题](#2. 平台相关性问题)
[3. 替代方案](#3. 替代方案)
一、图像位图与位运算
1. 颜色通道的位操作
图像位图中的每个像素通常由多个颜色通道(如RGB)组成,颜色值以整数形式存储。位运算可用于快速提取、合并或修改颜色通道。
示例:32位ARGB颜色值的操作
cpp
// 定义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混合(透明度混合),公式为:
cpp
result = (src * alpha) + (dst * (1 - alpha))
通过位运算优化计算过程:
cpp
// 快速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);
}
优化技巧:
- 预移位乘法 :将
alpha
和invAlpha
左移8位,用整数乘法的溢出特性优化除法(>> 8
等价于除以256)。
3. 位图压缩与掩码操作
在位图文件(如BMP)中,像素数据可能按位压缩存储。例如,1位位图(黑白图像)中,每个像素用一个位表示。
示例:读取1位位图数据
cpp
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
或原生数组实现位数组。以下为原生实现:
cpp
class BitArray {
public:
BitArray(size_t size) : size(size) {
data = new uint32_t[(size + 31) / 32]; // 每32位存储一个整数
}
~BitArray() { delete[] data; }
// 设置指定位为1
void set(size_t index) {
data[index / 32] |= (1U << (index % 32));
}
// 清除指定位为0
void reset(size_t index) {
data[index / 32] &= ~(1U << (index % 32));
}
// 检查位是否为1
bool 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. 位运算的集合操作
位数组支持高效的集合运算(如并集、交集),通过位运算批量处理。
cpp
// 求两个位数组的并集
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)利用位数组和多个哈希函数,高效判断元素是否存在于集合中。
cpp
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. 位运算的编译器优化
-
内联函数 :使用
constexpr
或inline
关键字提示编译器内联小型位操作函数。 -
循环展开 :编译器可能自动展开涉及位运算的循环(如
-O3
优化)。
2. 平台相关性问题
-
字节序(Endianness):位顺序在不同CPU架构(如x86 vs ARM)可能不同,影响跨平台数据解析。
-
位移操作符行为:右移有符号数时,C++标准未规定填充位(符号扩展或补零),需显式使用无符号类型。
3. 替代方案
-
SIMD指令集:如AVX-512,可同时对512位数据进行位运算,进一步提升性能。
-
标准库工具 :优先使用
std::bitset
(固定大小)或std::vector<bool>
(动态大小,可能压缩存储)。
四、总结
位图与位运算的联系体现在两个层面:
-
图像位图:位运算用于颜色通道操作、图像混合、压缩存储,提升处理效率。
-
位数组:位运算实现紧凑存储和高效集合操作,应用于布隆过滤器、权限系统等场景。
在C++中,通过合理使用位掩码、位移和逻辑操作,可显著优化内存占用和计算性能。实际开发中需注意平台差异,并结合标准库或SIMD指令进一步优化。