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)的更多相关文章

  1. 【AtCoder】ARC097 (C - F)题解

    C - K-th Substring 题解 找出第K大的子串,重复的不计入 这个数据范围可能有什么暴力可以艹过去吧,但是K放大的话这就是后缀自动机板子题啊= = 代码 #include <ios ...

  2. 【BZOJ2599】Race(点分治)

    [BZOJ2599]Race(点分治) 题面 BZOJ权限题,洛谷 题解 好久没写过点分治了... 在ppl的帮助下终于想起来了 orz ppl 首先回忆一下怎么求有没有正好是\(K\)的路径 维护一 ...

  3. 【HDU4622】Reincarnation(后缀自动机)

    [HDU4622]Reincarnation(后缀自动机) 题面 Vjudge 题意:给定一个串,每次询问l~r组成的子串的不同子串个数 题解 看到字符串的大小很小 而询问数太多 所以我们预处理任意的 ...

  4. 【BZOJ2882】工艺(后缀自动机)

    [BZOJ2882]工艺(后缀自动机) 题面 BZOJ权限题,良心洛谷 题解 还是一样的,先把串在后面接一遍 然后构建\(SAM\) 直接按照字典序输出\(n\)次就行了 #include<io ...

  5. 【BZOJ2882】工艺(后缀数组)

    [BZOJ2882]工艺(后缀数组) 题面 BZOJ权限题,我爱良心洛谷 题解 最容易的想法: 把字符串在后面接一份 然后求后缀数组就行了... #include<iostream> #i ...

  6. 【BZOJ3277】串(后缀自动机)

    [BZOJ3277]串(后缀自动机) 题面 BZOJ 题解 广义后缀自动机??? 照着别人的打了一遍.. 相当于每个串都构建一个后缀自动机 构建完一个串之后,直接把当前的last指回root就行了?? ...

  7. 【BZOJ3238】差异(后缀自动机)

    [BZOJ3238]差异(后缀自动机) 题面 BZOJ 题解 前面的东西直接暴力算就行了 其实没必要算的正正好 为了方便的后面的计算 我们不考虑\(i,j\)的顺序问题 也就是先求出\(\sum_{i ...

  8. 【BZOJ3998】弦论(后缀自动机)

    [BZOJ3998]弦论(后缀自动机) 题面 BZOJ 题解 这题应该很简单 构建出\(SAM\)后 求出每个点往后还能构建出几个串 按照拓扑序\(dp\)一些就好了 然后就是第\(k\)大,随便搞一 ...

  9. 【SPOJ】Substrings(后缀自动机)

    [SPOJ]Substrings(后缀自动机) 题面 Vjudge 题意:给定一个长度为\(len\)的串,求出长度为1~len的子串中,出现最多的出现了多少次 题解 出现次数很好处理,就是\(rig ...

随机推荐

  1. 【BZOJ1414】[ZJOI2009]对称的正方形(哈希)

    [BZOJ1414][ZJOI2009]对称的正方形(哈希) 题面 BZOJ 洛谷 题解 深思熟虑一波,发现一个矩阵如果左右对称的话,那么它每行都是一个回文串,同理,如果上下对称的话,那么每列都是一个 ...

  2. 【BZOJ2054】疯狂的馒头(并查集,线段树)

    [BZOJ2054]疯狂的馒头(并查集,线段树) 题面 BZOJ 然而权限题,随便找个离线题库看看题吧. 题解 线段树就是个暴力,如果数据可以构造就能卡掉,然而不能构造,要不然复杂度瓶颈成为了读入了. ...

  3. GROUP BY和 HAVING 及 统计函数 执行顺序等

    [我理解:where是对最外层结果进行条件筛选,而having是对分组时分组中的数据进行 组内条件筛选,注意:只能进行筛选,不能进行统计或计算,所有统计或计算都要放在最外层的select 后面,无论是 ...

  4. What does -1 mean in numpy reshape?

    The criterion to satisfy for providing the new shape is that 'The new shape should be compatible wit ...

  5. 创建 OpenStack云主机(十一)

    创建过程 创建虚拟网络 创建m1.nano规格的主机(相等于定义虚拟机的硬件配置) 生成一个密钥对(openstack的原理是不使用密码连接,而是使用密钥对进行连接) 增加安全组规则(用iptable ...

  6. hdu 5181 numbers

    http://acm.hdu.edu.cn/showproblem.php?pid=5181 题意: 有一个栈,其中有n个数1~n按顺序依次进入栈顶,在某个时刻弹出. 其中m个限制,形如数字A必须在数 ...

  7. Java入门系列(七)Java 集合框架(JCF, Java Collections Framework)

    Java 集合概述 List.Set.Map可以看做集合的三大类 java集合就像一个容器,可以将多个对象的引用丢进该容器中. Collection和Map是java集合的根接口. List List ...

  8. 使用渐进式JPEG来提升用户体验

    今天才认识到原来JPEG文件有两种保存方式他们分别是Baseline JPEG(标准型)和Progressive JPEG(渐进式).两种格式有相同尺寸以及图像数据,他们的扩展名也是相同的,唯一的区别 ...

  9. Angular 下的 directive (part 2)

    ngCloak ngCloak指令被使用在,阻止angular模板从浏览器加载的时候出现闪烁的时候.使用它可以避免闪烁问题的出现.   该指令可以应用于<body>元素,但首选使用多个ng ...

  10. TED_Topic7:How we unearthed the spinosaurus

    By Nizar Ibrahim A 50-foot-long carnivore who hunted its prey in rivers 97 million years ago, the sp ...