CF Round #524 Div.2 - 竞赛题解


『解析』

A-Petya and Origami

读懂题意就会做……根据题意可以求出3种"sheet"各自需要的数量,然后每一种的数量除以k向上取整后求和就是答案。

B-Margarite and the best present

简单的数学题,根据题意可以将数列分成两部分:

\[-1,-3,-5,-7,... (奇数且负数)\\
2,4,6,8,... (偶数且正数)
\]

hh……两个等差数列。稍微注意一下左右两端点的值然后等差数列求和就可以了

C-Masha and two friends

没看懂官方给的那个算法标签是什么……

竞赛题解 - CF Round #524 Div.2-LMLPHP

(@_@) 不管,反正我觉得我的算法没问题……(其实主要想讲这道题

首先进行的操作是将闭区间变成左开右闭区间和上开下闭区间,这样的效果就是这样:

这步操作主要是为了方便判断重叠。

然后把矩形存成了一个存储了上下左右边界(u,d,l,r)的结构体,然后根据它的左下角坐标和长宽就可以算出它里面的黑白格子分别有多少个。

for example……

接下来就套用漂浮法:先在第一层放上黑色的矩形(下面称矩形A)(因为题目中它是最后一个放上去的矩形),然后尝试漂浮白色的矩形(下面称矩形B)——如果与矩形A没有重叠,则全部漂浮上去;否则如果矩形B的左边与矩形A重叠,就把矩形B按矩形A的左边界分为左右两半,etc.

再举个例子:

OK~切下来的矩形都会“漂浮”上去,对这些矩形统计涂色后各颜色块的变化数量就可以了~

D-Olya and magical square

另外写了一篇博客\(QwQ\)


『源代码』

A-Petya and Origami

/*Lucky_Glass*/
#include<bits/stdc++.h>
using namespace std;
int main(){
int n,k;
scanf("%d%d",&n,&k);
double A=n*2,B=n*5,C=n*8;
printf("%lld\n",(long long)ceil(A/k)+(long long)ceil(B/k)+(long long)ceil(C/k));
return 0;
}

B-Margarite and the best present

/*Lucky_Glass*/
#include<bits/stdc++.h>
using namespace std;
int main(){
int T;scanf("%d",&T);
while(T--){
long long l,r;
scanf("%lld%lld",&l,&r);
long long fl,fr,gl,gr;
if(l%2) fl=l,gl=l+1;
else gl=l,fl=l+1;
if(r%2) fr=r,gr=r-1;
else gr=r,fr=r-1;
long long ans=0;
if(gl<=gr) ans+=(gl+gr)*((gr-gl)/2+1)/2;
if(fl<=fr) ans-=(fl+fr)*((fr-fl)/2+1)/2;
printf("%lld\n",ans);
}
return 0;
}

C-Masha and two friends

/*Lucky_Glass*/
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N=5;
struct SQUARE {
int l,r,u,d,col;
pair<ll,ll> GetArea() {
ll R=u-d,C=r-l,A,B;
if(C%2) {
if(R%2) A=R/2*C+C/2+1,B=R/2*C+C/2;
else A=R/2*C,B=R/2*C;
} else {
if(R%2) A=R/2*C+C/2,B=R/2*C+C/2;
else A=R/2*C,B=R/2*C;
}
if(d%2==l%2) return make_pair(A,B);
else return make_pair(B,A);
}
} squ[N+5];
int n;
ll wht,blk;
void Floatage(int dep,SQUARE now) {
if(now.l>=now.r || now.d>=now.u) return;
while(dep<=n && (now.l>=squ[dep].r || now.r<=squ[dep].l || now.u<=squ[dep].d || now.d>=squ[dep].u))
dep++;
if(dep>n) {
pair<ll,ll> _now=now.GetArea();
if(now.col) wht-=_now.first,blk+=_now.first;
else wht+=_now.second,blk-=_now.second;
return;
}
SQUARE las=squ[dep],nxt;
if(now.l<las.l) {
nxt=now;
nxt.r=las.l;
Floatage(dep+1,nxt);
now.l=las.l;
}
if(now.r>las.r) {
nxt=now;
nxt.l=las.r;
Floatage(dep+1,nxt);
now.r=las.r;
}
if(now.u>las.u) {
nxt=now;
nxt.d=las.u;
Floatage(dep+1,nxt);
now.u=las.u;
}
if(now.d<las.d) {
nxt=now;
nxt.u=las.d;
Floatage(dep+1,nxt);
now.d=las.d;
}
}
int main() {
int T;
scanf("%d",&T);
while(T--) {
SQUARE bak;
bak.l=bak.d=1;
n=2;
scanf("%d%d",&bak.u,&bak.r);
bak.u++;bak.r++;
pair<ll,ll> _bak=bak.GetArea();
wht=_bak.first,blk=_bak.second;
for(int i=0; i<2; i++) //left,down,right,up
scanf("%d%d%d%d",&squ[i].l,&squ[i].d,&squ[i].r,&squ[i].u),squ[i].col=i,
squ[i].r++,squ[i].u++;
for(int i=1; i>=0; i--)
Floatage(i+1,squ[i]);
printf("%lld %lld\n",wht,blk);
}
return 0;
}

\(\mathcal{THE\ END}\)

\(\mathfrak{Thanks\ for\ reading!}\)

有什么没看懂的可以在 \(lucky\[email protected]\) 里问(经常看邮箱一族)

05-02 11:34