[Atcoder Regular Contest 065] Tutorial
Link:
C:
最好采取逆序贪心,否则要多考虑好几种情况
(从前往后贪心的话不能无脑选“dreamer”,"er"可能为"erase"/"eraser"的前缀)
#include <bits/stdc++.h> using namespace std;
string s;
int main()
{
cin>>s;
for(int cur=s.size();cur;)
if(cur>=&&s.substr(cur-,)=="erase") cur-=;
else if(cur>=&&s.substr(cur-,)=="dream") cur-=;
else if(cur>=&&s.substr(cur-,)=="eraser") cur-=;
else if(cur>=&&s.substr(cur-,)=="dreamer") cur-=;
else return puts("NO"),;
puts("YES");
return ;
}
Problem C
注意:$strncpy$和$s.substr()$的第三个参数为长度
D:
题意:求与点$i$在图$G1$和图$G2$中的连通块序号均相同的点的个数
设1个点的序号对为$(a,b)$,其实就是求每对$(a,b)$的个数
然而我一开始沙茶得以为只能$O(n^2)$来统计……
明显用上$set$就只要$O(m*log(n))$了嘛,多维的统计都可以用$set$优化!
先跑两遍$dfs$求出每个点的$(a,b)$,再利用$set$优化$pair$的统计就好了
#include <bits/stdc++.h> using namespace std;
typedef pair<int,int> P;
const int MAXN=2e5+;
map<P,int> mp;
struct edge{int nxt,to;}e[MAXN<<];
int c1[MAXN],c2[MAXN],sum1[MAXN],sum2[MAXN];
int n,k,l,x,y,head[MAXN],vis[MAXN],res[MAXN],tot,cnt; void add_edge(int from,int to)
{
e[++tot].nxt=head[from];e[tot].to=to;head[from]=tot;
e[++tot].nxt=head[to];e[tot].to=from;head[to]=tot;
} void dfs(int x,int c,int* col)
{
vis[x]=true;col[x]=c;
for(int i=head[x];i;i=e[i].nxt)
if(!vis[e[i].to]) dfs(e[i].to,c,col);
} void solve(int esum,int* col)
{
tot=cnt=;
memset(head,,sizeof(head));
memset(vis,false,sizeof(vis));
for(int i=;i<=esum;i++)
scanf("%d%d",&x,&y),add_edge(x,y);
for(int i=;i<=n;i++)
if(!vis[i]) dfs(i,++cnt,col);
} int main()
{
scanf("%d%d%d",&n,&k,&l);
memset(res,0x3f,sizeof(res));
solve(k,c1);solve(l,c2);
for(int i=;i<=n;i++)
sum1[c1[i]]++,sum2[c2[i]]++;
for(int i=;i<=n;i++)
mp[P(c1[i],c2[i])]++;
for(int i=;i<=n;i++)
printf("%d ",mp[P(c1[i],c2[i])]); return ;
}
Problem D
如果只求连通块最好使用并查集
提升效率并能减少代码量
E:
好像题目中专门提到曼哈顿/切比雪夫距离的任意一个往令一个想就对了……
设$a,b$两点间的初始距离为$d$,那么将所有距离为$d$的两点都连边,找到$a$所在的连通块即可
但曼哈顿距离涉及到了两个量,很难进行统一处理,因此要转换成切比雪夫距离
于是问题转化为将$max(|X_i-X_j|,|Y_i-Y_j|)=d$的$i,j$连边
明显可以排序后采取二分的方式,对于每个$X_i$,找到$X_j=X_i+d$中符合$Y_i-d\le Y_j\le Y_i+d$的点的个数,记为$cnt[i]$
接下来将$X_i$与$Y_i$交换后进行相同的操作,只不过要排除$|X_i-X_j|=|Y_i-Y_j|=d$的情况
最后$dfs$求出$a$所在连通块中$cnt$的总和$res=\sum cnt[i]$
要注意连边的细节:不能暴力连边,那样复杂度变为了$O(n^2)$
可以发现每次是向一个区间连边,且只要保证连通性不变,因此只要将区间连成一条链就行了!复杂度为$O(n)$
#include <bits/stdc++.h> using namespace std;
typedef long long ll;
const int MAXN=1e5+;
struct edge{int nxt,to;}e[MAXN*];//一定要开8倍空间
struct data{int x,y,id;}dat[MAXN];
int n,d,a,b,x,y,head[MAXN],vis[MAXN],cnt[MAXN],tot=;
ll res=; void add_edge(int from,int to)
{
e[++tot].nxt=head[from];e[tot].to=to;head[from]=tot;
e[++tot].nxt=head[to];e[tot].to=from;head[to]=tot;
} bool operator < (data a,data b)
{
if(a.x!=b.x) return a.x<b.x;
return a.y<b.y;
} void solve(int k)
{
sort(dat+,dat+n+);
int cur,lst,l,r;
for(cur=lst=l=r=;cur<=n;cur++)
{
while((dat[l].x<dat[cur].x+d||dat[l].x==dat[cur].x+d&&dat[l].y<dat[cur].y-k)&&l<=n) l++;
while((dat[r].x<dat[cur].x+d||dat[r].x==dat[cur].x+d&&dat[r].y<=dat[cur].y+k)&&r<=n) r++;
if(l>n) break;
cnt[dat[cur].id]+=r-l;lst=max(lst,l);
if(lst>=r) continue; add_edge(dat[cur].id,dat[lst].id);
for(;lst<r-;lst++) add_edge(dat[lst].id,dat[lst+].id);
}
} void dfs(int x)
{
vis[x]=true;res+=cnt[x];
for(int i=head[x];i;i=e[i].nxt)
if(!vis[e[i].to]) dfs(e[i].to);
} int main()
{
scanf("%d%d%d",&n,&a,&b);
for(int i=;i<=n;i++)
scanf("%d%d",&x,&y),dat[i]={x+y,x-y,i}; d=max(abs(dat[a].x-dat[b].x),abs(dat[a].y-dat[b].y));
solve(d);
for(int i=;i<=n;i++) swap(dat[i].x,dat[i].y);
solve(d-); dfs(a);printf("%lld",res);
return ;
}
Problem E
Tip:此题边集数组要开8倍空间,但我觉得4倍就够了?跪求神犇们指点啊……
F:
对于每个点而言,其实只要关心其能翻转到的最右位置就好了,中间重复的翻转可以不管
于是可以先求出$lst[i]$表示第$i$位能翻转到的最右位置
令$dp[i][j]$为已经确定前$i-1$位,还有$j$个1可以自由支配时的方案数
常规地分第$i$位选$0/1$递推即可,$dp[n+1][0]$即为答案
#include <bits/stdc++.h> using namespace std;
const int MAXN=,MOD=1e9+;
char s[MAXN];
int n,m,x,y,pre[MAXN],lst[MAXN],dp[MAXN][MAXN]; int main()
{
scanf("%d%d%s",&n,&m,s+);s[n+]='';
for(int i=;i<=n+;i++) //对第n+1项也要更新
pre[i]=(s[i]-'')+pre[i-],lst[i]=i;
for(int i=;i<=m;i++)
scanf("%d%d",&x,&y),lst[x]=max(lst[x],y);
for(int i=;i<=n+;i++) lst[i]=max(lst[i],lst[i-]); dp[][pre[lst[]]]=;
for(int i=;i<=n;i++)
for(int j=;j<=n;j++)
if(dp[i][j])
{
int l=lst[i]+,r=lst[i+];
if(j) (dp[i+][pre[r]-pre[l-]+j-]+=dp[i][j])%=MOD;
if(lst[i]-(i-)-j) (dp[i+][pre[r]-pre[l-]+j]+=dp[i][j])%=MOD;
}
printf("%d",dp[n+][]);
return ;
}
Problem F
[Atcoder Regular Contest 065] Tutorial的更多相关文章
- [Atcoder Regular Contest 060] Tutorial
Link: ARC060 传送门 C: 由于难以维护和更新平均数的值: $Average->Sum/Num$ 这样我们只要用$dp[i][j][sum]$维护前$i$个数中取$j$个,且和为$s ...
- [Atcoder Regular Contest 061] Tutorial
Link: ARC061 传送门 C: 暴力$dfs$就好了 #include <bits/stdc++.h> using namespace std; typedef long long ...
- [Atcoder Regular Contest 064] Tutorial
Link: ARC064 传送门 C: 贪心+对边界的特殊处理 #include <bits/stdc++.h> using namespace std; typedef long lon ...
- [Atcoder Regular Contest 063] Tutorial
Link: ARC063 传送门 C: 将每种颜色的连续出现称为一段,寻找总段数即可 #include <bits/stdc++.h> using namespace std; ,len; ...
- [Atcoder Regular Contest 062] Tutorial
Link: ARC 062 传送门 C: 每次判断增加a/b哪个合法即可 并不用判断两个都合法时哪个更优,因为此时两者答案必定相同 #include <bits/stdc++.h> usi ...
- AtCoder Regular Contest 061
AtCoder Regular Contest 061 C.Many Formulas 题意 给长度不超过\(10\)且由\(0\)到\(9\)数字组成的串S. 可以在两数字间放\(+\)号. 求所有 ...
- AtCoder Regular Contest 094 (ARC094) CDE题解
原文链接http://www.cnblogs.com/zhouzhendong/p/8735114.html $AtCoder\ Regular\ Contest\ 094(ARC094)\ CDE$ ...
- AtCoder Regular Contest 092
AtCoder Regular Contest 092 C - 2D Plane 2N Points 题意: 二维平面上给了\(2N\)个点,其中\(N\)个是\(A\)类点,\(N\)个是\(B\) ...
- AtCoder Regular Contest 093
AtCoder Regular Contest 093 C - Traveling Plan 题意: 给定n个点,求出删去i号点时,按顺序从起点到一号点走到n号点最后回到起点所走的路程是多少. \(n ...
随机推荐
- quick 用系统浏览器打开url
需求描述: 在我们的游戏里面增加一个链接,直接用浏览器打开,进入到对应网站,进行一些支付活动. 解决: 于是我去百度了一下,发现了这篇文章,http://blog.csdn.net/teng_onth ...
- 逃生(HDU4857 + 反向拓扑排序)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4857 题面是中文题面,就不解释题意了,自己点击链接去看下啦~这题排序有两个条件,一个是按给定的那个序列 ...
- ...args剩余参数用法
剩余参数语法允许我们将一个不定数量的参数表示为一个数组. function sum(...theArgs) { return theArgs.reduce((previous, current) ...
- 用例图(Use Case Diagram)
用例图(Use Case Diagram) 执行者/参与者(Actor): 表示与您的应用程序或系统进行交互的用户.组织或外部系统.用一个小人表示. 用例(Use Case): 即系统具有的功能,在用 ...
- 空间数据库系列一:geomesa&sparksql 分析环境搭建
geomesa sparksql 分析环境搭建 1.安装hbase-1.3.2.1 standlone版本,作为geomesa的store a.修改配置文件:hbase-1.3.2.1/conf/hb ...
- js 实时显示字数
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- shellcheck 帮助你写出更好的脚本
简介 shellcheck 是一款实用的 shell脚本静态检查工具. 首先,可以帮助你提前发现并修复简单的语法错误,节约时间.每次都需要运行才发现写错了一个小地方,确实非常浪费时间. 其次,可以针对 ...
- 【bzoj4530】大融合(LCT的子树维护)
LCT维护子树并没有想象中的那么难,在这里只是复习下. (其他的维护子树的题目,可见:“共价大爷游长沙”) 只要记录下虚边连接的信息就好了. #include<bits/stdc++.h> ...
- webapi-2 接口参数
1. 实例 using System; using System.Collections.Generic; using System.Linq; using System.Net; using Sys ...
- django渲染模板时跟vue使用的{{ }}冲突解决方法
var vm = new Vue({ el: '#app', // 分割符: 修改vue中显示数据的语法, 防止与django冲突 delimiters: ['[[', ']]'], data: { ...