【位操作笔记】计算奇偶性 使用乘法

2024-06-22 04:18

本文主要是介绍【位操作笔记】计算奇偶性 使用乘法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

计算奇偶性(Compute parity) 使用乘法

计算奇偶性(Compute parity)指的是,计算一个数所包含1的个数是奇数还是偶数,例如一个8位数0x5b = 0b‭0101 1011‬,其中1的个数为5,是奇数;一个8位数0xa3 = 0b‭‭1010 0011‬,其中1的个数为4,是偶数。该算法可以用于奇偶校验位的计算与验证。

算法说明

使用乘法运算,仅在8次运算中计算32位数值的奇偶性 。实际就是先通过乘法计算出这个数里bit位置1的个数,然后判断个数是奇数还是偶数。

如果设置了奇数位数,返回true,否则返回false。

实现代码

bool computing_parity(unsigned int val)
{val ^= val >> 1;val ^= val >> 2;val = (val & 0x11111111U) * 0x11111111U;return (val >> 28) & 1;
}

算法计算过程

算法分为8步。

  1. 第一步和第二步val ^= val >> 1

    用于将相邻的两个bit位进行异或,结果存在偶数位上(从第0位开始算)。因为异或操作和奇偶性的特点,这个操作只会减少置位的bit数,但不影响奇偶性。
    例如下面是个32位数
    00
    按照位置分成偶数位和奇数位
    11
    右移1位
    22
    两个数进行异或,会得到一个数,但我们实际只关心这个结果的偶数位,如下所示的32位数,只关心绿色格子。
    33
    在忽略掉奇数位上的数值(既上图的白色格子)后,这其实是相当于把32位数压缩成16位数,奇偶性相同。
    两个数异或不影响奇偶性,因为如果两个数分别为1和0,则是1 ^ 0 = 1,还是奇数;如果两个数分别为1和1,1 ^ 1 = 0,还是偶数;如果两个数分别为0和0,0 ^ 0 = 0,还是偶数。

  2. 第三步和第四步val ^= val >> 2

    将上一步得到的结果,再进行相邻的两个数进行异或操作。这步进一步减少置位的bit数,但不影响奇偶性。
    继续使用上一步得到的数
    33
    再分成两种颜色
    44
    右移两位
    55
    两个数进行异或,会得到一个数,但我们实际只关心这个结果的4倍数的位,如下所示的32位数,只关心橙色格子。
    66
    在忽略掉奇数位上的数值(既上图的白色格子)后,这其实是相当于把32位数压缩成16位数,奇偶性相同。现在实际就只有8个bit位有意义。

  3. 第五步 val & 0x11111111U

    剔除无用的bit位的数据。
    下图中白色格子内的数值被清空。
    66

    此时得到的32位数据如下,a,b,c,d,e,f,g,h都是表示一个bit位,数值未知。此时a+b+c+d+e+f+g+h的和的奇偶性就是原数值val的奇偶性。

000a 000b 000c 000d 000e 000f 000g 000h
  1. 第六步* 0x11111111U

    将上一步得到的结果乘以0x11111111U
    0

  2. 第七步val >> 28

    上一步计算的结果,28-31bit位置存储着a+b+c+d+e+f+g+h的和,将这个数右移28位,得到的就是这个数里bit位置1的总数。
    1

  3. 第八步& 1

    上一步得到了val这个数里bit位置1的总数,然后& 1得到数的奇偶性,完成计算。

例如一个数为0x355C4E25,二进制为0b‭00110101010111000100111000100101‬,共15bit

  1. val ^= val >> 1
‭‭           0011 0101 0101 1100 0100 1110 0010 0101‬
>> 1
--------------------------------------------------0001 1010 1010 1110 0010 0111 0001 0010‬
^          0011 0101 0101 1100 0100 1110 0010 0101‬
--------------------------------------------------0010 1111 1111 0010 0110 1001 0011 0111
  1. val ^= val >> 2;
           0010 1111 1111 0010 0110 1001 0011 0111
>> 2
--------------------------------------------------0000 1011 1111 1100 1001 1010 0100 1101
^          0010 1111 1111 0010 0110 1001 0011 0111
--------------------------------------------------0010 0100 0000 1110 1111 0011 0111 1010
  1. val & 0x11111111U
           0010 0100 0000 1110 1111 0011 0111 1010
&          0001 0001 0001 0001 0001 0001 0001 0001
--------------------------------------------------0000 0000 0000 0000 0001 0001 0001 0000
  1. val = (val & 0x11111111U) * 0x11111111U
           0000 0000 0000 0000 0001 0001 0001 0000
*          0001 0001 0001 0001 0001 0001 0001 0001
--------------------------------------------------0000 0000 0000 0000 0000 0000 0000 00000001 0001 0001 0001 0001 0001 00010001 0001 0001 0001 0001 00010001 0001 0001 0001 00010000 0000 0000 00000000 0000 00000000 00000000
--------------------------------------------------0011 0011 0011 0011 0011 0010 0001 0000
  1. (val >> 28) & 1
           0011 0011 0011 0011 0011 0010 0001 0000
>> 28
--------------------------------------------------0011
&                                                1
--------------------------------------------------1

上一步得到了这个数里bit位置1的总数为3,然后& 1得到的值为1,表示奇偶性为奇数。

完成奇偶性计算。

完整过程如下
2

拓展

计算64位的奇偶性 。使用乘法运算,同样只用8次运算就能完成计算。

bool computing_parity(unsigned long long val)
{val ^= val >> 1;val ^= val >> 2;val = (val & 0x1111111111111111UL) * 0x1111111111111111UL;return (val >> 60) & 1;
}

[参考资料]

Bit Twiddling Hacks By Sean Eron Anderson

[Hacker’s Delight] 作者: Henry S. Warren Jr.


本文链接:https://blog.csdn.net/u012028275/article/details/112596947

这篇关于【位操作笔记】计算奇偶性 使用乘法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1083266

相关文章

springboot security使用jwt认证方式

《springbootsecurity使用jwt认证方式》:本文主要介绍springbootsecurity使用jwt认证方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录前言代码示例依赖定义mapper定义用户信息的实体beansecurity相关的类提供登录接口测试提供一

go中空接口的具体使用

《go中空接口的具体使用》空接口是一种特殊的接口类型,它不包含任何方法,本文主要介绍了go中空接口的具体使用,具有一定的参考价值,感兴趣的可以了解一下... 目录接口-空接口1. 什么是空接口?2. 如何使用空接口?第一,第二,第三,3. 空接口几个要注意的坑坑1:坑2:坑3:接口-空接口1. 什么是空接

springboot security快速使用示例详解

《springbootsecurity快速使用示例详解》:本文主要介绍springbootsecurity快速使用示例,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝... 目录创www.chinasem.cn建spring boot项目生成脚手架配置依赖接口示例代码项目结构启用s

Python如何使用__slots__实现节省内存和性能优化

《Python如何使用__slots__实现节省内存和性能优化》你有想过,一个小小的__slots__能让你的Python类内存消耗直接减半吗,没错,今天咱们要聊的就是这个让人眼前一亮的技巧,感兴趣的... 目录背景:内存吃得满满的类__slots__:你的内存管理小助手举个大概的例子:看看效果如何?1.

java中使用POI生成Excel并导出过程

《java中使用POI生成Excel并导出过程》:本文主要介绍java中使用POI生成Excel并导出过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录需求说明及实现方式需求完成通用代码版本1版本2结果展示type参数为atype参数为b总结注:本文章中代码均为

Spring Boot3虚拟线程的使用步骤详解

《SpringBoot3虚拟线程的使用步骤详解》虚拟线程是Java19中引入的一个新特性,旨在通过简化线程管理来提升应用程序的并发性能,:本文主要介绍SpringBoot3虚拟线程的使用步骤,... 目录问题根源分析解决方案验证验证实验实验1:未启用keep-alive实验2:启用keep-alive扩展建

使用Java实现通用树形结构构建工具类

《使用Java实现通用树形结构构建工具类》这篇文章主要为大家详细介绍了如何使用Java实现通用树形结构构建工具类,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录完整代码一、设计思想与核心功能二、核心实现原理1. 数据结构准备阶段2. 循环依赖检测算法3. 树形结构构建4. 搜索子

GORM中Model和Table的区别及使用

《GORM中Model和Table的区别及使用》Model和Table是两种与数据库表交互的核心方法,但它们的用途和行为存在著差异,本文主要介绍了GORM中Model和Table的区别及使用,具有一... 目录1. Model 的作用与特点1.1 核心用途1.2 行为特点1.3 示例China编程代码2. Tab

SpringBoot使用OkHttp完成高效网络请求详解

《SpringBoot使用OkHttp完成高效网络请求详解》OkHttp是一个高效的HTTP客户端,支持同步和异步请求,且具备自动处理cookie、缓存和连接池等高级功能,下面我们来看看SpringB... 目录一、OkHttp 简介二、在 Spring Boot 中集成 OkHttp三、封装 OkHttp

使用Python实现获取网页指定内容

《使用Python实现获取网页指定内容》在当今互联网时代,网页数据抓取是一项非常重要的技能,本文将带你从零开始学习如何使用Python获取网页中的指定内容,希望对大家有所帮助... 目录引言1. 网页抓取的基本概念2. python中的网页抓取库3. 安装必要的库4. 发送HTTP请求并获取网页内容5. 解