A. MEXanized Array 大致题意构造一个数组,其长度为 ,最大值为 , 为 ,问这个数组的所有值的和最大是多少
思路简单题,在 或 的场景下无解(不可能构造一个 不可达的数组),然后就随便构造就行了,保证 之后,剩下所有值取最大就行
AC code1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 void solve () { int _; cin >> _; for (int ts = 0 ; ts < _; ++ts) { int n, k, x; cin >> n >> k >> x; if (k > x + 1 || n < k) { cout << -1 << endl; continue ; } int sum = 0 ; for (int i = 0 ; i < k; ++i) sum += i; for (int i = k; i < n; ++i) sum += (x == k ? x - 1 : x); cout << sum << endl; } }
CPP
B. Friendly Arrays 大致题意给出两个数组 ,允许选择任意次的 数组中的任意一个 ,然后让 ,问最终得到的数组 中所有的异或和最大和最小的可能
思路某个比特为是 的情况下,在奇数个值异或和的结果则也是 ,而偶数个则为 。而或运算可以让 数组的每一个值的某些个位都变成 。基于此,只需要关心 的长度即可,若 为奇数,那么选尽可能多的 使得每个位都尽可能是 ,反之则尽可能不选,这样才能达到最大,同理可以得到最小的方案
AC code1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 void solve () { int _; cin >> _; for (int ts = 0 ; ts < _; ++ts) { int n, m; cin >> n >> m; int sa = 0 , sb = 0 , tmp; for (int i = 0 ; i < n; ++i) { cin >> tmp; sa ^= tmp; } for (int i = 0 ; i < m; ++i) { cin >> tmp; sb |= tmp; } cout << (n % 2 ? sa : sa & ~sb) << ' ' << (n % 2 ? sa | sb : sa) << endl; } }
CPP
C. Colorful Table 大致题意有一个数组 ,长度为 ,然后有一个对应的矩阵 ,为 ,其每一个位置的值
问对于每个数字 ,在矩阵 ,中能够找到对应一个最小的矩形,此矩形包含了所有出现 的位置,求出这个矩形的大小
思路对于任意一个值,假定其第一次在 中出现的位置为 ,它第一次出现在 地点一定是 ,同时其最后一次在矩阵中的位置一定是 ,其中 是在数组 ,中出现的,最后一个比当前值更大的下标
根据上面的规律,可以求出实际上每个值的位置,一定可以包裹比他大的那个值对应的矩阵,所以只需要根据值的大小排序一下他们在数组中第一次出现的位置,和最后一次出现的位置,然后从大到小遍历,保证小的值的区间能够覆盖到大的值的区间即可
AC code1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 void solve () { int _; cin >> _; for (int ts = 0 ; ts < _; ++ts) { int n, k; cin >> n >> k; vector<bool > flag (k, false ) ; vector<pair<int , int >> data (k, {-1 , -1 }); for (int i = 0 ; i < n; ++i) { int tmp; cin >> tmp; data[tmp - 1 ].first == -1 ? data[tmp - 1 ].first = data[tmp - 1 ].second = i : data[tmp - 1 ].second = i; flag[tmp - 1 ] = true ; } for (int i = k - 2 ; i >= 0 ; --i) { data[i].first = !flag[i] ? data[i + 1 ].first : (data[i + 1 ].first != -1 ? min (data[i].first, data[i + 1 ].first) : data[i].first); data[i].second = !flag[i] ? data[i + 1 ].second : (data[i + 1 ].first != -1 ? max (data[i].second, data[i + 1 ].second) : data[i].second); } for (int i = 0 ; i < k; ++i) cout << (!flag[i] ? 0 : (data[i].second - data[i].first + 1 ) + (data[i].second - data[i].first + 1 )) << ' ' ; cout << endl; } }
CPP
D. Prefix Purchase 大致题意有一个初始数组,每一个值都是 ,每次你可以选择花费 元,使得这个数组前 个元素加一,最多只能花费 元,问能够得到最大字典序的数组是什么
思路首先需把 的值进行单调递增栈处理一下,毕竟价格相同或更低的同时 更大肯定有优势
回到题目中的字典序,意味着只有越前面的值越大即可,所以要尽可能满足最前面的值最大,所以直接把 丢给处理后的第一个值,看看最多第一个值可以到多少
处理完成第一个值后,那就意味着后面无论怎么贪心,第一个值一定要达到这个,否则肯定不如现在更好。另外,对于字典序而言,约前面的值价值越高,所以要尽可能让前面的值大,贪心一下即可
AC code1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 void solve () { int _; cin >> _; for (int ts = 0 ; ts < _; ++ts) { int n, k; cin >> n; vector<pair<int , int >> data; for (int i = 0 ; i < n; ++i) { int tmp; cin >> tmp; while (!data.empty () && data.back ().first >= tmp) data.pop_back (); data.emplace_back (tmp, i); } data.emplace_back (INT_MAX, n - 1 ); cin >> k; vector<int > ans (data.size()) ; ans[0 ] = k / data.front ().first; k %= data.front ().first; for (int i = 1 ; i < data.size (); ++i) { int diff = data[i].first - data[i - 1 ].first; if (k < diff) { break ; } ans[i] = min (ans[i - 1 ], k / diff); k -= diff * ans[i]; } int cur = 0 ; for (int i = 0 ; i < n; ++i) { while (data[cur].second < i) cur++; cout << ans[cur] << " \n" [i == n - 1 ]; } } }
CPP