xtuoj 哈希

题目

思路

代码流程

预处理素数:用埃氏筛法生成足够大范围内的所有素数(按升序排列),为后续查找做准备;

缩小查找范围:基于鸽巢原理,直接定位到第一个≥n/2 的素数(sz<n/2 时必然不满足要求,无需校验)

校验合法素数:从上述素数开始,按升序逐个验证:

  • 统计每个素数 sz 下,数组元素哈希值的出现次数(计数原理);
  • 找到第一个满足 "所有哈希值计数≤2" 的素数,即为答案。

核心依据

  • 选素数:素数作为哈希表大小能让哈希值分布更均匀,减少规律性冲突;
  • 范围优化:鸽巢原理(n 个元素放入 sz 个位置,sz<n/2 则必存在≥3 个元素映射到同一位置);
  • 合法性校验:计数原理(统计哈希值出现次数,量化冲突次数)。

代码

cpp 复制代码
#include <stdio.h>
#include <stdbool.h>
#include <string.h>

#define MAX_SZ 10000

int T,n,cnt=0,a[1000];
int hash[MAX_SZ],primes[MAX_SZ];
bool st[MAX_SZ+1]={false};

void sieve(){
	for(int i=2;i<=MAX_SZ;i++){
		if(st[i]) continue;
		primes[cnt++]=i;
		for(int j=i*i;j<=MAX_SZ;j+=i){
			st[j]=true;
		}
	}
}

bool check(int n,int sz){
	memset(hash, 0, sizeof(int) * (sz+1));
	int i,k;
	for(i=0;i<n;i++){
		k=a[i]%sz;
		if(hash[k]++>1) return false;
	}
	return true;
}

int main() {
	int i;
	sieve();
	scanf("%d", &T);
	while (T--) {
		scanf("%d", &n);
		for (i = 0; i < n; i++) scanf("%d", &a[i]);
		for(i=0;primes[i]<(n>>1);i++);
		for(;!check(n,primes[i]);i++);
		printf("%d\n",primes[i]);
	}
	return 0;
}
相关推荐
2301_822703202 分钟前
Flutter 框架跨平台鸿蒙开发 - 创意声音合成器应用
算法·flutter·华为·harmonyos·鸿蒙
cmpxr_18 分钟前
【C】数组名、函数名的特殊
c语言·算法
KAU的云实验台28 分钟前
【算法精解】AIR期刊算法IAGWO:引入速度概念与逆多元二次权重,可应对高维/工程问题(附Matlab源码)
开发语言·算法·matlab
会编程的土豆35 分钟前
【数据结构与算法】再次全面了解LCS底层
开发语言·数据结构·c++·算法
大熊背1 小时前
如何利用Lv值实现三级降帧
算法·自动曝光·lv·isppipeline
大尚来也2 小时前
驾驭并发:.NET多线程编程的挑战与破局之道
java·前端·算法
向阳而生,一路生花2 小时前
深入浅出 JDK7 HashMap 源码分析
算法·哈希算法
君义_noip2 小时前
信息学奥赛一本通 4150:【GESP2509七级】⾦币收集 | 洛谷 P14078 [GESP202509 七级] 金币收集
c++·算法·gesp·信息学奥赛·csp-s
摸个小yu2 小时前
【力扣LeetCode热题h100】链表、二叉树
算法·leetcode·链表
汀、人工智能2 小时前
[特殊字符] 第93课:太平洋大西洋水流问题
数据结构·算法·数据库架构·图论·bfs·太平洋大西洋水流问题