這場的codeforces異常簡單
幾乎沒有什麼算法題//雖然codeforces一直如此,想法題多於算法題
A簽到題
B我的思路一開始很正確:
數連續出現的0的個數(夾在兩個1中間纔有效)
最後把這個連續0的個數都加1然後乘起來就是結果//就是高中數學中擋板問題
但是在做的時候出現了一點小紕漏
就是沒考慮只有夾在兩個1中間的連續的0是有效的
不然
00011000
剛開始000是無效的
最後的000也是無效的
特判一下就行了
#include <bits/stdc++.h>
using namespace std;
#define LL long long
//int cnt[110];
int ma[110],num[110];
int main(){
int n;
while(~scanf("%d",&n)){
int cnt=0,cs=0;
LL sum=1;
for(int i=0;i<n;i++)scanf("%d",&ma[i]);
for(int i=0;i<n;i++){
if(ma[i]==1){
cnt++;
if(cnt>1&&cs>0)sum*=(LL)(cs+1);
cs=0;
}
if(cnt>0){
if(ma[i]==0)cs++;
}
}
if(cnt)printf("%lld\n",sum);
else printf("0\n");
}
}
C題,暴力就行,題目給了2s,按複雜度確實是可以的
就是在算的時候有時要注意,r1可以爲0,r2也可以爲0,我在敲是剛開始沒注意到這點
#include <bits/stdc++.h>
#define FOR(i,n,m) for(long long i=n;i<m;i++)
#define FORR(i,n,m) for(long long i=n;i>=m;i--)
#define LL long long
#define inf 1e18
using namespace std;
const LL maxn = 2000 + 40 ;
struct po{
LL d;
LL num;
};
po d1[maxn],d2[maxn];
LL dis(LL x,LL y,LL x1,LL y1){
return (x-x1)*(x-x1)+(y-y1)*(y-y1);
}
bool cmp(po a,po b){
return a.d>b.d;
}
int main(){
LL n,x1,x2,y1,y2;
while(~scanf("%lld%lld%lld%lld%lld",&n,&x1,&y1,&x2,&y2)){
LL r1,r2;
FOR(i,0,n){
LL x,y;
scanf("%lld%lld",&x,&y);
d1[i].d=dis(x,y,x1,y1);d1[i].num=i;
d2[i].d=dis(x,y,x2,y2);d2[i].num=i;
}
d1[n].d=0;d1[n].num=n;
sort(d1,d1+n+1,cmp);//sort(d2,d2+n,cmp);
LL MAX=inf;
FOR(i,0,n+1){
r1=d1[i].d;
r2=0;
FORR(j,i-1,0){
int p=d1[j].num;
r2=max(d2[p].d,r2);
}
MAX=min(MAX,r1+r2);
}
printf("%lld\n",MAX);
}
}
D題,分類討論,在紙上畫一畫,思路很清晰,額,我分類討論的代碼有點挫
#include <bits/stdc++.h>
using namespace std;
int main(){
int x1,x2,x3,y1,y2,y3;
while(~scanf("%d%d%d%d%d%d",&x1,&y1,&x2,&y2,&x3,&y3)){
if(x1==x2&&x2==x3)printf("1\n");
else if(y1==y2&&y2==y3)printf("1\n");
else if(x1==x2||x2==x3||x1==x3){
if(x1==x2){
int MIN=min(y1,y2),MAX=max(y1,y2);
if(y3>MIN&&y3<MAX)printf("3\n");
else printf("2\n");
}
else if(x2==x3){
int MIN=min(y3,y2),MAX=max(y3,y2);
if(y1>MIN&&y1<MAX)printf("3\n");
else printf("2\n");
}
else if(x1==x3){
int MIN=min(y1,y3),MAX=max(y1,y3);
if(y2>MIN&&y2<MAX)printf("3\n");
else printf("2\n");
}
}
else if(y1==y2||y2==y3||y1==y3){
if(y1==y2){
int MIN=min(x1,x2),MAX=max(x1,x2);
if(x3>MIN&&x3<MAX)printf("3\n");
else printf("2\n");
}
else if(y2==y3){
int MIN=min(x3,x2),MAX=max(x3,x2);
if(x1>MIN&&x1<MAX)printf("3\n");
else printf("2\n");
}
else if(y1==y3){
int MIN=min(x1,x3),MAX=max(x1,x3);
if(x2>MIN&&x2<MAX)printf("3\n");
else printf("2\n");
}
}
else printf("3\n");
}
}
E題
用一個有點有意思的解法:
莫隊定理:
好像是有個學生髮明的,感覺也是蠻厲害的
#include<bits/stdc++.h>
using namespace std;
const int maxn = 120010;
int a[maxn],pos[maxn];
long long ans,flag[5000000];
long long Ans[maxn];
int k;
struct query
{
int l,r,id;
}Q[maxn];
bool cmp(query a,query b)
{
if(pos[a.l]==pos[b.l])
return a.r<b.r;
return pos[a.l]<pos[b.l];
}
void Updata(int x)
{
ans+=flag[a[x]^k];
flag[a[x]]++;
}
void Delete(int x)
{
flag[a[x]]--;
ans-=flag[a[x]^k];
}
int main()
{
int n,m;
scanf("%d%d%d",&n,&m,&k);
int sz =ceil(sqrt(1.0*n));
for(int i=1;i<=n;i++)
{
scanf("%d",&a[i]);
pos[i]=(i-1)/sz;
}
for(int i=1;i<=n;i++)
a[i]^=a[i-1];
for(int i=1;i<=m;i++)
{
scanf("%d%d",&Q[i].l,&Q[i].r);
Q[i].id = i;
}
sort(Q+1,Q+1+m,cmp);
int l = 1,r = 0;
ans=0;
flag[0]=1;
for(int i=1;i<=m;i++)
{
int id = Q[i].id;
while(r<Q[i].r)
{
r++;
Updata(r);
}
while(l>Q[i].l)
{
l--;
Updata(l-1);
}
while(r>Q[i].r)
{
Delete(r);
r--;
}
while(l<Q[i].l)
{
Delete(l-1);
l++;
}
Ans[id]=ans;
}
for(int i=1;i<=m;i++)
printf("%lld\n",Ans[i]);
}