@description@

给定一个 N 个点 M 条边的图,每条为黑色或者白色。

现在让你求一个生成树,使得生成树中黑色边数量等于白色边数量。

Input

第一行两个整数 n, m (1 ≤ n ≤ 10^3, 1 ≤ m ≤ 10^5)。

接下来 m 行,每行描述了一条边 x, y,并给定它的颜色 —— S 为白色,M 为黑色。

注意可以有重边自环。

Output

如果无解,输出 -1。

否则,先输出边的数量,再在下一行依次给出每条边的编号(根据输入顺序)。

如果有多解,输出任意一个解。

Examples

Input

3 3

1 2 S

1 3 M

2 3 S

Output

2

2 1

@solution@

考虑黑色边中有一些必要边:即使加入所有的白边,也需要这些必要边才能保证连通。

白色边,类似地也有必要边的定义:加入所有的黑边也需要这些边才能连通。

可以通过并查集 + 先扫描白色边 + 再尝试加入黑色边求出黑色必要边。同理可以求出白色必要边。

不妨考虑一种情况:即所有边都不是必要边。

这种情况下,黑色边可以让整张图保持连通,白色边也可以让整张图保持连通。

那么我只需要任意加入 (n - 1) / 2 条不形成环的白边,一定可以另外选出 (n - 1) / 2 条黑边形成生成树。

至于为什么。将白边连接的连通块缩成一个新点,则新图中黑边依然可以让整张图保持连通,而连通图一定可以求出一个生成树。

实现直接并查集加 (n - 1) / 2 条白边,再尽量多地塞黑边。

假如我把必要边连接的连通块缩成点,此时所有边都不为必要边,则归于上面那一种情况。

注意特殊情况,如 n - 1 为奇数,或整张图根本不连通,或必要边太多等。

@accepted code@

#include<cstdio>
#include<vector>
#include<algorithm>
using namespace std;
const int MAXN = 100000;
int x[MAXN + 5], y[MAXN + 5];
vector<int>v[2], ans;
bool tag[2][MAXN + 5];
int fa[MAXN + 5], n, m;
int find(int x) {
return fa[x] = (fa[x] == x ? x : find(fa[x]));
}
bool unite(int x, int y) {
int fx = find(x), fy = find(y);
if( fx != fy ) {
fa[fx] = fy;
return true;
}
else return false;
}
int tot[2];
void get(int t) {
for(int i=1;i<=n;i++) fa[i] = i;
for(int i=0;i<v[!t].size();i++)
unite(x[v[!t][i]], y[v[!t][i]]);
for(int i=0;i<v[t].size();i++)
if( unite(x[v[t][i]], y[v[t][i]]) )
ans.push_back(v[t][i]), tot[t]++, tag[t][i] = true;
}
int main() {
scanf("%d%d", &n, &m);
for(int i=1;i<=m;i++) {
scanf("%d%d", &x[i], &y[i]);
char ch = getchar();
while( ch != 'S' && ch != 'M' )
ch = getchar();
if( ch == 'S' ) v[0].push_back(i);
if( ch == 'M' ) v[1].push_back(i);
}
if( (n - 1) & 1 ) {
puts("-1");
return 0;
}
int mid = (n - 1) >> 1;
get(0), get(1);
if( tot[0] > mid || tot[1] > mid ) {
puts("-1");
return 0;
}
for(int i=1;i<=n;i++) fa[i] = i;
for(int i=0;i<ans.size();i++) unite(x[ans[i]], y[ans[i]]);
for(int i=0;i<v[0].size()&&tot[0]<mid;i++)
if( !tag[0][i] && unite(x[v[0][i]], y[v[0][i]]) )
ans.push_back(v[0][i]), tot[0]++, tag[0][i] = true;
for(int i=0;i<v[1].size()&&tot[1]<mid;i++)
if( !tag[1][i] && unite(x[v[1][i]], y[v[1][i]]) )
ans.push_back(v[1][i]), tot[1]++, tag[1][i] = true;
if( tot[0] != mid || tot[1] != mid ) {
puts("-1");
return 0;
}
printf("%d\n", ans.size());
for(int i=0;i<ans.size();i++)
printf("%d ", ans[i]);
}

@details@

定睛一看。嗯?wqs二分?

不对wqs二分只能求最小权和不能求方案。

注意各种无解的情况吧。其他没有什么细节。

如果一句话总结的话,大概就是:任意一个连通块必然有生成树。

@codeforces - 141E@ Clearing Up的更多相关文章

  1. CodeForces 141E: ...(最小生成树)

    [条件转换] 两两之间有且只有一条简单路径<==>树 题意:一个图中有两种边,求一棵生成树,使得这棵树中的两种边数量相等. 思路: 可以证明,当边的权是0或1时,可以生成最小生成树到最大生 ...

  2. python爬虫学习(5) —— 扒一下codeforces题面

    上一次我们拿学校的URP做了个小小的demo.... 其实我们还可以把每个学生的证件照爬下来做成一个证件照校花校草评比 另外也可以写一个物理实验自动选课... 但是出于多种原因,,还是绕开这些敏感话题 ...

  3. 【Codeforces 738D】Sea Battle(贪心)

    http://codeforces.com/contest/738/problem/D Galya is playing one-dimensional Sea Battle on a 1 × n g ...

  4. 【Codeforces 738C】Road to Cinema

    http://codeforces.com/contest/738/problem/C Vasya is currently at a car rental service, and he wants ...

  5. 【Codeforces 738A】Interview with Oleg

    http://codeforces.com/contest/738/problem/A Polycarp has interviewed Oleg and has written the interv ...

  6. CodeForces - 662A Gambling Nim

    http://codeforces.com/problemset/problem/662/A 题目大意: 给定n(n <= 500000)张卡片,每张卡片的两个面都写有数字,每个面都有0.5的概 ...

  7. CodeForces - 274B Zero Tree

    http://codeforces.com/problemset/problem/274/B 题目大意: 给定你一颗树,每个点上有权值. 现在你每次取出这颗树的一颗子树(即点集和边集均是原图的子集的连 ...

  8. CodeForces - 261B Maxim and Restaurant

    http://codeforces.com/problemset/problem/261/B 题目大意:给定n个数a1-an(n<=50,ai<=50),随机打乱后,记Si=a1+a2+a ...

  9. CodeForces - 696B Puzzles

    http://codeforces.com/problemset/problem/696/B 题目大意: 这是一颗有n个点的树,你从根开始游走,每当你第一次到达一个点时,把这个点的权记为(你已经到过不 ...

随机推荐

  1. Browsersync 浏览器自动刷新

    Browsersync 是一个很好用的工具,它可以实时监测文件的变动然后自动刷新浏览器,不用每次去点刷新或F5,特别在调试样式时非常有用. browsersync中文网  http://www.bro ...

  2. python统计一个文本中重复行数的方法

    python统计一个文本中重复行数的方法 这篇文章主要介绍了python统计一个文本中重复行数的方法,涉及针对Python中dict对象的使用及相关本文的操作,具有一定的借鉴价值,需要的朋友可以参考下 ...

  3. 计算机组成原理(电脑硬件&语言分类)

    计算机组成原理 一.电脑硬件配置 CPU :中央处理器(人类的大脑) -飞机 内存:存放一些临时数据(人类的短暂记忆-右脑) -高铁 硬盘:存储永久数据(左脑-长期记忆) - 汽车 输入输出:键盘鼠标 ...

  4. python基础--线程、进程

    并发编程: 操作系统:(基于单核研究) 多道技术: 1.空间上的复用 多个程序共用一个计算机 2.时间上的复用 切换+保存状态 例如:洗衣 烧水 做饭 切换: 1.程序遇到IO操作系统会立刻剥夺着CP ...

  5. day38 05-Spring的BeanFactory与ApplicationContext区别

    ApplicationContext怎么知道它是一个工厂呢? BeanFactory也可以做刚才那些事情,只不过ApplicationContext对它有扩展.ApplicationContext间接 ...

  6. NOIP模拟 9.09

    AK300分 果实计数 (count.pas/.c/.cpp) 时间限制:1s,空间限制32MB 题目描述: 淘淘家有棵奇怪的苹果树,这棵树共有n+1层,标号为0~n.这棵树第0层只有一个节点,为根节 ...

  7. 如何制作可以在 MaxCompute 上使用的 crcmod

    之前我们介绍过在 PyODPS DataFrame 中使用三方包.对于二进制包而言,MaxCompute 要求使用包名包含 cp27-cp27m 的 Wheel 包.但对于部分长时间未更新的包,例如 ...

  8. 微信小程序--底部tab样式修改

    tab图标个数是最少2个,最多5个 主题默认是默认的浅灰色线条 修改后(只有black和white两种样式修改) 在app.json中

  9. IDG资本全球拼图:近10年揽26家独角兽,最敢出手VC再造"VC+"

    IDG资本全球拼图:近10年揽26家独角兽,最敢出手VC再造"VC+" 2017-04-01 15:33   两天前,IDG资本合伙人过以宏提出的“VC+”,又有了新的内涵——全球 ...

  10. jstree设置checkbox单选

    jstree设置插件checkbox只允许单选 jstree version console.log($.jstree.version); 3.3.8 单选配置参数: $.jstree.default ...