有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以及异或运算法的更多相关文章

  1. C# 异或校验算法

    C# 的异或校验算法 直接上代码 public partial class FormCRC : Form { public FormCRC() { InitializeComponent(); } p ...

  2. Java运算符 算术运算法

    运算符 算术运算法:+,-,*,/,%,++,– 复制运算符:= 关系运算符:>,<,>=,<=,==,!= instanceof 逻辑运算符:&&,||,! ...

  3. py02_04:三元运算法

    a if a > b else c     #  a>b 成立,则为真,如果a>b为假,则返回c

  4. [Java 泥水匠] Java Components 之二:算法篇之项目实践中的位运算符(有你不懂的哦)

    作者:泥沙砖瓦浆木匠网站:http://blog.csdn.net/jeffli1993个人签名:打算起手不凡写出鸿篇巨作的人,往往坚持不了完成第一章节. 交流QQ群:[编程之美 365234583] ...

  5. 不同的GCD算法

    分类: C语言程序2014-10-08 15:10 28人阅读 评论(0) 收藏 举报 gcdC语言程序位运算 早在公元前300年左右,欧几里得就在他的著作<几何原本>中给出了高效的解法- ...

  6. 记一道有意思的算法题Rotate Image(旋转图像)

    题出自https://leetcode.com/problems/rotate-image/ 内容为: You are given an n x n 2D matrix representing an ...

  7. 《OD学算法》排序

    参考 http://www.cnblogs.com/kkun/archive/2011/11/23/2260312.html http://blog.csdn.net/wuxinyicomeon/ar ...

  8. C#中常用的排序算法的时间复杂度和空间复杂度

    常用的排序算法的时间复杂度和空间复杂度   常用的排序算法的时间复杂度和空间复杂度 排序法 最差时间分析 平均时间复杂度 稳定度 空间复杂度 冒泡排序 O(n2) O(n2) 稳定 O(1) 快速排序 ...

  9. STL函数模板(即算法)一览

    查找算法 adjacent_find:找出一个串中第一个不符合次序的地方 find,find_if:找出第一个符合条件的元素 find_first_of:在一个串中寻找第一个与另一个串中任意一个元素相 ...

随机推荐

  1. go语言笔记2

    上接Go语言学习笔记(一) 11    Go错误处理11.1    nil函数通常在最后的返回值中返回错误信息.使用errors.New 可返回一个错误信息: package main         ...

  2. test20190903 JKlover

    100+65+100=265,T2就差了一点. 乌合之众 给出一个 n × n 的, 元素为自然数的矩阵.这个矩阵有许许多多个子矩阵, 定义它的所有子矩阵形成的集合为 S . 对于一个矩阵 k , 定 ...

  3. axure快速上手

    Axure RP是一个专业的快速原型设计工具.Axure(发音:Ack-sure),代表美国Axure公司:RP则是Rapid Prototyping(快速原型)的缩写.Axure RP是美国Axur ...

  4. .net 代码调用cmd执行.exe程序,获取控制台输出信息

    使用.net core 对老项目升级, .net core 使用TripleDES.Create() 加密众iv字节限制 与 framework中的不同, 新项目还需要兼容老项目版本,还不想通过web ...

  5. 学习:类和对象——对象模型和this指针

    成员变量和成员函数分开存储: 在C++中,类内的成员变量和成员函数分开存储 第一点:空对象占用内存空间1个字节 第二点:只有非静态成员变量才属于类的对象上,非静态成员函数和静态成员函数和静态成员变量不 ...

  6. fitnesse wiki界面设置变量

    有时候我们可能多组测试数据会到同一个值,这样我们就可以设置一个变量,修改时只需要修改一个地方即可,而不需要对每组测试数据的这列数据进行修改 如下图: (1)定义变量:!define A {10}  , ...

  7. 14.go内置的rate包学习2(有花操作,必看)

    package main import ( "fmt" "golang.org/x/time/rate" "time" ) func mai ...

  8. 【后缀数组】【LuoguP2408】 不同子串个数

    题目链接 题目描述 给你一个长为N的字符串,求不同的子串的个数 我们定义两个子串不同,当且仅当有这两个子串长度不一样 或者长度一样且有任意一位不一样. 子串的定义:原字符串中连续的一段字符组成的字符串 ...

  9. 原创:协同过滤之ALS

    推荐系统的算法,在上个世纪90年代成型,最早应用于UserCF,基于用户的协同过滤算法,标志着推荐系统的形成.首先,要明白以下几个理论:①长尾理论②评判推荐系统的指标.之所以需要推荐系统,是要挖掘冷门 ...

  10. Android中LayoutInflater()方法

    在实际开发中LayoutInflater这个类还是非常有用的,它的作用类似于findViewById().不同点是LayoutInflater是用来找res/layout/下的xml布局文件,并且实例 ...