传送门

来源:2019 年“浪潮杯”第十届山东省 ACM 省赛

题意:

  对于一个包含n个数的(n为奇数)序列val[ ],排序后的 val[ (n+1) / 2 ] 定义为 median;

  有 n 个数,并有 m 组关系,对于第 i 组关系 ai,bi 代表第 val[ai] > val[bi];

  但并没有给出具体的数值;

  输出一个包含 n 个元素的数组 s[ ] ;

  让你判断第 i 个数 val[ i ]是可能为中位数,如果是,第 i 位为 1;

  如果不是,第 i 位为 0;

  输出 n 个数,其中第 i 个数为 0 或 1,含义如上;

题解:

  首先,特判两种情况:

    ①ai = bi

    ②给出的 m 个关系有环;

  对于这两种情况,输出 n 个 0;

  除了这两种情况外,就是一个有向无环图;

  如何判断第 i 位是否为 median 呢?

  搜索:

  正向搜索比第 i 个数小的数的总个数 tot1;

  反向搜索比第 i 个数大的数的总个数 tot2;

  那么,还剩下 res = n-(tot1+tot2+1) 个数;

  如果 res ≥ |tot1-tot2|,那么第 i 个数就是中位数;

AC代码:

 #include<bits/stdc++.h>
using namespace std;
#define mem(a,b) memset(a,b,sizeof(a))
const int maxn=1e4+; int n,m;
int num;
int head[maxn];
bool vis[];
bool isCir;
char s[];
struct Edge
{
int to;
int next;
}G[maxn<<];
void addEdge(int u,int v)
{
G[num]={v,head[u]};
head[u]=num++;
}
struct SCC
{
vector<int >vs;
void DFS(int u)
{
vis[u]=true;
for(int i=head[u];~i;i=G[i].next)
{
int v=G[i].to;
if((i&) || vis[v])
continue;
DFS(v);
}
vs.push_back(u);
}
void RDFS(int u,int k)
{
vis[u]=true;
for(int i=head[u];~i;i=G[i].next)
{
int v=G[i].to;
if(!(i&) || vis[v])
continue;
RDFS(v,k);
}
}
int scc()
{
vs.clear();
mem(vis,false);
for(int i=;i <= n;++i)
if(!vis[i])
DFS(i); int k=;
mem(vis,false);
for(int i=vs.size()-;i >= ;--i)
if(!vis[vs[i]])
RDFS(vs[i],++k);
return k;
}
}_scc;
int DFS(int u)///正向搜索比第 i 个数小的数
{
int ans=;
vis[u]=true;
for(int i=head[u];~i;i=G[i].next)
{
int v=G[i].to;
if((i&) || vis[v])
continue;
ans += DFS(v);
}
return ans;
}
int RDFS(int u)///反向搜索比第 i 个数大的数
{
int ans=;
vis[u]=true;
for(int i=head[u];~i;i=G[i].next)
{
int v=G[i].to;
if(!(i&) || vis[v])
continue;
ans += RDFS(v);
}
return ans;
}
bool isSat(int u)
{
mem(vis,false);
int tot1=DFS(u)-;
mem(vis,false);
int tot2=RDFS(u)-; return abs(tot1-tot2) <= (n-tot1-tot2-) ? true:false;
}
char *Solve()
{
mem(s,'');
s[n]='\0';
int k=_scc.scc();///强连通分量分解判断是否含有环
if(k < n)
isCir=true;
if(isCir)
return s; for(int i=;i <= n;++i)
if(isSat(i))///判断第 i 个数是否为median
s[i-]='';
return s;
}
void Init()
{
num=;
mem(head,-);
isCir=false;
}
int main()
{
// freopen("C:\\Users\\hyacinthLJP\\Desktop\\in&&out\\contest","r",stdin);
int test;
scanf("%d",&test);
while(test--)
{
scanf("%d%d",&n,&m);
Init();
for(int i=;i <= m;++i)
{
int u,v;
scanf("%d%d",&u,&v);
if(u == v)
isCir=true;
addEdge(u,v);///正向边
addEdge(v,u);///反向边
}
printf("%s\n",Solve());
}
return ;
}

凯少思路:

如果 tot1 ≤ n/2 && tot2 ≤ n/2 返回 true;

在返回结果的时候改成这句也可以,我的判断方法也对;

代码:

 #include <iostream>
#include <vector>
#include <cstring>
#include <queue>
#include <cstdio>
#include <algorithm>
#include <cmath>
#include <stack>
using namespace std;
const int ss = ;
int v1[ss];
int v2[ss];
int dfn[ss];
int low[ss];
int vis[ss];
int son1[ss];
int son2[ss];
int pot[ss];
vector < int >G[ss];
vector < int >P[ss];
stack < int >S;
int scc, tim, n, m, t;
void start()
{
memset(v1, , sizeof(v1));
memset(v2, , sizeof(v2));
memset(dfn, , sizeof(dfn));
memset(vis, , sizeof(vis));
memset(low, , sizeof(low));
memset(pot, , sizeof(pot));
memset(son1, , sizeof(son1));
memset(son2, , sizeof(son2));
for (int i = ; i <= n; i++)
{
G[i].clear();
P[i].clear();
}
while (!S.empty())
S.pop();
scc = tim = n = m = ;
}
void tar(int u)
{
int v;
low[u] = dfn[u] = ++tim;
S.push(u);
vis[u] = ;
for (int i = ; i < G[u].size(); i++)
{
v = G[u][i];
if (!dfn[v])
{
tar(v);
if (low[u] > low[v])
low[u] = low[v];
} else if (vis[v] && low[u] > dfn[v])
low[u] = dfn[v];
}
if (low[u] == dfn[u])
{
scc++;
do
{
v = S.top();
S.pop();
vis[v] = ;
} while (v != u);
}
}
int dfs1(int x)
{
v1[x] = ;
for (int i = ; i < G[x].size(); i++)
{
if (!v1[G[x][i]])
son1[x] += dfs1(G[x][i]);
}
return son1[x] + ;
}
int dfs2(int x)
{
v2[x] = ;
for (int i = ; i < P[x].size(); i++)
{
if (!v2[P[x][i]])
son2[x] += dfs2(P[x][i]);
}
return son2[x] + ;
}
void print(int x, int p)
{
for (int i = ; i <= n; i++)
{
if (i == p)
printf("");
else
printf("");
}
printf("\n");
}
int main()
{
cin >> t;
while (t--)
{
start();
int ok = ;
cin >> n >> m;
for (int i = ; i <= m; i++)
{
int a, b;
scanf("%d%d", &a, &b);
if (a == b)
ok = ;
G[a].push_back(b);
P[b].push_back(a);
}
if (!ok)
{
print(n, );
} else
{
for (int i = ; i <= n; i++)
if (!dfn[i])
tar(i);
if (scc != n)
print(n, );
else
{
int point = ;
for (int i = ; i <= n; i++)
{
dfs1(i);
dfs2(i);
if ((son1[i] <= (n / )) && (son2[i] <= (n / )))
pot[i] = ;
memset(son1, , sizeof(son1));
memset(son2, , sizeof(son2));
memset(v1, , sizeof(v1));
memset(v2, , sizeof(v2));
}
for (int i = ; i <= n; i++)
cout << pot[i];
cout << "\n";
}
}
}
return ;
}

因为 n 很小,所以对每个点跑两次DFS并不会超时;

但,如果 n 大了呢,那制定不能每个点跑两次DFS了,那该肿么办呢?

本蒟蒻还没想出来~~~~~~~~

据说,此题正解为求解两次拓扑序;

https://paste.ubuntu.com/p/xXxYdnDRBV/

zoj 4124 "Median" (思维?假的图论?)的更多相关文章

  1. ZOJ 4124 拓扑排序+思维dfs

    ZOJ - 4124Median 题目大意:有n个元素,给出m对a>b的关系,问哪个元素可能是第(n+1)/2个元素,可能的元素位置相应输出1,反之输出0 省赛都过去两周了,现在才补这题,这题感 ...

  2. ZOJ 3829 贪心 思维题

    http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3829 现场做这道题的时候,感觉是思维题.自己智商不够.不敢搞,想着队友智商 ...

  3. D - The Lucky Week ZOJ - 3939 (思维)

    题目链接: D - The Lucky Week  ZOJ - 3939 题目大意:幸运的星期指,星期一为每个月的1 or 11 or 21号.给出第一个幸运星期的时间,问从当前的日起开始.第n个的日 ...

  4. Beauty of Array ZOJ - 3872(思维题)

    Edward has an array A with N integers. He defines the beauty of an array as the summation of all dis ...

  5. ACM-ICPC 2018 青岛赛区现场赛 D. Magic Multiplication && ZOJ 4061 (思维+构造)

    题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=4061 题意:定义一个长度为 n 的序列 a1,a2,..,an ...

  6. JOISC2014 Day2 E "交朋友" (思维+假的SCC)

    传送门 题目描述 你是活跃在历史幕后的一名特工,为了世界和平而夜以继日地努力着. 这个世界有N个国家,编号为1..N; 你的目的是在这N个国家之间建立尽可能多的友好关系. 你为了制定一个特工工作的计划 ...

  7. ZOJ Monthly, January 2018 训练部分解题报告

    A是水题,此处略去题解 B - PreSuffix ZOJ - 3995 (fail树+LCA) 给定多个字符串,每次询问查询两个字符串的一个后缀,该后缀必须是所有字符串中某个字符串的前缀,问该后缀最 ...

  8. 怎么用MindManager制作议论文思维导图

    大家都写过作文吧,做小学到高考到大学,这是谁也摆脱不了的,但是大家写作文会提前把自己的思路整理出来吗?让自己行文更为顺畅,作文更为流利吗?特别是关于议论文,一直是高考写作的一个重点篇目,写好议论文,就 ...

  9. [NOIP 2005] 运输计划

    link 这是一道假的图论 思维难度很低,代码量偏高 就是一道板子+二分 树上差分就AC了 注意卡常即可 二分枚举答案x,为时间长度 将每一个长度大于x的计划链长记录下来(有几个,总需要减少多少长度) ...

随机推荐

  1. iOS GCD 使用

    1. dispatch queue的概念 dispatch queue分成以下三种: a)运行在主线程的Main queue,通过dispatch_get_main_queue获取.dispatch_ ...

  2. python判断输入日期是该年的第几天

    1.输入日期,判断日期是该年度的第几天 iyear = int(input("请输入年:\n")) imonth = int(input("请输入月:\n")) ...

  3. 实现自定义docker 镜像共享

    我觉得docker最大的便利性体现在可以实现镜像共享,方便团队在同一环境下开发.当然docker的强大之处不止于此. 接下来我用一个例子来演示如何进行docker镜像共享,步骤如下(Ubuntu): ...

  4. Hdu 1800 字符串hash

    题目链接 题意: 给出n(n<=3000)个字符串(长度<30,数字组成,肯能存在前导0), 问该序列最少可以分成多少个单调序列.可以转化成求相同字符串的个数的最大值 附上代码: /*** ...

  5. 关于Apple Watch,听听开发了两个月Watch App的工程师怎么说

    今年1月份有幸应苹果邀请,秘密参与苹果 Watch App 的真机现场调试.4月份,Apple Watch 会正式上市.在这之前,也算是亲自抢先体验了 Apple Watch,以及开发了一下 Watc ...

  6. 洛谷2387 BZOJ3669魔法森林题解

    题目链接 BZ链接 这道题被很多人用spfa水了过去,表示很... 其实spfa很好卡,这组数据可以卡掉大多数spfa 链接:密码:rjvk 这里讲一下LCT的做法 我们按照a将边排序,然后依次添加 ...

  7. 第十四届中北大学ACM程序设计竞赛 J.ZBT的游戏

    问题描述 第14届中北大学程序设计竞赛来了,集训队新买了一大堆气球,气球一共有K种颜色(1<=K<=256),气球的颜色从1-K编号. ZBT童心未泯,他发明了一种摆放气球的游戏,规则如下 ...

  8. ROS报错:IOError:[Errno 13]permission denied: /home/neousys/.ros/roscore-11311.pid"

    在安装ROS后启动ROS,输入:roscore 时报错: 这个问题是由于该路径下ros文件权限造成的. 输入以下命令修改权限: sudo chmod -R ~/.ros/ 修改完成后再次输入rosco ...

  9. Where是深复制

    从一个List中通过Where过滤出来的子List或Fist出来的单独元素,与原List是深复制关系.也就是说修改子list,或单独元素,并不会影响原List中的对象. 如:Geom g = Geom ...

  10. iptables 累计(Accounting)

    对於每一条规则,核心各自设置两个专属的计数器,用于累计符合该条件的封包数,以及这些封包的总位元组数.这两项资讯可用於统计网路用量. 举例来說,假设有一台Internet闸道器路,eth0接内部网络,e ...