A是补题所以有代码
A:
VP时读错题,以为只能走整数点,且只能上下左右走。
其实随意,有缝就行。
想一下就知道,左下角不能到右上角的条件必须是,圆连接了左上和右下部分,使得路被阻断。
然后并查集判联通搞搞就行
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
//typedef __int128 LL;
//typedef unsigned long long ull;
//#define F first
//#define S second
typedef long double ld;
typedef pair<int,int> pii;
typedef pair<ll,ll> pll;
typedef pair<ld,ld> pdd;
const ld PI=acos(-1);
const ld eps=1e-9;
//unordered_map<int,int>mp;
#define ls (o<<1)
#define rs (o<<1|1)
#define pb push_back
const int seed=131;
const int M = 1e4+7;
int fa[M];
int get(int x)
{
if(fa[x]==x)return x;
return fa[x]=get(fa[x]);
}
void merge(int x,int y)
{
int gx=get(x),gy=get(y);
if(gx!=gy)
fa[gx]=gy;
}
struct node
{
double x,y,r;
}C[M];
double dis(double x1,double y1,double x2,double y2)
{
return sqrt((x2-x1)*(x2-x1)+(y2-y1)*(y2-y1));
}
int main()
{
ios::sync_with_stdio(false);
cin.tie(0);
int n,m,k;
cin>>n>>m>>k;
for(int i=1;i<=k+5;i++)fa[i]=i;
for(int i=1;i<=k;i++)
cin>>C[i].x>>C[i].y>>C[i].r;
int x0=k+1,xn=k+2,y0=k+3,yn=k+4;
for(int i=1;i<=k;i++)
{
for(int j=1;j<=k;j++)
{
if(i==j)continue;
if(dis(C[i].x,C[i].y,C[j].x,C[j].y)<=C[i].r+C[j].r)
merge(i,j); //圆i,j联通
}
if(n-C[i].x<=C[i].r)merge(i,yn);//圆i与右边界联通
if(C[i].x<=C[i].r)merge(i,y0);//圆i与 左边界联通
if(m-C[i].y<=C[i].r)merge(i,xn);//圆i与上边界联通
if(C[i].y<=C[i].r)merge(i,x0);//圆i与下边界联通
}
merge(x0,yn);//矩形右边界和下边界联通
merge(y0,xn);//矩形左边界和上边界联通
if(get(x0)==get(y0))puts("N");//左边界或上边界能通过圆到达下边界或者右边界
else puts("S");
return 0;
}
D
直接树链剖分即可。
剖的时候不是按子树大小,而是子树深度。
G
先把数的积通过取对数转化成求和:log(a*b)=loga+logb;
然后建出二分图,左边100个点代表第几行,右边100点代表第几列,边权i,j为第i行j列的值,最多10000条边。
建个源点汇点。泡个最小费用流最大流即可。
J :
模拟即可。。
循环前判断下是否有人赢
打个map每次循环判断下。
L:
找出规律 DFS一下就行,
M:
二分答案。。check一下就行
还有2个签到
B:
判下第一个是否最大,不唯一也输出S
H:
a*b 然后乘百分之10-90向上取整 输出