4647: 【GESP2409五级】挑战怪物
内存限制:128 MB
时间限制:1.000 S
评测方式:文本比较
命题人:
提交:1
解决:1
题目描述
⼩杨正在和一个怪物战斗,怪物的血量为$h$ ,只有当怪物的血量恰好为$0$ 时小杨才能够成功击败怪物。
小杨有两种攻击怪物的方式:
物理攻击。假设当前为小杨第$i$ 次使用物理攻击,则会对怪物造成$2^{i-1}$ 点伤害。
魔法攻击。小杨选择任意一个质数$x$ ( $x$不能超过怪物当前血量),对怪物造成$x$ 点伤害。由于小杨并不擅长魔法,他只能使用至多一次魔法攻击。
小杨想知道自己能否击败怪物,如果能,小杨想知道自己最少需要多少次攻击。
小杨有两种攻击怪物的方式:
物理攻击。假设当前为小杨第$i$ 次使用物理攻击,则会对怪物造成$2^{i-1}$ 点伤害。
魔法攻击。小杨选择任意一个质数$x$ ( $x$不能超过怪物当前血量),对怪物造成$x$ 点伤害。由于小杨并不擅长魔法,他只能使用至多一次魔法攻击。
小杨想知道自己能否击败怪物,如果能,小杨想知道自己最少需要多少次攻击。
输入
第一行包含一个正整数$t$ ,代表测试用例组数。
接下来是$t$ 组测试用例。对于每组测试用例,第一行包含一个正整数$h$ ,代表怪物血量。
接下来是$t$ 组测试用例。对于每组测试用例,第一行包含一个正整数$h$ ,代表怪物血量。
输出
对于每组测试用例,如果小杨能够击败怪物,输出一个整数,代表小杨需要的最少攻击次数,如果不能击败怪物,输出$-1$ 。
样例输入 复制
3
6
188
9999
样例输出 复制
2
4
-1
提示
#include <bits/stdc++.h>
using namespace std;
vector < int > prime;
bool is_prime[100010];
void Eratosthenes(int n) {
is_prime[0] = is_prime[1] = false;
for (int i = 2; i <= n; ++i) is_prime[i] = true;
for (int i = 2; i <= n; ++i) {
if (is_prime[i]) {
prime.push_back(i);
if ((long long) i * i > n) continue;
for (int j = i * i; j <= n; j += i)
is_prime[j] = false;
}
}
}
int main() {
Eratosthenes(100000);
int t;
cin >> t;
while (t--) {
int tmp = 1;
int x;
cin >> x;
int ans = 0;
while (1) {
if (is_prime[x]) {
ans++;
break;
}
x -= tmp;
ans++;
if (x <= 0) {
if (x < 0) ans = -1;
break;
}
tmp *= 2;
}
cout << ans << "\n";
}
}