2019杭电多校 第六场 6641(原1008) TDL(规律题)

2019杭电多校 第六场 6641(原1008) TDL【规律+gcd】

题目:http://acm.hdu.edu.cn/showproblem.php?pid=6641

大致题意:

给你两个正整数k和m,要求你找出符合 (f(n,m)−n)⊕n=k 这个条件式的 n,(⊕表示异或xor,f(n,m)表示求出比n大的第m小的互质数)。

思路:

首先,分析 f(n,m)-n 可以知道,我们求出n的第m小的互质数,然后又减去了n,题目中可以看到m的范围是1~100,所以说m最大不超过100,通过暴力代码可以发现这样一个规律:
在这里插入图片描述
在这里插入图片描述
用暴力代码随便跑几个数的最近的100个互质数,可以发现几乎都是在1000范围以内(甚至更小的范围)就可以找完这个数的100个互质数,那么我们可以知道 f(n,m)-n 的取值范围大概不超过 1000

接下来,从二进制的角度分析 (f(n,m)−n)⊕n=k

由题可知,1≤k≤1e18 ,假设k很大,取1e15,为了方便,下面我们令

 p = (f(n,m)−n)    即 p ⊕ n = k

因为p的值非常小,k值非常大,然而 p ⊕ n = k,从二进制异或的角度来看,存在 n ≈ k 。所以得出结论 n就在k的附近

所以我们用暴力代码,直接搜索 k±1000 范围内的数字是否符合条件,如果存在则输出 n,不存在则输出 -1 。

代码部分:

暴力代码:
#include<cstdio>
#include<cstdlib>
#include<algorithm>
#include<iostream>
#include<string.h>
using namespace std;

//这是一个很神奇的 gcd() 代码
int gcd(int a, int b) {
	while(b^=a^=b^=a%=b);
	return a;
}

int main() {
	int n;
	const int m = 100;
	while(~scanf("%d", &n)) {
		int count = 0;
		int num = 1;
		//count 是用来控制输出100个互质数的
		while(count != 100) {
		//补充一下概念:两个数的最大公约数为1,则为互质数
			if(gcd(n+num, n) == 1) {
				printf("%d ", n+num);
				count++;
			}
			num++;
		}
		cout << endl << endl;
	}
	return 0;
}
ac代码:
#include<cstdio>
#include<cstdlib>
#include<algorithm>
#include<iostream>
#include<string.h>
using namespace std;
#define ll long long
ll k, m;

ll gcd(ll a, ll b) {
	while(b^=a^=b^=a%=b);
	return a;
}

void solve() {
	ll n = k-1000;
	if(n <= 0)
		n = 1;
	
	ll result = -1;
	//p用来暴力寻找n的互质数
	//count表示第几小互质数
	while(n <= k+1000) {
		ll count = 0, p = 0;
		while(count != m) {
			p++;
			if(gcd(n + p, n) == 1)
				count++;
		}
		ll temp = p ^ n;
		if(temp == k) {
			result = n;
			break;
		}
		n++;
	}
	
	if(result != -1)
		printf("%lld\n", result);
	else
		printf("%lld\n", result);
	return;
}

int main() {
	int T;
	while(~scanf("%d",&T)) {
		while(T--) {
			scanf("%lld %lld",&k, &m);
			solve();
		}
	}
	return 0;
}

总结

当时做,几乎要放弃的一道题目,因为完全没有头绪,但就是总感觉(f(n,m)−n)是一个非常神奇非常关键的东西,看到那么多人过自己过不了好不甘心啊,就一直盯着看,幸亏最后做出来了哈哈哈,ac的那一刻的喜悦无法言说,爽!rua!!!冲冲冲。

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

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值