0%

思路讲解

三个集合的容斥原理

$\left | A \cup B \cup C \right | = \left | A \right | + \left | B \right | + \left | C \right | - \left | A \cap B \right | - \left | A \cap C \right | - \left | B \cap C \right | + \left | A \cap B \cap C \right | $

image

那么不难发现规律,就是奇加偶减。这就是这段代码的由来。

1
2
3
ll cn=bi.count();
if(cn%2==1) del+=(sz+1)*sz/2;
else del-=(sz+1)*(sz)/2;

那么直接计算合法路径数量比较难,但是我们可以枚举不合法子集。如果s是101(二进制),那么就是第一个素因子要不符合要求(LCM只关注这些数当中的最大的,你可以理解为对素因子取了一个并集,那么gcd就是取了一个交集)。

1
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
34
35
36
37
38
39
40
41
42
43
44
45
46
// s如果这一位是1,就代表这一位的质因数必须不满足要求
FOR(s,0,(1<<siz)-1){
vb vis(N+5,0);
bitset<11> bi(s);
// 检查该数字(该点)是否满足要求
auto check=[&](ll u)->bool{
FOR(i,0,siz-1){
if(!bi[i]) continue;
// 该数不满足要求,因为我们要求这一位质因数数量缺失,或者太多
if(fach[u][i]==cnt[i]){
return false;
}
}
return true;
};
FOR(u,1,N){
if(vis[u]) continue;
if(!valid[u]) continue;
vis[u]=true;
if(!check(u)) continue;
queue<ll> q;q.push(u);
ll sz=0; // 维护联通块内的点的数量
while(!q.empty()){
ll u=q.front();q.pop();
vis[u]=true;
++sz;
for(auto v:g[u]){
if(vis[v]) continue;
if(!valid[v]) continue;
if(!check(v)) continue;
q.push(v);
}
}
// sz*(sz-1)/2+sz
ll cn=bi.count();
if(cn%2==1) del+=(sz+1)*sz/2;
else del-=(sz+1)*(sz)/2;
// ans+=(sz+1)*sz/2;
}
#ifdef LOCAL
debug(del);
#endif
}
ll ans=-del;
cout<<ans<<"\n";
}

AC代码

https://acm.hdu.edu.cn/contest/view-code?cid=1172&rid=22776

心路历程(WA,TLE,MLE……)

思路讲解

那么容易想到,按照x排序,可以让sumx最大的方式就是排序,然后左边匹配右边。

而且,这样子匹配,左边的谁匹配右边的谁也是不重要的,至少对于sumx不重要,因此我们就可以按照使sumy最大的方式进行匹配。

同理,计算按照y排列能得到的答案即可。

AC代码

https://codeforces.com/contest/2122/submission/329811774

心路历程(WA,TLE,MLE……)

思路讲解

114x1x2+5141y1y2+919810axay114∣x_1−x_2∣+5141∣y_1−y_2∣+919810|a_x - a_y|

这个式子看着很唬人,其实还可以,不难发现

114100+5141100919810=394310114*100+5141*100-919810= -394310

所以说就不用考虑这个x,也不用考虑这个y,只需要考虑高度(即a)就可以了。

局部最高点也只需要这段代码(把旁边的四个格子扫一圈就行)。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
FOR(i,1,N){
FOR(j,1,M){
// 已经布设了传送门
if(i==1 && j==1) continue;
bool isHi=true;
FOR(k,0,3){
ll x=i+dx[k],y=j+dy[k];
if(x<1 || x>N) continue;
if(y<1 || y>M) continue;
if(A[x][y]>A[i][j]){
isHi=false;
break;
}
}
if(isHi){
ans+=cs;
wp.EB(i,j);
}
}
}

AC代码

https://acm.hdu.edu.cn/contest/view-code?cid=1172&rid=19896

心路历程(WA,TLE,MLE……)

思路讲解

START195B——Game (Hard)(分治搜索) 打codechef还是有用的,或者说,非常有用。

先对询问进行排序,去重。

1
2
vll uq(all(que));
sort(all(uq));uq.resize(unique(all(uq))-uq.begin());

然后后面就是官解的思路,直接枚举边,然后可以得到一个sum→w(即sumW map)

1
2
3
4
5
6
7
8
9
10
11
map<ll,ll> sumW;
FOR(i,1,N){
for(auto &[v,t,w]:g[i]){
ll sum=revd[v]+dist[i]+t;
if(sumW.count(sum)){
sumW[sum]=max(sumW[sum],w);
}else{
sumW[sum]=w;
}
}
}

接着对这个sumW map进行一些处理,即使得sum越大,w也越大(sum大,w小的直接抛弃),接着装入sums数组(vector)中,方便我们二分查找。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
for(auto it=sumW.begin();it!=sumW.end();){
ll sum=it->first,w=it->second;
if(w < mxw){ // 你的sum比别人大,你的w还比别人小,你存在的意义是什么?
it++;
sumW.erase(prev(it)); // 因此删除
}else{
mxw=w; // 更新mxw
it++;
}
}
vector<pll> sums;
for(auto &[sum,w]:sumW){
sums.emplace_back(sum,w);
}

接着利用该单调性,递归二分搜索。(这可能也是官解的思路,二分可行域)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
vll cun(Q+5,0);
function<void(ll,ll,ll,ll)> dfs=[&](ll ia,ll ib,ll kc,ll kd)->void{
// kc,kd指的是询问的时候升级次数的编号,或者简单一点,就是uq的下标
if(kc>kd) return;
ll k=uq[kc+kd>>1];
ll km=kc+kd>>1;
ll ans=INF;
ll mid=0;
// 在可行域中暴力计算答案
FOR(i,ia,ib){ // ia ib 指的是sums的下标
ll sum=sums[i].first,w=sums[i].second;
ll lans=sum-w*k;
if(lans<=ans){
mid=i;
ans=lans;
}
}

cun[km]=ans;
dfs(mid,ib,km+1,kd);
dfs(ia,mid,kc,km-1);
};

AC代码

https://ac.nowcoder.com/acm/contest/view-submission?submissionId=78313063

心路历程(WA,TLE,MLE……)

vll cun(Q+5,0);

别写成N+5了。

思路讲解

核心代码就这

1
2
3
4
5
6
7
8
9
10
11
12
	ll l=pos[0],r=pos[1];
if(l>r) swap(l,r);
ll ans=r-l+1;
ll cnt=1;
FOR(i,2,SZ(pos)-1){ // 一个点,要么在这个区间内,要么在这个区间外
ll p=pos[i]; // 在这个区间内,必定不更新这个这个区间
l=min(l,p),r=max(r,p); // 在这个区间外,必定更新这个区间
ans=max(ans,r-l+1-cnt); // 所以说一个数更新区间的话,前面的数一定都在这个区间内
++cnt;
}
cout<<ans<<"\n";
}

我们再来看看树状数组怎么做。

AC代码

https://acm.hdu.edu.cn/contest/view-code?cid=1172&rid=16219

心路历程(WA,TLE,MLE……)