Problem - E2 - Codeforces
题目大意:在无向图中,初始有一个点,然后将k个点连接到1号点上,之后每次操作分别将k歌点连接到之前新加的点上,这样的操作至少有1次,t次询问,每次询问给出一个数n,问n能否是某个符合上述条件的图的总点数
k>1;1<=t<=1e4;1<=n<=1e18
思路:我们令每张合法图的总点数为s,操作的次数(包括第一次连接k歌点到1号点)为i。因为s=1+k+k*k+k*k*k+...+k的i次方,可以看出这是一个等比数列,利用前n项和公式,得出 ,因为s最大是1e18,所以i理论最大值是62左右(赛后实际测出的最大值是58),k的最大值可以认为约等于1e9(因为优化效率极低,所有没有测k的最大值),这样我们可以枚举i,然后二分k,因为很明显i相同时,k越大,s越大,反之亦然,用s作为约束条件,这样的时间复杂度就是O(t*i*log1e9*log2i),2s可以稳过。但要注意在计算k的i+1次方时会爆__int128,需要用unsigned __int128
//#include<__msvc_all_public_headers.hpp>
#include<bits/stdc++.h>
using namespace std;
const int N = 2e5 + 5;
typedef unsigned __int128 ll;
ll qpow(ll a, ll b)
{//快速幂
	ll ret = 1;
	while (b)
	{
		if (b & 1)
		{
			ret = ret * a ;
		}
		b >>= 1;
		a = a * a ;
	}
	return ret;
}
int main()
{
	ios::sync_with_stdio(false);
	cin.tie(0);
	cout.tie(0);
	int t;
	cin >> t;
	while (t--)
	{
		long long  n;
		cin >> n;
		int ans = 0;
		for (ll i = 2; i <= 58; i++)
		{//枚举操作次数
			ll l = 2, r = 1000000000;
			ll mid;
			while (l <= r)
			{//二分k
				mid = (l + r) >> 1;
				ll s = (qpow(mid, i + 1) - 1) / (mid - 1);//直接求等差数列前i项和
				if (s > n)
				{
					r = mid - 1;
				}
				if (s < n)
					l = mid + 1;
				if (s == n)
				{//找到n直接退出
					ans = 1;
					break;
				}
			}
			if (ans)
				break;
		}
		cout << (ans ? "YES" : "NO") << endl;
	}
	return 0;
}


















