poj_3349 哈希

题意:n片雪花,判断是否存在两个“相同的”雪花。

每个雪花由6个整数组成,两个雪花相同是说从一个雪花的任意一个数开始遍历可以得到另一个雪花的6个整数,无论是顺时针或者是逆时针方向,比如:123456有如下12种组合

123456 | 654321

234561 | 543216

345612 | 432165

456123 | 321654

561234 | 216543

612345 | 165432

这里认为其中任意两组数都是“相同的”。


思路:使用哈希,读入一个雪花时,将该雪花的12种可能性全部存入哈希表中,若后面的输入与前面的冲突,认为存在重复的雪花。

尽可能的避免若干个不同的雪花的哈希值相同,所以使用HASH_BASE为12*100000以上的最小素数,如果两个雪花哈希值相同,则依次比较该值下的所有雪花们,因为极有可能存在相同的情况。


code:

#include <iostream>
#include <fstream>
#include <cstdio>
#include <algorithm> 

using namespace std;

const int HASH_BASE = 1200007;//100000*12以上的最小素数
const int NODE_NUMBER = 1200010;//node个数

int hashTable[HASH_BASE];
int cur;

struct HashNode{
	int num[6];
	int next;
};

HashNode node[NODE_NUMBER];

void initHashTable()
{
	cur = 0;
	for (int i = 0; i < HASH_BASE; i++)
	{
		hashTable[i] = -1;
	}
}

unsigned getHash(int num[])
{
	unsigned int value = 0;
	for (int i = 0; i < 6; i++)
	{
		value += num[i];
	}
	return value%HASH_BASE;
}

void insertHashNode(int num[], unsigned value)
{
	for (int i = 0; i < 6; i++)
	{
		node[cur].num[i] = num[i];
	}
	node[cur].next = hashTable[value];
	hashTable[value] = cur;
	cur++;
}

bool compare(int num1[], int num2[])
{
	for (int i = 0; i < 6; i++)
	{
		if (num1[i] != num2[i])
			return false;
	}
	return true;
}

bool searchHash(int num[])
{
	unsigned value = getHash(num);
	int next = hashTable[value];
	while (next != -1)
	{
		if (compare(num, node[next].num))
			return true;
		next = node[next].next;
	}
	insertHashNode(num, value);
	return false;
}

int main()
{
	fstream in("input.txt");
	int num[2][12];//各种情况,num[0]->clockwise,num[1]->counterclockwise
	int i, j;
	int n;
	bool find = false;
	initHashTable();
	scanf("%d", &n);
	for (i = 0; i < n; i++)
	{
		//设置12种可能情况
		for (j = 0; j < 6; j++)
		{
			scanf("%d", &num[0][j]);
			num[0][6 + j] = num[0][j];
		}
		for (j = 0; j < 6; j++)
		{
			num[1][j] = num[1][6 + j] = num[0][5 - j];
		}
		if (find)
			continue;
		for (j = 0; j < 6; j++)
		{
			if (searchHash(num[0] + j) || searchHash(num[1] + j))
			{
				find = true;
				break;
			}
		}
	}
	if (find)
		printf("Twin snowflakes found.\n");
	else
		printf("No two snowflakes are alike.\n");
	//system("pause");
	return 0;
}

注意使用int num[2][12] 技巧来模拟一个雪花的12种情况。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值