bitmap以及异或运算法
一
有40亿个整数,再给一个新的整数,需要判断新的整数是否在1亿个整数中。
此处需要用到bitmap方法,每个整数用一个bit表示,1表示存在,0表示不存在。因此一个4字节的int=32个bit也就是可以表示32个数字,那么1亿个数字需要 100000000/32个int型。也就是需要申请100000000/32个字节的数组。需要耗费25M的内存。
这种方法也降低了数组需要的内存。
申请数组,N代表多少个数字,WORD代表一个int字节能表示多少位数字
#define WORD 32
#define SHIFT 5
#define MASK 0x1f
#define N 100000000
int bitmap[1 + N / WORD];
整型数组全部初始化为0.
void init_bitmap()
{
int size;
size = 1 + N / WORD;
memset(bitmap, 0, size);
}
将数字对应的Bit置为1。对于号码 89256,由于89256 mod 32=2789…8,这样我们应该置a[2789]中32位字符串的第8位(从低位数起)为1。SHIFT在这里表示右移的个数,i>>SHIFT就等于89256 mod 32=2789. MASK也就是16进制的31. i & MASK就等于i/32(求余操作)
void set(i)
{
bitmap[i >> SHIFT] |= (1<< (i & MASK));
}
判断是否存在,和前面用到的方法一样,不同的是这里运用的是与运算
int Judge_exist(i)
{
return bitmap[i >> SHIFT] & (1 << (i & MASK));
}
将对应的数代表的bit位清零
void clear(i)
{
bitmap[i >> SHIFT] &= ~(1 << (i & MASK));
}
二:
在不创建临时变量的情况下进行交换两个数。既然不能创建临时变量,那么就需要采取异或运算
对bit进行异或运算。
1 a^=b 将a和b bit中不一样的点位提取出来存入a
2 b^=a 将b中和a中不一样的点位取反,也就实现了b=a
3 a^=b 将a中和b不一样的点位取反,也就实现了a=b
例如a=3:0011 b=7:0111
a^=b a=0100
b^=a b=0011=3
a^=b a=0111=7
代码实现
void swap_without_temp(int *a, int *b)
{
(*a) ^= (*b);
(*b) ^= (*a);
(*a) ^= (*b);
}
三:
不进行移位操作判断数字的bit有多少个1
代码如下:
int count_bit_1(int a)
{
int count = 0;
while (a > 0)
{
count++;
a = a & a - 1;
}
return count;
}
n-1发生了什么
(1) 二进制数n,n-1后,如果最后一位是0,将向前一位借2,2-1=1。最后一位为1。如果前一位为0,将继续向前一位借2,加上本身少掉的1.则变为1。一直遇到1。减为0.
所以 二进制 xxxx10000-1 = xxxx01111
(2) n&n-1
按照上述 n=xxxx10000,n-1=xxxx01111
xxxx10000
xxxx01111
四
给定一个非空整数数组,除了某个元素只出现一次以外,其余每个元素均出现两次。
找出那个只出现了一次的元素。 相同的值进行异或等于0,如果其余元素都出现两次,而只有一个元素出现一次,那么把数组中所有的
元素都进行异或运算
int findOnce(int arr[], int size)
{
int i;
int result;
result = 0;
for (i = 0; i < size; i++)
{
result ^= arr[i];
}
return result;
}
五
给定一个整数数组 nums,其中恰好有两个元素只出现一次,其余所有元素均出现两次。找出只出现一次的那两个元素。
1 首先将所有元素进行异或运算,得到的结果是仅出现一次的两个元素异或值
2 采用differ &= -differ 得到只出现在其中一个元素中的1个点位
3 用该点位去和数组中的元素做与运算。这个点位存在的所有数在一起做异或,这个点位不存在的所有数一起做异或
这样就可以将区分的做异或运算从而分别得到两个值
void findOnce_update(int arr[], int size,int *first,int *second)
{
int i;
int differ;
differ = 0;
for (i = 0; i < size; i++)
{
differ ^= arr[i];
}
differ &= -differ;
for (i = 0; i < size; i++)
{
if ((differ & arr[i]) == 0)
*first ^= arr[i];
else
*second ^= arr[i];
}
}
bitmap以及异或运算法的更多相关文章
- C# 异或校验算法
C# 的异或校验算法 直接上代码 public partial class FormCRC : Form { public FormCRC() { InitializeComponent(); } p ...
- Java运算符 算术运算法
运算符 算术运算法:+,-,*,/,%,++,– 复制运算符:= 关系运算符:>,<,>=,<=,==,!= instanceof 逻辑运算符:&&,||,! ...
- py02_04:三元运算法
a if a > b else c # a>b 成立,则为真,如果a>b为假,则返回c
- [Java 泥水匠] Java Components 之二:算法篇之项目实践中的位运算符(有你不懂的哦)
作者:泥沙砖瓦浆木匠网站:http://blog.csdn.net/jeffli1993个人签名:打算起手不凡写出鸿篇巨作的人,往往坚持不了完成第一章节. 交流QQ群:[编程之美 365234583] ...
- 不同的GCD算法
分类: C语言程序2014-10-08 15:10 28人阅读 评论(0) 收藏 举报 gcdC语言程序位运算 早在公元前300年左右,欧几里得就在他的著作<几何原本>中给出了高效的解法- ...
- 记一道有意思的算法题Rotate Image(旋转图像)
题出自https://leetcode.com/problems/rotate-image/ 内容为: You are given an n x n 2D matrix representing an ...
- 《OD学算法》排序
参考 http://www.cnblogs.com/kkun/archive/2011/11/23/2260312.html http://blog.csdn.net/wuxinyicomeon/ar ...
- C#中常用的排序算法的时间复杂度和空间复杂度
常用的排序算法的时间复杂度和空间复杂度 常用的排序算法的时间复杂度和空间复杂度 排序法 最差时间分析 平均时间复杂度 稳定度 空间复杂度 冒泡排序 O(n2) O(n2) 稳定 O(1) 快速排序 ...
- STL函数模板(即算法)一览
查找算法 adjacent_find:找出一个串中第一个不符合次序的地方 find,find_if:找出第一个符合条件的元素 find_first_of:在一个串中寻找第一个与另一个串中任意一个元素相 ...
随机推荐
- jmeter+nmon+crontab简单的执行接口定时压测
一.概述 临时接到任务要对系统的接口进行压测,上面的要求就是:压测,并发2000 在不熟悉系统的情况下,按目前的需求,需要做的步骤: 需要有接口脚本 需要能监控系统性能 需要能定时执行脚本 二.观察 ...
- Kotlin属性引用进阶与构造方法引用
继续还是探讨Kotlin反射相关的知识点,说实话这块不是太好理解,待在实际工作中去对它进行实践慢慢来加深印象. 属性引用进阶: 在Kotlin中的反射其实是跟Java的反射有对应关系的,具体相关的定义 ...
- python测试开发django-rest-framework-59.restful接口开发
前言 REST 不是什么具体的软件或者代码,而是一种思想.现在流行前后端分离开发项目,一般用 json 来交换数据. 相信写过模板的同学都知道,只要哪怕页面中的数据有一丝丝变动,那整个页面都需要重新渲 ...
- 实现批量添加10个用户,用户名为user01-10,密码为user后面跟3个随机字符
#!/bin/bash ` do user="user$i" password=$( | md5sum | ) useradd user$i echo "$user$pa ...
- Hibernate的悲观锁和乐观锁
前一篇博客我们从数据库角度分析,锁可以分为三种,分别为共享锁,独占锁和更新锁.我们从程序的角度来看锁可以分为两种类型,悲观锁和乐观锁,Hibernate提供对这两种锁 的支持,我们来了解一下Hiber ...
- 【python】requests 异常处理
以下是request.exceptions下的各种异常错误: RequestException: HTTPError(RequestException) UnrewindableBodyError(R ...
- hiveSQL常用日期函数
注意 MM,DD,MO,TU 等要大写 Hive 可以在 where 条件中使用 case when 已知日期 要求日期 语句 结果 本周任意一天 本周一 select date_sub(next_d ...
- Laravel —— could not find driver
Laravel 中的数据库是以 PDO 的方式连接的 数据库连接失败时,先检查问题所在,再对症下药 本文以 pgsql 为例 1.判断 pgsql 是否启动 $ ps -ef | grep pgsql ...
- learning java FileVisitor 遍丽文件及路径
import java.io.IOException; import java.nio.file.*; import java.nio.file.attribute.BasicFileAttribut ...
- WinDbg常用命令系列---内存查看d*
d*命令显示给定范围内的内存内容. d{a|b|c|d|D|f|p|q|u|w|W} [Options] [Range] dy{b|d} [Options] [Range] d [Options] [ ...