内存限制下查找数组的重复数

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/moses1213/article/details/52192501

题目:给定一个数组,包含1到n的整数,n最大为32000,数组可能含有重复的值,且n的取值不定。若只有4KB的内存可用,该如果打印数组中所有重复的元素?

思路:检测重复数的题目做过很多了,通常是建议一个哈希表,对访问过的元素进行标记,当访问的元素未标记时,标记这一元素,如果已经标记,说明重复,输出。哈希表可以用数组实现,假设申请int型的大小为32000的数组,那么使用的空间大小为32000*4B,超过了内存限制。转而去用位操作实现,一个int型占32位,总共需要32000个位,所以有1000个int型的数就可以了,这时候占用内存空间为1000*4B小于4KB(这里的K=1024>1000)。

class BitSet
{
private: 
	int* bitset;

public: 
	BitSet(int size);
	bool Get(int num);
	void Set(int num);
};

BitSet::BitSet(int size)
{
	bitset = new int[size>>5];    //除以32
}

bool BitSet::Get(int num)
{
	int wordNumber = num >> 5;
	int bitNumber = num & 0x1F;
	return (bitset[wordNumber] & (1 << bitNumber)) != 0;
}

void BitSet::Set(int num)
{
	int wordNumber = num >> 5;  //除以32
	int bitNumber = num & 0x1F; //除2取余数
	bitset[wordNumber] |= 1 << bitNumber;
}

void CheckRepetitive(int* array, int length)
{
	if(array == NULL || length < 1)
		return;
	
	BitSet hash(length);	
	for(int i = 0; i < length; ++i)
	{
		int num = array[i] - 1;  //位从0开始,而数组从1开始
		if(hash.Get(num))
			cout << array[i] << " ";
		else
			hash.Set(num);
	}
}


猜你喜欢

转载自blog.csdn.net/moses1213/article/details/52192501