0%

思路讲解

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……)

思路讲解

一个图中,如果所有节点的入度和出度都正好是1,那么这个图必然是由若干个不相交的环组成的。

题目的排列给出保证了所有点的出度和入度都为1。

可以使用并查集维护。特别需要注意的就是siz==2的情况,虽然cnt不统计,但是后续计算当替罪羊的时候要统计。

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
// 只可能偶数个奇数环
if(cnt[1]==2){
// 如果两个奇数环,那么两个不匹配点一定从两个奇数环出
ans=binpow(2,cnt[0])*mul[1];
ans%=mod;
}else if(cnt[1]>=4){ // 奇数环数量大于4,必然无法组成合法对
cout<<0<<"\n";
return;
}else{
vis.assign(N+5,0);
FOR(i,1,N){
if(vis[fa.find(i)]) continue;
vis[fa.find(i)]=true;
ll siz=fa.size(i);
if(siz==2){ // 这里一定不要遗漏
ans+=binpow(2,cnt[0]);
ans%=mod;
}else{
// 选择一黑一白,图的剩余部分必然可以完美匹配(注意一黑一白不一定相邻)
ll t=siz*siz%mod*binpow(4,mod-2);t%=mod;
ans+=binpow(2,cnt[0]-1)*t;
ans%=mod;
}
}
}

AC代码

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

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

思路讲解

我们在想问题的时候,只要确定我们检查的数对其他数有没有增加作用,不用太想。

AC代码

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

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