ACM-ICPC 2016 大连赛区现场赛 K. Guess the number && HDU 5981(思维+DP)

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5981

题意:A在[L, R]之间随机选取一个数X,之后B来猜这个数,如果猜的数比X小,那么A就告诉B猜小了,如果猜的数大于X,那么以后A永远只会回答B是否猜对了,问在最坏的情况下B至少要猜多少次,并求出有多少种方案。

题解:参考自:https://blog.csdn.net/jaihk662/article/details/77435217

      补充关于猜测次数的理解:如果第一次猜测的数大于等于 p[R],在最坏情况下,该数必定是大于所猜测的数的。假设第一次猜测的数为 p[R] + 1,则最坏情况下剩余规模为 R - ( p[R] + 1 ),其他同理,故减去这部分的前缀和。

 

 1 #include <bits/stdc++.h>
 2 using namespace std;
 3 #define ll long long
 4 #define ull unsigned long long
 5 #define mst(a,b) memset((a),(b),sizeof(a))
 6 #define mp(a,b) make_pair(a,b)
 7 #define pi acos(-1)
 8 #define pii pair<int,int>
 9 #define pb push_back
10 const int INF = 0x3f3f3f3f;
11 const double eps = 1e-6;
12 const int MAXN = 5e6 + 10;
13 const int MAXM = 1e3 + 10;
14 const ll mod = 100000073;
15 
16 int p[MAXN],more[MAXN];
17 ll cnt[MAXN],sum[MAXN];
18 
19 int main() {
20 #ifdef local
21     freopen("data.txt", "r", stdin);
22 #endif
23     int pre = 1;
24     for(int i = 1; i < MAXN && pre < MAXN; i++) {
25         int mx = min(MAXN - 1, i * (i + 1) / 2);
26         for(int j = pre; j <= mx; j++) p[j] = i;
27         more[i] = i * (i + 1) / 2;
28         pre = mx + 1;
29     }
30     cnt[1] = 1, cnt[2] = 2;
31     sum[1] = 1, sum[2] = 3;
32     for(int i = 3; i < MAXN; i++) {
33         cnt[i] = (sum[more[p[i] - 1]] - sum[i - (p[i] + 1)] + mod) % mod;
34         sum[i] = (sum[i - 1] + cnt[i]) % mod;
35     }
36     int a,b;
37     while(~scanf("%d%d",&a,&b))
38         printf("%d %lld\n",p[b - a + 1], cnt[b - a + 1]);
39     return 0;
40 }

 

posted on 2018-10-09 17:25  scau_lok  阅读(303)  评论(0编辑  收藏  举报

导航