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 ", 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 ", 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 }
感觉遇到题,还是先想清楚,有什么性质,复杂度满足要求么,最后才是码代码!