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

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

位图(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; }

    // 设置指定位为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. 位运算的集合操作

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

// 求两个位数组的并集
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指令进一步优化。

 

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/968898.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

数据结构——结构体位域、typedef类型重定义、宏、共用体union、枚举、虚拟内存划分

一、结构体位域 1.1 结构体位域的基础 结构体位域&#xff1a;把结构体字节大小扣到极致的一个类型&#xff0c;以bit单位 格式&#xff1a;struct 位域体名{数据类型 位域名:位域大小;数据类型 位域名:位域大小;...};解析&#xff1a;位域体名&#xff1a;可有可无&#xff…

CZML 格式详解,javascript加载导出CZML文件示例

示例地址&#xff1a;https://dajianshi.blog.csdn.net/article/details/145573994 CZML 格式详解 1. 什么是 CZML&#xff1f; CZML&#xff08;Cesium Zipped Markup Language&#xff09;是一种基于 JSON 的文件格式&#xff0c;用于描述地理空间数据和时间动态场景。它专…

SQL递归技巧

1.读样例 with recursive cet_dpt(id, parent_id, path, org_category, level,depart_name) as (select id ,parent_id,depart_name as path,org_category,1 as level,sd.depart_namefrom isolarerp.sys_depart sdwhere del_flag 0and sd.org_code A09B15union al…

django配置跨域

1、第一种 from django.views.decorators.csrf import csrf_exemptcsrf_exempt第二种 安装 pip install django-cors-headers在配置文件settings.py进入 INSTALLED_APPS [..."corsheaders", # 添加 ]MIDDLEWARE [corsheaders.middleware.CorsMiddleware, # 添加…

设置mysql的主从复制模式

mysql设置主从复制模式似乎很容易&#xff0c;关键在于1&#xff09;主库启用二进制日志&#xff0c;2&#xff09;从库将主库设为主库。另外&#xff0c;主从复制&#xff0c;复制些什么&#xff1f;从我现在获得的还很少的经验来看&#xff0c;复制的内容有表&#xff0c;用户…

蓝耘智算平台:开启企业级 DeepSeek 智能助手的搭建捷径

文章目录 一、深度解密 DeepSeek 技术矩阵1.1 模型架构创新1.2 核心能力全景 二、私有化部署&#xff1a;企业的明智之选2.1 企业级部署场景2.2 硬件选型策略 三、蓝耘平台&#xff1a;部署全流程大揭3.1 环境准备阶段Step 1&#xff1a;访问蓝耘智算云官网完成企业认证Step 2&…

Android原生的HighCPU使用率查杀机制

摘要 原生的HighCPU使用率查杀机制是基于读取/proc/pid/stat中的utime stime后&#xff0c;根据CPU使用率 (utime stime / totalTime)*100%进行实现&#xff0c;当检测后台进程的CPU使用率超过阈值时&#xff0c;执行查杀和统计到电池数据中。 细节点&#xff1a; 1. 原生根…

数据库安全、分布式数据库、反规范化等新技术(高软19)

系列文章目录 3.7数据库安全、分布式数据库、反规范化等新技术 前言 本节数据库安全、分布式数据库、反规范化等新技术相关概念与技术。 一、数据库 1.数据库安全 2.数据库备份 二、分布式数据库 1.数据库分布 2.数据仓库 3.数据仓库结构 4.商业智能&#xff08;BI&#xf…

数据结构实现顺序表的尾插,尾删,按值查找/修改/删除,按下标查找/增加/删除

头文件&#xff1a;head.h #ifndef __HEAD_H__ #define __HEAD_H__#include <stdio.h> #include <string.h> #include <stdlib.h> #define MAXSIZE 20enum num {success,false-1};typedef int datatype;typedef struct {int len;datatype data[MAXSIZE]; }S…

新手自学:如何用gromacs对简单分子复合物进行伞形采样

1、建立体系: 1、将蛋白的pdb文件转化为gmx: gmx pdb2gmx -f 2BEG_model1_capped.pdb -ignh -ter -o complex.gro 这个网页可以实现将多肽序列转化为pdb: ProBuilder On-line 这个教程的蛋白2BFG包含两条链(chain A和B) 在生成的topol文件中,增加如下的内容,效果就…

2025 BabitMF 第一期开源有奖活动正式开启 !

为了促进开源社区的交流与成长&#xff0c;字节跳动开源的多媒体处理框架 BabitMF &#xff08;GitHub - BabitMF/bmf: Cross-platform, customizable multimedia/video processing framework. With strong GPU acceleration, heterogeneous design, multi-language support, e…

Ollama 自定义导入模型

文章目录 一、从 GGUF 导入1.1 CCUF 介绍1.2 导入方式 二、由模型直接导入2.1 模型下载2.2 使用 llama.cpp 进行转换&#xff08;1&#xff09;克隆 llama.cpp 库到本地&#xff0c;并安装相关库&#xff08;2&#xff09;环境验证&#xff08;3&#xff09;执行转换程序 2.3 使…

J6 X8B/X3C切换HDR各帧图像

1、OV手册上的切换命令 寄存器为Ox5074 各帧切换&#xff1a; 2、地平线control tool实现切换命令 默认HDR模式出图&#xff1a; HCG出图&#xff1a; LCG出图 SPD出图 VS出图

GESP5级语法知识(十一):高精度算法(一)

高精度加法&#xff1a; #include<iostream> #include<string> #include<algorithm> using namespace std; const int N501;//高精度数的最长长度 //c[]a[]b[]:高精度加法方案一&#xff1a;对应位相加&#xff0c;同时处理进位 void h_add_1(int a[],int b…

【Git版本控制器】:第二弹——工作区,暂存区,版本库,

&#x1f381;个人主页&#xff1a;我们的五年 &#x1f50d;系列专栏&#xff1a;Linux网络编程 &#x1f337;追光的人&#xff0c;终会万丈光芒 &#x1f389;欢迎大家点赞&#x1f44d;评论&#x1f4dd;收藏⭐文章 ​ 相关笔记&#xff1a; https://blog.csdn.net/djd…

Transformer 模型介绍(一)——综述

Transformer 是一种完全基于注意力机制的神经网络模型&#xff0c;首次在2017年的论文《Attention Is All You Need》中提出。该模型最初用于机器翻译任务&#xff0c;并在特定任务中表现优于谷歌的其他神经网络机器翻译模型。Transformer 也是 Seq2Seq&#xff08;序列到序列&…

【Linux】多线程 -> 从线程概念到线程控制

线程概念 在一个程序里的一个执行路线就叫做线程&#xff08;thread&#xff09;。更准确的定义是&#xff1a;线程是“一个进程内部的控制序列”。一切进程至少都有一个执行线程。线程在进程内部运行&#xff0c;本质是在进程地址空间内运行。在Linux系统中&#xff0c;在CPU眼…

.NET Web-静态文件访问目录浏览

一、Web根目录访问 创建wwwroot文件夹app.UseStaticFiles(); // 启⽤静态⽂件中间件url/路径 进行访问 二、Web根目录之外的文件 app.UseStaticFiles(new StaticFileOptions {FileProvider new PhysicalFileProvider(Path.Combine(builder.Environment.ContentRootPath,&qu…

cap1:TensorRT是什么?

文章目录 1、什么是 TensorRT&#xff1f;2、TensorRT 的优势3、TensorRT 加速 PyTorch 模型的基本流程3.1 训练模型和保存模型3.2 导出模型3.3 转换为 TensorRT 引擎3.4 加载与推理 4、基础环境配置4.1 安装nvidia驱动4.2 安装CUDA4.3 安装cuDNN 在软件工程领域&#xff0c;部…

JVM——堆的回收:引用计数发和可达性分析法、五种对象引用

目录 引用计数法和可达性分析法 引用计数法&#xff1a; 可达性分析算法&#xff1a; 五种对象引用 软引用&#xff1a; 弱引用&#xff1a; 引用计数法和可达性分析法 引用计数法&#xff1a; 引用计数法会为每个对象维护一个引用计数器&#xff0c;当对象被引用时加1&…