Codeforces 1186 F. Vus the Cossack and a Graph —— 线段树,贪心

2024-04-06 23:38

本文主要是介绍Codeforces 1186 F. Vus the Cossack and a Graph —— 线段树,贪心,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

This way

题意:

现在有一张简单图,n个点m条边,让你留下最多 ⌈ n + m 2 ⌉ \lceil{\frac{n+m}{2}}\rceil 2n+m条边,假设每个点的当前度数为di,每个点的最终度数要大于等于 ⌈ d i 2 ⌉ \lceil{\frac{d_i}{2}}\rceil 2di。问你留下来哪些边

题解:

我最近做题目都先考虑线段树是否能做…不知道这是不是一个好习惯,但是这道题依旧还是用线段树过了。
首先我考虑减少当前度数最大的点的边,然后和这个点相连的那些点应该删谁?如果删除度数较大的那个的话:
在这里插入图片描述
就比如当前是红色的点的度数最大,蓝色的点的度数比黑色的点的度数大,如果我删掉蓝色的点和红色点的连边,就有可能红色的点的度数正好变成了最小值,那么红黑两点的边就不能删了,这样可能会导致错误。如果删掉红黑两点的边,那么蓝色就可以和它连的别的点继续删。所以我每次找出最大的度数的点之后,对于它连的点按照度数为关键字从小到大排序,然后在合法的情况下依次删掉。
每个点做完之后,这个点下次就不能在被找到,那么就要在线段树中将其抹去。

#include<bits/stdc++.h>
using namespace std;
#define pa pair<int,int>
const int N=1e6+5;
int d[N],f[N];
vector<int>vec[N];
pa mx[N*4];
void build(int l,int r,int root){if(l==r){mx[root]={l,f[l]};return ;}int mid=l+r>>1;build(l,mid,root<<1);build(mid+1,r,root<<1|1);if(mx[root<<1].second>=mx[root<<1|1].second)mx[root]=mx[root<<1];elsemx[root]=mx[root<<1|1];
}
void update(int l,int r,int root,int p,int v){if(l==r){mx[root].second+=v;return ;}int mid=l+r>>1;if(mid>=p)update(l,mid,root<<1,p,v);elseupdate(mid+1,r,root<<1|1,p,v);if(mx[root<<1].second>=mx[root<<1|1].second)mx[root]=mx[root<<1];elsemx[root]=mx[root<<1|1];
}
struct edge{int x,y;
}e[N];
unordered_map<int,bool>del[N];
vector<pa>s;
int main()
{int n,m;scanf("%d%d",&n,&m);for(int i=1;i<=m;i++){scanf("%d%d",&e[i].x,&e[i].y);vec[e[i].x].push_back(e[i].y),vec[e[i].y].push_back(e[i].x);f[e[i].y]++,f[e[i].x]++;}for(int i=1;i<=n;i++)d[i]=(f[i]+1)/2;int res=(m-n)/2;build(1,n,1);while(res>0){s.clear();int x=mx[1].first;for(auto i:vec[x]){if(del[x].count(i)||f[i]==d[i])continue;s.push_back({f[i],i});}sort(s.begin(),s.end());for(auto i:s){f[i.second]--,f[x]--,res--;update(1,n,1,i.second,-1);del[i.second][x]=del[x][i.second]=1;if(f[x]==d[x])break;}update(1,n,1,x,-1e9);}printf("%d\n",m-(m-n)/2+res);for(int i=1;i<=m;i++){if(del[e[i].x].count(e[i].y))continue;printf("%d %d\n",e[i].x,e[i].y);}return 0;
}

这篇关于Codeforces 1186 F. Vus the Cossack and a Graph —— 线段树,贪心的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/881137

相关文章

poj3468(线段树成段更新模板题)

题意:包括两个操作:1、将[a.b]上的数字加上v;2、查询区间[a,b]上的和 下面的介绍是下解题思路: 首先介绍  lazy-tag思想:用一个变量记录每一个线段树节点的变化值,当这部分线段的一致性被破坏我们就将这个变化值传递给子区间,大大增加了线段树的效率。 比如现在需要对[a,b]区间值进行加c操作,那么就从根节点[1,n]开始调用update函数进行操作,如果刚好执行到一个子节点,

hdu1394(线段树点更新的应用)

题意:求一个序列经过一定的操作得到的序列的最小逆序数 这题会用到逆序数的一个性质,在0到n-1这些数字组成的乱序排列,将第一个数字A移到最后一位,得到的逆序数为res-a+(n-a-1) 知道上面的知识点后,可以用暴力来解 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#in

hdu1689(线段树成段更新)

两种操作:1、set区间[a,b]上数字为v;2、查询[ 1 , n ]上的sum 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#include<queue>#include<set>#include<map>#include<stdio.h>#include<stdl

usaco 1.3 Barn Repair(贪心)

思路:用上M块木板时有 M-1 个间隙。目标是让总间隙最大。将相邻两个有牛的牛棚之间间隔的牛棚数排序,选取最大的M-1个作为间隙,其余地方用木板盖住。 做法: 1.若,板(M) 的数目大于或等于 牛棚中有牛的数目(C),则 目测 给每个牛牛发一个板就为最小的需求~ 2.否则,先对 牛牛们的门牌号排序,然后 用一个数组 blank[ ] 记录两门牌号之间的距离,然后 用数组 an

hdu 1754 I Hate It(线段树,单点更新,区间最值)

题意是求一个线段中的最大数。 线段树的模板题,试用了一下交大的模板。效率有点略低。 代码: #include <stdio.h>#include <string.h>#define TREE_SIZE (1 << (20))//const int TREE_SIZE = 200000 + 10;int max(int a, int b){return a > b ? a :

hdu 1166 敌兵布阵(树状数组 or 线段树)

题意是求一个线段的和,在线段上可以进行加减的修改。 树状数组的模板题。 代码: #include <stdio.h>#include <string.h>const int maxn = 50000 + 1;int c[maxn];int n;int lowbit(int x){return x & -x;}void add(int x, int num){while

poj 3190 优先队列+贪心

题意: 有n头牛,分别给他们挤奶的时间。 然后每头牛挤奶的时候都要在一个stall里面,并且每个stall每次只能占用一头牛。 问最少需要多少个stall,并输出每头牛所在的stall。 e.g 样例: INPUT: 51 102 43 65 84 7 OUTPUT: 412324 HINT: Explanation of the s

poj 2976 分数规划二分贪心(部分对总体的贡献度) poj 3111

poj 2976: 题意: 在n场考试中,每场考试共有b题,答对的题目有a题。 允许去掉k场考试,求能达到的最高正确率是多少。 解析: 假设已知准确率为x,则每场考试对于准确率的贡献值为: a - b * x,将贡献值大的排序排在前面舍弃掉后k个。 然后二分x就行了。 代码: #include <iostream>#include <cstdio>#incl

poj 1127 线段相交的判定

题意: 有n根木棍,每根的端点坐标分别是 px, py, qx, qy。 判断每对木棍是否相连,当他们之间有公共点时,就认为他们相连。 并且通过相连的木棍相连的木棍也是相连的。 解析: 线段相交的判定。 首先,模板中的线段相交是不判端点的,所以要加一个端点在直线上的判定; 然后,端点在直线上的判定这个函数是不判定两个端点是同一个端点的情况的,所以要加是否端点相等的判断。 最后

HDU4737线段树

题目大意:给定一系列数,F(i,j)表示对从ai到aj连续求或运算,(i<=j)求F(i,j)<=m的总数。 const int Max_N = 100008 ;int sum[Max_N<<2] , x[Max_N] ;int n , m ;void push_up(int t){sum[t] = sum[t<<1] | sum[t<<1|1] ;}void upd