码迷,mamicode.com
首页 > 其他好文 > 详细

某deed笔试题

时间:2016-10-16 00:54:10      阅读:171      评论:0      收藏:0      [点我收藏+]

标签:

1.  删除ra,输入s,然后从前往后扫,遇到直接删除,O(n),算水题吧。

2. 矩阵乘法,看完题,感觉这么简单,估计有什么套路,仔细再读一遍,发现真是水题,50*50*50=125000,在2s时限内完全可以,而且数据范围很小,最大也是125000,不需要long long,直接写。

3. 读完题,看数据范围,maxn = 6, 6!=720, 然后是运算符的方式有2^5 = 32,然后总的复杂度为720 * 32 * 5 = 115200,这在1s的时限内完全可以,全排列可以用next_pemutation,运算符可以用二进制枚举,代码如下,(最后一个测试数据wa,我没找出来,后来换递归的ac了)。(找到错误了,比如 n = 1, k = 1, a1 = 10; 这里初始值应该设置为INT_MAX,结果才正确)。

 1 /*
 2 ID: y1197771
 3 PROG: test
 4 LANG: C++
 5 */
 6 #include<bits/stdc++.h>
 7 #define pb push_back
 8 #define FOR(i, n) for (int i = 0; i < (int)n; ++i)
 9 #define dbg(x) cout << #x << " at line " << __LINE__ << " is: " << x << endl
10 typedef long long ll;
11 using namespace std;
12 typedef pair<int, int> pii;
13 const int maxn = 1e3 + 10;
14 int a[10];
15 int n, k;
16 void solve() {
17     cin >> n >> k;
18     for (int i = 0; i < n; i++) cin >> a[i];
19     sort(a, a + n);
20     int res = k;
21     do {
22         if(res == 0) break;
23         for (int i = 0; i < (1 << n - 1); i++) {
24             int s = a[0];
25             for (int j = 0; j < n - 1; j++) {
26                 if(i >> j & 1) {
27                     s += a[j + 1];
28                 } else {
29                     s *= a[j + 1];
30                 }
31             }
32             if(abs(s - k) < res) res = abs(s - k);
33         }
34     } while(next_permutation(a, a + n));
35     cout << res << endl;
36 }
37 int main() {
38     freopen("test.in", "r", stdin);
39     //freopen("test.out", "w", stdout);
40     solve();
41     return 0;
42 }

4. 先看懂题意,然后考虑怎么做,我没做出来!

分析:1. 考虑最后的结果不是1010就是0101,然后可以考虑从0000怎么转移过去,但是考虑n=1e5,数据范围发,状态数非常多,这种方法无法实现。

2. 考虑子问题性质,长问题是否可以转化为短问题。找一些短的小例子,划一下从0转移到结果的情况。然后就是动态规划dp,考虑长度dp[n],

3. dp[0] = 1, dp[1] = 1, dp[2] = 1, dp[3] = 5.0/3,dp[4] = 2.然后考虑5的时候怎么转移,枚举第一个涂的位置,下标从1-n,然后比如(10000,01000,00100,00010,00001),1代表涂黑,接下看怎么考虑,对于10000,10无法改变,只需计算000,右边的3个0的期望黑色的个数,而这个期望已经算出来,由于先是涂的第一个黑色,右边三个0算的时候还要乘一个转移概率1/5,然后2边期望加起来。这里为什么需要加起来,那就需要考虑期望的性质,这里是期望黑色球的个数,可以把10000分开2段,总的期望等于左端的期望加上右端的期望,就是总的期望。 最后,第一个涂的位置有5种,然后把所有的期望加起来即可。下面用公式推导一下。

dp[n] = ∑ni=1 (dp[i - 1 - 1] / n + dp[n - i - 1] / n + 1/n) = ∑ni=1 (dp[i - 1 - 1] / n + dp[n - i - 1] / n) + 1 = 2 / n * ∑n-2i=1 (dp[i]) + 1.

就是上面的公式,这个公式应该不难理解吧,接下来,就可以直接码代码了。

 1 /*
 2 ID: y1197771
 3 PROG: test
 4 LANG: C++
 5 */
 6 #include<bits/stdc++.h>
 7 #define pb push_back
 8 #define FOR(i, n) for (int i = 0; i < (int)n; ++i)
 9 #define dbg(x) cout << #x << " at line " << __LINE__ << " is: " << x << endl
10 typedef long long ll;
11 using namespace std;
12 typedef pair<int, int> pii;
13 const int maxn = 1e5 + 10;
14 int n;
15 double dp[maxn];
16 void solve() {
17     dp[0] = 0;
18     dp[1] = dp[2] = 1;
19     dp[3] = 5.0 / 3; dp[4] = 2;
20     cin >> n;
21     if(n < 5) {
22         printf("%.10f\n", dp[n]);
23         return;
24     }
25     double s = dp[1] + dp[2] + dp[3];
26     for (int i = 5; i <= n; i++) {
27         dp[i] = s * 2 / i + 1;
28         s += dp[i - 1];
29     }
30     printf("%.10f\n", dp[n]);
31 
32 }
33 int main() {
34     //freopen("test.in", "r", stdin);
35     //freopen("test.out", "w", stdout);
36     solve();
37     return 0;
38 }

感觉遇到题,还是先想清楚,有什么性质,复杂度满足要求么,最后才是码代码!

某deed笔试题

标签:

原文地址:http://www.cnblogs.com/y119777/p/5965524.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!