【AtCoder】AGC029(A-E)
A - Irreversible operation
题解
把每个B后面的W个数累加起来即可
代码
#include <bits/stdc++.h>
#define fi first
#define se second
#define pii pair<int,int>
#define pdi pair<db,int>
#define mp make_pair
#define pb push_back
#define enter putchar('\n')
#define space putchar(' ')
#define eps 1e-8
#define mo 974711
#define MAXN 1000005
//#define ivorysi
using namespace std;
typedef long long int64;
typedef double db;
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) {x = -x;putchar('-');}
if(x >= 10) {
out(x / 10);
}
putchar('0' + x % 10);
}
char s[200005];
int64 ans = 0;
int main() {
#ifdef ivorysi
freopen("f1.in","r",stdin);
#endif
scanf("%s",s + 1);
int64 t = 0;
int N = strlen(s + 1);
for(int i = N ; i >= 1 ; --i) {
if(s[i] == 'W') ++t;
else ans += t;
}
out(ans);enter;
}
B - Powers of two
题解
这比赛打的太难受了
这个我花了一个树结构,蒙了一个dp,然后wa了,去硬上T2
最后发现我把一个N改成离散化后的tot就A了
就是考虑一个点,和它组成恰好大于它的\(2^{i}\)的值只有一个,构成了一片森林,对于每个森林从底向上更新,如果父亲和儿子能配对就尽可能多的配对
每个\(2^{i}\)的点若有剩余再两两配对
代码
#include <bits/stdc++.h>
#define fi first
#define se second
#define pii pair<int,int>
#define pdi pair<db,int>
#define mp make_pair
#define pb push_back
#define enter putchar('\n')
#define space putchar(' ')
#define eps 1e-8
#define mo 974711
#define MAXN 200005
//#define ivorysi
using namespace std;
typedef long long int64;
typedef double db;
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) {x = -x;putchar('-');}
if(x >= 10) {
out(x / 10);
}
putchar('0' + x % 10);
}
struct node {
int to,next;
}E[MAXN * 10];
map<int,int> mm;
int val[MAXN],A[MAXN],N,tot,ans;
int head[MAXN],sumE,dp[MAXN];
bool vis[MAXN];
int lowbit(int x) {
return x & (-x);
}
void add(int u,int v) {
E[++sumE].to = v;
E[sumE].next = head[u];
head[u] = sumE;
}
void Init() {
read(N);
for(int i = 1 ; i <= N ; ++i) {
read(A[i]);
val[i] = A[i];
mm[A[i]]++;
}
sort(val + 1,val + N + 1);
tot = unique(val + 1,val + N + 1) - val - 1;
for(int i = 1 ; i <= N ; ++i) {
int a = val[i];
int k = lower_bound(val + 1,val + tot + 1,a) - val;
for(int64 t = 1 ; t <= 1073741824 ; t <<= 1) {
if(t > 2 * a) {
if(mm[t - a] == 0) continue;
int h = lower_bound(val + 1,val + tot + 1,t - a) - val;
add(k,h);
}
}
}
}
void dfs(int u) {
vis[u] = 1;
dp[u] = mm[val[u]];
for(int i = head[u] ; i ; i = E[i].next) {
int v = E[i].to;
if(!vis[v]) {
dfs(v);
int t = min(dp[u],dp[v]);
dp[u] -= t;
dp[v] -= t;
ans += t;
}
}
}
void Solve() {
for(int i = 1 ; i <= tot ; ++i) {
if(!vis[i]) dfs(i);
if(lowbit(val[i]) == val[i]) ans += dp[i] / 2;
}
out(ans);enter;
}
int main() {
#ifdef ivorysi
freopen("f1.in","r",stdin);
#endif
Init();
Solve();
}
C - Lexicographic constraints
题解
好不容易想到了一个分段维护一个字符个数进制下的数
没想到二分,gg了
代码
#include <bits/stdc++.h>
#define fi first
#define se second
#define pii pair<int,int>
#define pdi pair<db,int>
#define mp make_pair
#define pb push_back
#define enter putchar('\n')
#define space putchar(' ')
#define eps 1e-8
#define mo 974711
#define MAXN 200005
//#define ivorysi
using namespace std;
typedef long long int64;
typedef double db;
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) {x = -x;putchar('-');}
if(x >= 10) {
out(x / 10);
}
putchar('0' + x % 10);
}
int N;
int A[MAXN];
int pw[MAXN][25],cnt[MAXN],ans;
vector<pii > v;
void Init() {
read(N);
for(int i = 1 ; i <= N ; ++i) read(A[i]);
cnt[1] = 1000000000;
for(int i = 1 ; i <= N ; ++i) {
pw[i][0] = 1;
for(int j = 1 ; j <= 20 ; ++j) {
if(pw[i][j - 1] > N / i) {
cnt[i] = j - 1;break;
}
pw[i][j] = pw[i][j - 1] * i;
}
}
}
bool check(int mid) {
v.clear();
v.pb(mp(A[1],0));int sum = A[1];
for(int i = 2 ; i <= N ; ++i) {
if(A[i] <= A[i - 1]) {
if(mid == 1) return false;
if(sum < A[i - 1]) {v.pb(mp(A[i - 1] - sum,0));sum = A[i - 1];}
int s = v.size() - 1;
for(int k = s ; k >= 0 ; --k) {
if(sum - v[k].fi >= A[i]) {sum -= v[k].fi;v.pop_back();}
else {
int t = sum - A[i];
if(t > cnt[mid]) v[k].se = 0;
else v[k].se /= pw[mid][t];
v[k].fi -= t;
sum = A[i];
}
if(sum == A[i]) break;
}
s = v.size() - 1;
for(int k = s ; k >= 0 ; --k) {
if(v[k].fi <= cnt[mid]) {
if(v[k].se + 1 < pw[mid][v[k].fi]) {++v[k].se;break;}
else {
v[k].se = 0;if(k == 0) return false;
}
}
else {++v[k].se;break;}
}
}
}
return true;
}
void Solve() {
int l = 1,r = N;
while(l < r) {
int mid = (l + r) >> 1;
if(check(mid)) r = mid;
else l = mid + 1;
}
out(l);enter;
}
int main() {
#ifdef ivorysi
freopen("f1.in","r",stdin);
#endif
Init();
Solve();
}
D - Grid game
题解
这题好水啊= =开D我就win了啊,sdC一直不过太难受了啊
就是考虑我们肯定是走到某个障碍物上面,游戏结束了
如何判断能不能走到呢,首先如果这一列是\(j\),我们所在的行\(i\)必须大于等于\(j\),如果有一个连成一串的障碍物从\((j,j)\)开始往下压
每次最多压了多少,就是当前列最靠下不能走的点\(h\),求一个\(t = h - i + 1\)
\(t\)每次取一个前缀\(max\),然后对于每一列把行大于等于\(j + t\)的障碍物都设成不合法即可
把每一列能走到最靠上的障碍物取一个\(min\)
代码
#include <bits/stdc++.h>
#define fi first
#define se second
#define pii pair<int,int>
#define pdi pair<db,int>
#define mp make_pair
#define pb push_back
#define enter putchar('\n')
#define space putchar(' ')
#define eps 1e-8
#define mo 974711
#define MAXN 200005
//#define ivorysi
using namespace std;
typedef long long int64;
typedef double db;
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) {x = -x;putchar('-');}
if(x >= 10) {
out(x / 10);
}
putchar('0' + x % 10);
}
int H,W,N;
vector<int> v[MAXN];
bool pass[MAXN];
void Solve() {
read(H);read(W);read(N);
int x,y;
for(int i = 1 ; i <= N ; ++i) {
read(x);read(y);
v[y].pb(x);
}
for(int i = 1 ; i <= W ; ++i) v[i].pb(H + 1);
sort(v[1].begin(),v[1].end());
int ans = v[1][0] - 1;
int t = 0;
for(int i = 2 ; i <= W ; ++i) {
sort(v[i].begin(),v[i].end());
int s = v[i].size();
int m = 0;
for(int k = 0 ; k < s ; ++k) {
if(v[i][k] <= i + t) {pass[v[i][k]] = 1;m = max(v[i][k],m);}
else if(pass[v[i][k] - 1]) {pass[v[i][k]] = 1;m = max(v[i][k],m);}
else {
ans = min(ans,v[i][k] - 1);
break;
}
}
t = max(t,m - i + 1);
}
out(ans);enter;
}
int main() {
#ifdef ivorysi
freopen("f1.in","r",stdin);
#endif
Solve();
}
E - Wandering TKHS
题解
关注一下一个路径的根缀最大值
把根缀最大值相同的部分挑出来
我们发现一个根缀最大值是\(u\),\(u\)的某个儿子是\(v\),\(v\)中的所有节点要走到\(u\),就要经过\(v\)中所有根缀最大值为\(u\)的部分,这个可以差分实现
而根缀最大值自身讨论起来有点麻烦,直接搜索即可
代码
#include <bits/stdc++.h>
#define fi first
#define se second
#define pii pair<int,int>
#define pdi pair<db,int>
#define mp make_pair
#define pb push_back
#define enter putchar('\n')
#define space putchar(' ')
#define eps 1e-8
#define mo 974711
#define MAXN 200005
//#define ivorysi
using namespace std;
typedef long long int64;
typedef double db;
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) {x = -x;putchar('-');}
if(x >= 10) {
out(x / 10);
}
putchar('0' + x % 10);
}
int N;
struct node {
int to,next;
}E[MAXN * 2];
int head[MAXN],sumE;
int mv[MAXN],siz[MAXN],d[MAXN],c[MAXN],son[MAXN];
void add(int u,int v) {
E[++sumE].next = head[u];
E[sumE].to = v;
head[u] = sumE;
}
void calc(int u,int fa,int val) {
son[u] = 1;
for(int i = head[u] ; i ; i = E[i].next) {
int v = E[i].to;
if(v != fa && v < val) {
calc(v,u,val);
son[u] += son[v];
}
}
}
void dfs(int u,int fa) {
mv[u] = max(mv[fa],u);
siz[u] = 1;
for(int i = head[u] ; i ; i = E[i].next) {
int v = E[i].to;
if(v != fa) {
dfs(v,u);
siz[u] += siz[v];
d[u] += d[v];
}
}
if(mv[u] == u) {
calc(u,fa,mv[fa]);
d[u] = -siz[u];
}
if(mv[fa] == fa) d[u] += siz[u];
}
void dfs2(int u,int fa) {
if(fa) {
if(mv[u] == u) {c[u] += son[u];}
else if(mv[fa] == fa) {c[u] -= son[u];c[u] += d[u];}
c[u] += c[fa];
}
for(int i = head[u] ; i ; i = E[i].next) {
int v = E[i].to;
if(v != fa) {
dfs2(v,u);
}
}
}
void Init() {
read(N);
int x,y;
for(int i = 1 ; i < N ; ++i) {
read(x);read(y);add(x,y);add(y,x);
}
}
void Solve() {
dfs(1,0);
dfs2(1,0);
for(int i = 2 ; i <= N ; ++i) {
out(c[i]);
i == N ? enter : space;
}
}
int main() {
#ifdef ivorysi
freopen("f1.in","r",stdin);
#endif
Init();
Solve();
}
F - Construction of a tree
\(2 \cdot 10^{5}\)的网络流,有毒,不会卡常,弃疗了,再见
【AtCoder】AGC029(A-E)的更多相关文章
- 【AtCoder】ARC097 (C - F)题解
C - K-th Substring 题解 找出第K大的子串,重复的不计入 这个数据范围可能有什么暴力可以艹过去吧,但是K放大的话这就是后缀自动机板子题啊= = 代码 #include <ios ...
- 【BZOJ2599】Race(点分治)
[BZOJ2599]Race(点分治) 题面 BZOJ权限题,洛谷 题解 好久没写过点分治了... 在ppl的帮助下终于想起来了 orz ppl 首先回忆一下怎么求有没有正好是\(K\)的路径 维护一 ...
- 【HDU4622】Reincarnation(后缀自动机)
[HDU4622]Reincarnation(后缀自动机) 题面 Vjudge 题意:给定一个串,每次询问l~r组成的子串的不同子串个数 题解 看到字符串的大小很小 而询问数太多 所以我们预处理任意的 ...
- 【BZOJ2882】工艺(后缀自动机)
[BZOJ2882]工艺(后缀自动机) 题面 BZOJ权限题,良心洛谷 题解 还是一样的,先把串在后面接一遍 然后构建\(SAM\) 直接按照字典序输出\(n\)次就行了 #include<io ...
- 【BZOJ2882】工艺(后缀数组)
[BZOJ2882]工艺(后缀数组) 题面 BZOJ权限题,我爱良心洛谷 题解 最容易的想法: 把字符串在后面接一份 然后求后缀数组就行了... #include<iostream> #i ...
- 【BZOJ3277】串(后缀自动机)
[BZOJ3277]串(后缀自动机) 题面 BZOJ 题解 广义后缀自动机??? 照着别人的打了一遍.. 相当于每个串都构建一个后缀自动机 构建完一个串之后,直接把当前的last指回root就行了?? ...
- 【BZOJ3238】差异(后缀自动机)
[BZOJ3238]差异(后缀自动机) 题面 BZOJ 题解 前面的东西直接暴力算就行了 其实没必要算的正正好 为了方便的后面的计算 我们不考虑\(i,j\)的顺序问题 也就是先求出\(\sum_{i ...
- 【BZOJ3998】弦论(后缀自动机)
[BZOJ3998]弦论(后缀自动机) 题面 BZOJ 题解 这题应该很简单 构建出\(SAM\)后 求出每个点往后还能构建出几个串 按照拓扑序\(dp\)一些就好了 然后就是第\(k\)大,随便搞一 ...
- 【SPOJ】Substrings(后缀自动机)
[SPOJ]Substrings(后缀自动机) 题面 Vjudge 题意:给定一个长度为\(len\)的串,求出长度为1~len的子串中,出现最多的出现了多少次 题解 出现次数很好处理,就是\(rig ...
随机推荐
- ssh框架配置过程
1.pom.xml配置依赖 <?xml version="1.0" encoding="UTF-8"?> <project xmlns=&qu ...
- 解题:JSOI 2011 柠檬
题面 显然分出来的每段两端颜色相同,否则把一边归给旁边的答案不变劣,于是可以$O(n^2)$地dp了:设$dp[i]$表示到第$i$个位置为止的最优解,$dp[i]=dp[j]+a[i]*(s[j]- ...
- 【codevs4696】等差数列
题目大意:给定 N 个整数组成的集合,向集合中添加一个整数,使得这 N+1 个整数组成等差数列,求这样的整数有多少个. 题解: 引理1:若原集合中只有一个元素,则有无数种可能. 引理2:若原集合中有且 ...
- 转头条:阿里p7架构师:三年经验应该具备什么样的技能?
问:工作中,有时候实现一个功能,会去看有没有现成的轮子可用.对于重复造轮子与改造轮子有什么看法? 答:一定会的,其实这也是一个提高技术能力的方法,比如今天想做个日期转换的功能,JDK8有日期的新特性就 ...
- vue+webpack开发(一)
一开始接触这个vue+webpack的时候,实在是摸不着头脑,根本无从下手. 但是经过这两天的研究,其实你会发现vue其实并不难,难度都在webpack你对webpack的理解. webpack顾名思 ...
- OpenResty 高阶实战之————Redis授权登录使用短连接(5000)和长连接(500W) 使用连接池AB压力测试结果
一.短连接开始测试 ab -n 5000 -c 100 -k 127.0.0.1/test_redis_short #demo1 Concurrency Level: Time taken for t ...
- js 获取格林尼治时间戳
昨天在一论坛里看到有朋友问 js 如何获取格林尼治时间戳.不少朋友第一反应是 toGMTString ...确实可以得到格林尼治时间,但不是时间戳.虽然我也没有啥好的方法一步到位的获取,不过至少是获取 ...
- 《Linux命令行与shell脚本编程大全》 第五章理解shell
5.1 1. cat /etc/passwd 可以查看每个用户自己的默认的shell程序. 2.默认的交互shell会在用户登录某个虚拟控制台终端时启动. 不过还有另外一个默认的shell是/bin/ ...
- Web性能优化系列(1):Web性能优化分析
本文由 伯乐在线 - 鸭梨山大 翻译,sunbiaobiao 校稿.未经许可,禁止转载!英文出处:gokulkrishh.github.io.欢迎加入翻译小组. 如果你的网站在1000ms内加载完成, ...
- 详谈ASP.NET的DataReader对象
最近频繁用到了DataReader这个对象,其实对于DataReader,之前也用到过,说实话我个人觉得很不好懂.相比之下觉得DataSet对象好用的多,但是有时取出的数据不需要很多的时候,DataR ...