DAY5共2題:
-
儲物點的距離(字首和)
-
糖糖別胡說,我真的不是簽到題目(multiset,思維)
? 作者:Eriktse
? 簡介:19歲,211計算機在讀,現役ACM銀牌選手?力爭以通俗易懂的方式講解演算法!❤️歡迎關注我,一起交流C++/Python演算法。(優質好文持續更新中……)?
? 原文連結(閱讀原文獲得更好閱讀體驗):https://www.eriktse.com/algorithm/1096.html
儲物點的距離
題目連結:https://ac.nowcoder.com/acm/problem/14683
預處理出各點搬運到點1
和點n
的代價字首和,以及區間重量和。
假如我們要將區間[5, 7]
的物品全部搬運到3
,代價應該是區間[5, 7]
的物品全部搬運到的1
,然後減去多搬的代價:[5, 7]
的重量和 * dist(1, 3)。
上面是x < l
的情況,x > r
的情況類似。
當l <= x <= r
只需將區間[l, r]
分為兩部分求和即可。
記得取模,距離要取模,字首和也要取模。
#include <bits/stdc++.h>
#define int long long
using namespace std;
const int maxn = 2e5 + 9, p = 1e9 + 7;
//sum_l[i]表示區間[1, i]的物品都運到點1的代價之和
//prefix_a[i]表示區間[1, i]的物品重量之和
//pos[i]是第i個點的位置,透過a[]作字首和得到
int a[maxn], pos[maxn], sum_l[maxn], sum_r[maxn], prefix_a[maxn];
int n, m;
//取模函式
int mo(int x){return (x % p + p) % p;}
int f(int x, int l, int r)
{
if(l > r)return 0;
int res = 0;
if(x <= l)
{
res = mo(sum_l[r] - sum_l[l - 1]);
res = mo(res - mo(pos[x] - pos[1]) * mo(prefix_a[r] - prefix_a[l - 1]) % p);
}
else if(x >= r)
{
res = mo(sum_r[r] - sum_r[l - 1]);
res = mo(res - mo(pos[n] - pos[x]) * mo(prefix_a[r] - prefix_a[l - 1]) % p);
}
return res;
}
signed main()
{
scanf("%lld %lld",&n, &m);
pos[1] = 1;
for(int i = 2, d;i <= n; ++ i)scanf("%lld", &d), pos[i] = pos[i - 1] + d;
for(int i = 1;i <= n; ++ i)scanf("%lld", a + i);
for(int i = 1;i <= n; ++ i)
{
sum_l[i] = mo(sum_l[i - 1] + a[i] * mo(pos[i] - pos[1]) % p);
sum_r[i] = mo(sum_r[i - 1] + a[i] * mo(pos[n] - pos[i]) % p);
}
for(int i = 1;i <= n; ++ i)prefix_a[i] = mo(prefix_a[i - 1] + a[i]);
while(m --)
{
int x, l, r;scanf("%lld %lld %lld", &x, &l, &r);
int ans = 0;
if(l <= x and x <= r)ans = mo(f(x, l, x - 1) + f(x, x + 1, r));
else ans = f(x, l, r);
printf("%lld\n", ans);
}
return 0;
}
糖糖別胡說,我真的不是簽到題目
題目連結:https://ac.nowcoder.com/acm/problem/14583
本題有兩種解法,第一種容易理解,第二種效率更優。
第一種解法:正向,multiset。
發功次數可以用一個桶來記錄,讓[1, i]
的所有點的屬性值都+1
,相當於把後面的都-1
,用一個fix
表示偏移量。
建立兩個multiset
表示兩組中的糖糖,好處是可以快速找出能力值最小的,從而去除掉。
掃一遍,把第i只糖糖加入到屬於它的集合中,然後將另外一個集合中已有的能力值小的糖糖進行刪除,再檢查此時是否有發功。
最後集合中留下的糖糖個數即為答案。
#include <bits/stdc++.h>
#define int long long
using namespace std;
const int maxn = 1e6 + 9, inf = 8e18;
struct Node
{
int a, b;
}p[maxn];
int add[maxn];
void solve()
{
int n, m;scanf("%lld %lld",&n, &m);
memset(add, 0, sizeof(int) * (n + 2));
for(int i = 1;i <= n; ++ i)
scanf("%lld %lld", &p[i].a, &p[i].b);
//注意同一時間可能施法多次
for(int i = 1, x;i <= m; ++ i)scanf("%lld", &x), add[x] ++;
int fix = 0, cnt = 0;
multiset<int> st[2];
for(int i = 1;i <= n; ++ i)
{
int a = p[i].a, b = p[i].b - fix;
st[a].insert(b);
while(!st[a ^ 1].empty() and *st[a ^ 1].begin() < b)
st[a ^ 1].erase(st[a ^ 1].begin()), cnt ++;
fix += add[i];
}
printf("%lld\n", n - cnt);
}
signed main()
{
int _;scanf("%lld", &_);
while(_ --)solve();
return 0;
}
第二種解法:反向,思維。
我們想這麼一個問題,一個糖糖x
是否會被刪除取決於在x
出現之後,在另外一個集合中,是否出現了能力值高於x
的能力值的糖糖y
。
那麼我們逆向遍歷,維護兩個集合的最值,當前的新加入的糖糖x
的能力值如果低於另外一個集合中已經存在的(即右邊的)糖糖能力值的最大值,說明他後面會被某個糖糖y
刪除掉,直接打上標記,但是我們不用真的刪除。
糖糖x
還可以用於刪除左邊的另一個集合的能力值較小的糖糖。注意此時的發功應該在迴圈開始時進行修改。
#include <bits/stdc++.h>
#define int long long
using namespace std;
const int maxn = 1e6 + 9, inf = 8e18;
struct Node
{
int a, b;
}p[maxn];
int add[maxn];
void solve()
{
int n, m;scanf("%lld %lld",&n, &m);
memset(add, 0, sizeof(int) * (n + 2));
for(int i = 1;i <= n; ++ i)
scanf("%lld %lld", &p[i].a, &p[i].b);
//注意同一時間可能施法多次
for(int i = 1, x;i <= m; ++ i)scanf("%lld", &x), add[x] ++;
int fix = 0, cnt = 0;
int mx[2] = {-inf, -inf};
for(int i = n;i >= 1; -- i)
{
fix += add[i];
int a = p[i].a, b = p[i].b + fix;
mx[a] = max(mx[a], b);
if(mx[a ^ 1] > b)cnt ++;
}
printf("%lld\n", n - cnt);
}
signed main()
{
int _;scanf("%lld", &_);
while(_ --)solve();
return 0;
}
? 本文由eriktse原創,創作不易,如果對您有幫助,歡迎小夥伴們點贊?、收藏⭐、留言?