传送门

首先考虑如何算出答案,考虑枚举中间那个点,显然每个点作为中间的点的次数为入度乘出度

所以答案就是每个点的入度乘出度之和

然后每个点开一个 $vector$ 维护从它出去的点数,每次修改的时候直接暴力改出度然后暴力删边并加入新边

这样可以证明复杂度是对的,这里有两种证明,其中第二种是来自官方题解

证明 $1$ :

不妨让值小的点指向值更大的点

首先对于所有出度小于根号 $m$ 的点,单次暴力复杂度显然小于 $\sqrt {m}$,那么总复杂度不超过 $q \sqrt {m}$

对于出度大于根号 $m$ 的点 $x$,考虑把初始所有点 $1$ 到 $n$ 再加上所有的修改操作连在后面,排成一排看成一个数组

从 $n+1$ 开始从左到右枚举数组,一旦枚举到位置 $i$ 为 $x$ 意思就是这里修改了一次 $x$,设 $x$ 上一次出现的位置为 $las$

那么显然只有数组里 $las+1$ 到 $i$ 的点的如果和 $x$ 有边,那么才会产生 $1$ 代价,因为所有的 $[las+1,i]$ 在数组里排成了连续的一排

(就是 $[1,5],[6,10],[11,30]...$这样的),所以对于节点 $x$ 最终均摊复杂度最多就是 $O(n)$ 的

又因为出度大于 $\sqrt {m}$ 的点不会超过 $\sqrt {m}$ 个,所以对于出度大于 $\sqrt {m}$ 的点,总的复杂度也不超过 $n \sqrt {m}$

那么综合一下得到总复杂度就是 $n \sqrt {n}$ (这里 $n,m,q$ 同阶)

证明 $2$:

不妨让值小的点指向值更大的点

考虑把图中的节点按照度数从大到小排成一排(注意这里是度数不是出度,所以之后图不会改变)

那么首先对于每个节点 $x$ ,$x$ 左边的并且和 $x$ 有边相连的点不超过 $\sqrt {2m}$

因为如果超过 $\sqrt {2m}$ ,那么首先 $x$ 的度数就超过 $\sqrt {2m}$ 并且左边点数也大于等于 $\sqrt {2m}$,又因为左边的点度数都比 $x$ 大

那么总度数显然超过 $2m$ ,所以这种情况一定不可能出现

定义一张图的势能为图中所有点指向右边的点的边的数量,首先初始图的势能最大为 $m$

由之前的证明可以得到每次操作完一个点 $x$ 以后,$x$ 的左边最多有 $\sqrt {2m}$ 条边变成指向右边,并且剩下的边( $x$ 指向右边的边)都变成指向左边,所以每一次操作最多使图的势能增加 $\sqrt {2m}$ ,那么最终增加的势能的总和最多为 $m+q\sqrt {2m}$

那么首先把边从指向左边翻转到指向右边(即增加势能)最多就为 $m+q\sqrt {2m}$

又发现如果我们要把边从指向右边变成指向左边,那么就一定要消耗图中的势能,又因为势能总增加量最多为 $m+q\sqrt {2m}$ 所以最多消耗的势能也就是 $m+q\sqrt {2m}$

所以翻转边的总复杂度为 $O(m+q\sqrt {2m})$

证明完毕

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<vector>
using namespace std;
typedef long long ll;
inline int read()
{
int x=,f=; char ch=getchar();
while(ch<''||ch>'') { if(ch=='-') f=-; ch=getchar(); }
while(ch>=''&&ch<='') { x=(x<<)+(x<<)+(ch^); ch=getchar(); }
return x*f;
}
const int N=2e5+;
int n,m,du[N],out[N];
ll Ans;
vector <int> V[N];
inline ll calc(int i) { return 1ll*out[i]*(du[i]-out[i]); }
int main()
{
n=read(),m=read(); int a,b;
for(int i=;i<=m;i++)
{
a=read(),b=read(),du[a]++,du[b]++;
if(a>b) swap(a,b);
V[a].push_back(b); out[a]++;
}
for(int i=;i<=n;i++) Ans+=calc(i);
printf("%lld\n",Ans);
int Q=read();
for(int i=;i<=Q;i++)
{
a=read(); int len=V[a].size();
for(int j=;j<=len;j++)
{
b=V[a][len-j]; V[a].pop_back();
V[b].push_back(a);
Ans-=calc(a)+calc(b);
out[a]--; out[b]++;
Ans+=calc(a)+calc(b);
}
printf("%lld\n",Ans);
}
return ;
}

Codeforces 1229C. Konrad and Company Evaluation的更多相关文章

  1. Konrad and Company Evaluation

    F. Konrad and Company Evaluation 参考:[codeforces 1230F]Konrad and Company Evaluation-暴力 思路:题意分析见参考博客. ...

  2. Codeforces Round #588 (Div. 1) C. Konrad and Company Evaluation

    直接建反边暴力 复杂度分析见https://blog.csdn.net/Izumi_Hanako/article/details/101267502 #include<bits/stdc++.h ...

  3. 【CF1210D】Konrad and Company Evaluation(vector,图论)

    题意:有i个人,m对两两之间的关系,第i个人初始的薪水为i,有q次操作,第i次操作会把v[i]号的薪水提升成n+i 如果两个人之间存在关系,薪水高的会向薪水低的炫耀 定义u,v,w为一个三元组,当u向 ...

  4. codeforces 566D D. Restructuring Company(并查集)

    题目链接: D. Restructuring Company time limit per test 2 seconds memory limit per test 256 megabytes inp ...

  5. 【codeforces 794C】Naming Company

    [题目链接]:http://codeforces.com/contest/794/problem/C [题意] 有n个位置; 两个人; 每个人都有n个字符组成的集合s1,s2(可以有重复元素); 然后 ...

  6. codeforces 794 C. Naming Company(贪心)

    题目链接:http://codeforces.com/contest/794/problem/C 题意:有两个人每个人都有一个长度为n的字符串,两人轮流拿出一个字符串,放在一个长度为n的字符串的指定位 ...

  7. Codeforces 566 D. Restructuring Company

    Description 一开始有 \(n\) 个元素,可以进行几个操作. 合并 \(x,y\) . 合并 \(x,x+1,...,y\) . 询问 \(x,y\) 是否在一个集合中. Sol 并查集+ ...

  8. Codeforces Round #588 (Div. 1) 简要题解

    1. 1229A Marcin and Training Camp 大意: 给定$n$个对$(a_i,b_i)$, 要求选出一个集合, 使得不存在一个元素好于集合中其他所有元素. 若$a_i$的二进制 ...

  9. Codeforces Round #588 (Div. 2)

    传送门 A. Dawid and Bags of Candies 乱搞. Code #include <bits/stdc++.h> #define MP make_pair #defin ...

随机推荐

  1. RTS打卡计划第四周

    Algorithms: https://leetcode-cn.com/problems/subarray-sum-equals-k/comments/ 此问题开始考虑空间换时间,结果完全不用空间,不 ...

  2. excel怎么只打印某页?excel怎么只打印某几页

    有时候我们需要打印的excel文件,内容较多有好几页,而我们只需要打印里面的部分内容,为了减少纸张.碳粉的浪费,我们怎样精准打印某页或某几页呢?   工具/原料   Excel / WPS软件 方法/ ...

  3. 数据库开源框架之ormlite

    主页: http://ormlite.com/ 配置: 添加以下依赖 * compile 'com.j256.ormlite:ormlite-android:4.48' * compile 'com. ...

  4. 六十一:Flask.Session之flask操作session

    1.设置session:使用flask.session就可以操作字典,操作方式和操作字典一样:session['key']=value2.获取session,和获取字典的值一样:session['ke ...

  5. kubernetes学习:CKA考试认证

    考点 CKA认证针对考核成为当业界的Kubernetes管理员所需的技能. CKA认证考试包括这些一般领域及其在考试中的权重: 应用程序生命周期管理 -  8% 安装.配置和验证 -  12% 核心概 ...

  6. delphi raise 语句: 抛出异常

    //例1:begin  raise Exception.Create('抛出异常');end;//例2:begin  raise Exception.CreateFmt('%s %d', ['错误代码 ...

  7. 异常之【You have an error in your SQL syntax】

    异常如下: ### Error updating database.  Cause: com.mysql.jdbc.exceptions.jdbc4.MySQLSyntaxErrorException ...

  8. java:Echarts,POI

    1.Echarts: demo.js: function demo(selector){ var myEcharts=echarts.init(selector); var option = { ti ...

  9. 分布式架构-Redis 从入门到精通 完整案例 附源码

    导读 篇幅较长,干货十足,阅读需要花点时间,全部手打出来的字,难免出现错别字,敬请谅解.珍惜原创,转载请注明出处,谢谢~! NoSql介绍与Redis介绍 什么是Redis? Redis是用C语言开发 ...

  10. 微信小程序页面阻止默认滑动事件

    在页面上要加入一个悬浮的按钮,这个按钮需要可以拖动,在元素中使用catchtouchstart,catchtouchmove,catchtouchend来控制悬浮按钮的拖动,但是在ios系统中,微信小 ...