【AtCoder】ARC101题解
C - Candles
题解
点燃的一定是连续的一段,枚举左端点即可
代码
#include <bits/stdc++.h>
#define enter putchar('\n')
#define space putchar(' ')
#define pii pair<int,int>
#define fi first
#define se second
#define mp make_pair
#define MAXN 1000005
#define mo 999999137
#define pb push_back
//#define ivorysi
using namespace std;
typedef long long int64;
typedef double db;
template<class T>
void read(T &res) {
res = 0;T f = 1;char c = getchar();
while(c < '0' || c > '9') {
if(c == '-') f = -1;
c = getchar();
}
while(c >= '0' && c <= '9') {
res = res * 10 + c - '0';
c = getchar();
}
res *= f;
}
template<class T>
void out(T x) {
if(x < 0) {x = -x;putchar('-');}
if(x >= 10) out(x / 10);
putchar('0' + x % 10);
}
int N,K;
int64 a[100005];
int main() {
#ifdef ivorysi
freopen("f1.in","r",stdin);
#endif
read(N);read(K);
for(int i = 1 ; i <= N ; ++i) {
read(a[i]);
}
int64 ans = 1e16;
for(int i = 1 ; i <= N - K + 1; ++i) {
int t = i + K - 1;
if(1LL * a[i] * a[t] > 0) {
ans = min(ans,max(abs(a[i]),abs(a[t])));
}
else ans = min(ans,min(-2 * a[i] + a[t],2 * a[t] - a[i]));
}
out(ans);enter;
return 0;
}
D - Median of Medians
二分一个值作为中位数的中位数,把大于这个数的设成1,小于等于这个数的设成0
然后我们就需要知道小于等于这个数做中位数的区间有多少个,用树状数组维护,和全部区间个数的一半比较一下即可
题解
#include <bits/stdc++.h>
#define enter putchar('\n')
#define space putchar(' ')
#define pii pair<int,int>
#define fi first
#define se second
#define mp make_pair
#define MAXN 1000005
#define mo 999999137
#define pb push_back
//#define ivorysi
using namespace std;
typedef long long int64;
typedef double db;
template<class T>
void read(T &res) {
res = 0;T f = 1;char c = getchar();
while(c < '0' || c > '9') {
if(c == '-') f = -1;
c = getchar();
}
while(c >= '0' && c <= '9') {
res = res * 10 + c - '0';
c = getchar();
}
res *= f;
}
template<class T>
void out(T x) {
if(x < 0) {x = -x;putchar('-');}
if(x >= 10) out(x / 10);
putchar('0' + x % 10);
}
int N;
int64 M;
int a[100005],num[100005],cnt;
int sum[100005][2],c[200005];
int tr[200005];
int lowbit(int x) {return x & -x;}
void Insert(int x,int v) {
while(x <= 2 * N + 1) {
tr[x] += v;
x += lowbit(x);
}
}
int Query(int x) {
int res = 0;
while(x > 0) {
res += tr[x];
x -= lowbit(x);
}
return res;
}
bool check(int x) {
for(int i = 1 ; i <= N ; ++i) {
sum[i][0] = sum[i - 1][0] + (a[i] <= x);
sum[i][1] = sum[i - 1][1] + (a[i] > x);
}
memset(tr,0,sizeof(tr));
for(int i = 1 ; i <= N ; ++i) Insert(sum[i][1] - sum[i][0] + N + 1,1);
int64 res = 0;
for(int i = 1 ; i <= N ; ++i) {
int t = sum[i - 1][1] - sum[i - 1][0] + N + 1;
res += Query(t - 1);
Insert(sum[i][1] - sum[i][0] + N + 1,-1);
}
return res >= M;
}
int main() {
#ifdef ivorysi
freopen("f1.in","r",stdin);
#endif
read(N);
for(int i = 1 ; i <= N ; ++i) {read(a[i]);num[i] = a[i];}
M = 1LL * N * (N + 1) / 2;
M = M / 2 + 1;
sort(num + 1,num + N + 1);
cnt = unique(num + 1,num + N + 1) - num - 1;
int L = 1,R = cnt;
while(L < R) {
int mid = (L + R) >> 1;
if(check(num[mid])) R = mid;
else L = mid + 1;
}
out(num[L]);enter;
return 0;
}
E - Ribbons on Tree
题解
我们用容斥来考虑,设\(g(i)\)为至少有i条边没有被覆盖的方案数,那么答案就是
\(\sum_{i = 1}^{n} (-1)^{i}g(i)\)考虑一边dp一边求g的系数
用dp[u][i][0/1]表示以u为根的子树里,i个点匹配了,系数为-1或1的方案数
转移的话用类似树背包的转移
如果不是根的话枚举父边是不是没有被覆盖
#include <bits/stdc++.h>
#define enter putchar('\n')
#define space putchar(' ')
#define pii pair<int,int>
#define fi first
#define se second
#define mp make_pair
#define MAXN 5005
#define mo 99994711
#define pb push_back
//#define ivorysi
using namespace std;
typedef long long int64;
typedef double db;
template<class T>
void read(T &res) {
res = 0;T f = 1;char c = getchar();
while(c < '0' || c > '9') {
if(c == '-') f = -1;
c = getchar();
}
while(c >= '0' && c <= '9') {
res = res * 10 + c - '0';
c = getchar();
}
res *= f;
}
template<class T>
void out(T x) {
if(x < 0) {x = -x;putchar('-');}
if(x >= 10) out(x / 10);
putchar('0' + x % 10);
}
const int MOD = 1000000007;
int N;
struct node {
int to,next;
}E[MAXN * 2];
int head[MAXN],sumE;
int dp[MAXN][MAXN][2],fac[MAXN],inv[MAXN],invfac[MAXN],p[MAXN],siz[MAXN],tmp[MAXN][2];
void add(int u,int v) {
E[++sumE].to = v;
E[sumE].next = head[u];
head[u] = sumE;
}
int mul(int a,int b) {
return 1LL * a * b % MOD;
}
int inc(int a,int b) {
return a + b >= MOD ? a + b - MOD : a + b;
}
void update(int &x,int y) {
x = inc(x,y);
}
int C(int n,int m) {
if(n < m) return 0;
return mul(fac[n],mul(invfac[m],invfac[n - m]));
}
void dfs(int u,int fa) {
dp[u][0][0] = 1;
siz[u] = 0;
for(int i = head[u] ; i ; i = E[i].next) {
int v = E[i].to;
if(v != fa) {
dfs(v,u);
for(int i = 0 ; i <= siz[u] + siz[v] ; ++i) tmp[i][0] = tmp[i][1] = 0;
for(int i = 0 ; i <= siz[u] ; ++i) {
for(int j = 0 ; j <= siz[v] ; ++j) {
for(int k = 0 ; k <= 1 ; ++k) {
for(int h = 0 ; h <= 1 ; ++h) {
update(tmp[i + j][k ^ h],mul(dp[u][i][k],dp[v][j][h]));
}
}
}
}
for(int i = 0 ; i <= siz[u] + siz[v] ; ++i) {
dp[u][i][0] = tmp[i][0];
dp[u][i][1] = tmp[i][1];
}
siz[u] += siz[v];
}
}
++siz[u];
if(fa) {
for(int i = siz[u] - 1; i >= 0 ; --i) {
for(int k = 0 ; k <= 1 ; ++k) {
update(dp[u][siz[u]][k],mul(dp[u][i][k ^ 1],p[siz[u] - i]));
}
}
}
}
void Solve() {
read(N);
int u,v;
for(int i = 1 ; i < N ; ++i) {
read(u);read(v);add(u,v);add(v,u);
}
inv[1] = 1;
for(int i = 2 ; i <= N ; ++i) inv[i] = mul(inv[MOD % i],MOD - MOD / i);
fac[0] = invfac[0] = 1;
for(int i = 1 ; i <= N ; ++i) fac[i] = mul(fac[i - 1],i),invfac[i] = mul(invfac[i - 1],inv[i]);
p[0] = 1;
for(int i = 1 ; i <= N ; ++i) {
if(i & 1) p[i] = 0;
else p[i] = mul(p[i - 2],i - 1);
}
dfs(1,0);
int ans = 0;
for(int i = 0 ; i <= N ; ++i) {
ans = inc(ans,mul(dp[1][i][0],p[N - i]));
ans = inc(ans,MOD - mul(dp[1][i][1],p[N - i]));
}
out(ans);enter;
}
int main() {
#ifdef ivorysi
freopen("f1.in","r",stdin);
#endif
Solve();
return 0;
}
F - Robots and Exits
题解
把每个机器人到左边最近的距离记作\(a_i\),到右边最近的距离记作\(b_i\)
我们把这个点\((a_i,b_i)\)画在平面直角坐标系里
然后我们记\((x,y)\)为最左的点远离初始点x的距离,到过最右的点远离初始点y的距离
每次可以走到\((x + 1,y)\)或\((x,y + 1)\)
在路径下方的点是从右边出口走的,上方的点是从左边出口走的
我们就是要看走的折线能把点分成几个集合,用树状数组维护来转移即可
#include <iostream>
#include <cstring>
#include <cstdio>
#include <algorithm>
#define enter putchar('\n')
#define space putchar(' ')
#define fi first
#define se second
#define mp make_pair
#define MAXN 100005
//#define ivorysi
#define pii pair<int,int>
using namespace std;
typedef long long int64;
template<class T>
void read(T &res) {
res = 0;char c = getchar();T f = 1;
while(c < '0' || c > '9') {
if(c == '-') f = -1;
c = getchar();
}
while(c >= '0' && c <= '9') {
res = res * 10 + c - '0';
c = getchar();
}
res *= f;
}
template<class T>
void out(T x) {
if(x < 0) {putchar('-');x = -x;}
if(x >= 10) out(x / 10);
putchar('0' + x % 10);
}
const int MOD = 1000000007;
int N,M;
int x[MAXN],y[MAXN],tot,v[MAXN],cnt,tr[MAXN],f[MAXN];
pii poi[MAXN];
int lowbit(int x) {
return x & -x;
}
int inc(int a,int b) {
return a + b >= MOD ? a + b - MOD : a + b;
}
int mul(int a,int b) {
return 1LL * a * b % MOD;
}
void Insert(int x,int v) {
while(x <= cnt) {
tr[x] = inc(tr[x],v);
x += lowbit(x);
}
}
int Query(int x) {
int res = 0;
while(x > 0) {
res = inc(res,tr[x]);
x -= lowbit(x);
}
return res;
}
void Solve() {
read(N);read(M);
for(int i = 1 ; i <= N ; ++i) read(x[i]);
for(int i = 1 ; i <= M ; ++i) read(y[i]);
int p = 1;
for(int i = 1 ; i <= N ; ++i) {
while(p + 1 <= M && y[p + 1] < x[i]) ++p;
if(p >= M) break;
if(x[i] > y[p]) poi[++tot] = mp(x[i] - y[p],y[p + 1] - x[i]);
}
sort(poi + 1,poi + tot + 1);
tot = unique(poi + 1,poi + tot + 1) - poi - 1;
for(int i = 1 ; i <= tot ; ++i) {
v[++cnt] = poi[i].se;
}
v[++cnt] = 0;
sort(v + 1,v + cnt + 1);
cnt = unique(v + 1,v + cnt + 1) - v - 1;
Insert(1,1);
int ans = 1;p = 0;
for(int i = 1 ; i <= tot ; ++i) {
while(p + 1 <= tot && poi[p + 1].fi < poi[i].fi) {
++p;
int t = lower_bound(v + 1,v + cnt + 1,poi[p].se) - v;
Insert(t,f[p]);
}
int t = lower_bound(v + 1,v + cnt + 1,poi[i].se) - v;
f[i] = Query(t - 1);
ans = inc(ans,f[i]);
}
out(ans);enter;
}
int main() {
#ifdef ivorysi
freopen("f1.in","r",stdin);
#endif
Solve();
}
【AtCoder】ARC101题解的更多相关文章
- AtCoder ExaWizards2019题解
AtCoder ExaWizards2019题解 AtCoder (因为代码直接用模板写的,可能有点冗长) A.Regular Triangle 给你三根棍子的长度,问你能否用他们组成等边三角形. 什 ...
- Atcoder ARC101 E 树dp
https://arc101.contest.atcoder.jp/tasks/arc101_c 题解是也是dp,好像是容斥做的,但是看不懂,而且也好像没讲怎么变n^2,看了写大佬的代码,自己理解了一 ...
- Atcoder ARC101 Ribbons on Tree
题解: 前面牛客网的那个比赛也有一道容斥+dp 两道感觉都挺不错的 比较容易想到的是 f[i][j]表示枚举到了i点,子树中有j个未匹配 这样的话我们需要枚举儿子中匹配状态 这样是n^2的(这是个经典 ...
- KYOCERA Programming Contest 2021(AtCoder Beginner Contest 200) 题解
KYOCERA Programming Contest 2021(AtCoder Beginner Contest 200) 题解 哦淦我已经菜到被ABC吊打了. A - Century 首先把当前年 ...
- AT2370 Piling Up
https://www.luogu.org/jump/atcoder/2370 题解 答案不是\(2^{2m}\)因为每轮的第一次取球可能会不够. 我们可以设\(dp[i][j]\)表示到了第\(i\ ...
- Triple Shift
来源:Atcoder ARC 136 B - Triple Shift (atcoder.jp) 题解:这道题我们不可能去硬模拟(大多数这种题都不能这样去模拟的),然后我们就要去发现特性, 发现把 a ...
- 重修 Slope Trick(看这篇绝对够!)
Slope Trick 算法存在十余载了,但是我没有找到多少拍手叫好的讲解 blog,所以凭借本人粗拙的理解来写这篇文章. 本文除标明外所有图片均为本人手绘(若丑见谅),画图真的不容易啊 qwq(无耻 ...
- AtCoder Regular Contest 094 (ARC094) CDE题解
原文链接http://www.cnblogs.com/zhouzhendong/p/8735114.html $AtCoder\ Regular\ Contest\ 094(ARC094)\ CDE$ ...
- AtCoder ExaWizards 2019 简要题解
AtCoder ExaWizards 2019 简要题解 Tags:题解 link:https://atcoder.jp/contests/exawizards2019 很水的一场ARC啊,随随便便就 ...
随机推荐
- 【bzoj2434】 Noi2011—阿狸的打字机
http://www.lydsy.com/JudgeOnline/problem.php?id=2434 (题目链接) 题意 给出一个字符串,$P$表示输出,$B$表示退格.$m$组询问$(x,y)$ ...
- 解题:HNOI 2015 开店
题面 根据树上距离的计算方法,可以先把答案化成$\sum dep_i+n*dep_u-\sum 2*dep[LCA(i,u)]$的形式,然后维护$\sum 2*dep[LCA(i,u)]$ 把妖怪们按 ...
- 如何构建 Redis 高可用架构?
温国兵 民工哥技术之路 今天 1 .题记 Redis 是一个开源的使用 ANSI C 语言编写.支持网络.可基于内存亦可持久化的日志型.Key-Value 数据库,并提供多种语言的 API. 如今,互 ...
- JavaScript中函数和类(以及this的使用<重点>,以及js和jquery讲解,原生js实现jquery)
1.javascript中以函数来表示类: 一般函数是小写开头:function foo() 类开头是大写:function Foo() 实例化类: obj = new Foo() 其他属性就同类是一 ...
- 常见HTTP状态码(200、301、302、500等)解说
对网站管理工作者来说有个词不陌生,HTTP状态码,它是用以表示网页服务器HTTP响应状态的3位数字代码.状态码的第一个数字代表了响应的五种状态之一. 1XX系列:指定客户端应相应的某些动作,代表请求已 ...
- Spark记录-Spark On YARN内存分配(转载)
Spark On YARN内存分配(转载) 说明 按照Spark应用程序中的driver分布方式不同,Spark on YARN有两种模式: yarn-client模式.yarn-cluster模式. ...
- POJ-3295 Tautology (构造)
https://vjudge.net/problem/POJ-3295 题意 有五种运算符和五个参数,现在给你一个不超过100字符的算式,问最后结果是否恒为1? 分析 首先明确各运算符的意义,K(&a ...
- HDU 1524 树上无环博弈 暴力SG
一个拓扑结构的图,给定n个棋的位置,每次可以沿边走,不能操作者输. 已经给出了拓扑图了,对于每个棋子找一遍SG最后SG和就行了. /** @Date : 2017-10-13 20:08:45 * @ ...
- .NET面试题系列(九)C# 结构体与类的区别
谈一下什么时候使用结构,什么使用类. 我们知道,结构存储在栈中,而栈有1个特点,就是空间较小,但是访问速度较快,堆空间较大,但是访问速度相对较慢.所以当我们描述1个轻量级对象的时候,可以将其定义为结构 ...
- html5 canvas 弧形描边渐变
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...