(话说其实我想填的是去年CSP的坑...但是貌似有一道题我还不会写咕咕咕...

先写一下这一次的题解吧.
T1:序列.题意省略。

两种操作。这种题要先分析部分分 给出了全部都是2操作的子任务。

发现A 2 B,B 2 C这个时候可以推到 A 2 C也就是所以被2相连的点都存在这种关系。

考虑缩点 把这些点都缩到一起表示他们的权值可以随便传递.

这个时候对于当前子任务我们可以很容易回答,就是看某个集合的权值和是否为0.

考虑有操作1的时候 A 1 B B 1 C 可以发现这可以转换成 A 2 C 转换成这样做之后我们再次把2操作进行缩点。

那么整张图变成了什么?对于1操作直接连边然后看能否缩点 2操作直接缩点 所以整张图变成了若干个联通块 且每个联通块至多只有两个点。

我们简单的分析一下什么时候有解,分类讨论一下即可。

值得一提的是对于1操作变2操作时我们连边需要拿2操作过后缩过的点连边,否则可能会出现(a,b) (c,d) 而c和d之前被缩起来了。
可以证明拿缩过点连边是正确的。(这个地方是坑点 考试的时候这个地方我就没考虑到。
```
const int MAXN=100010;
int n,m,cnt,flag,len,T;
int f[MAXN],vis[MAXN];ll a[MAXN];
int lin[MAXN],ver[MAXN我们很容易就可以得到一个非常暴力的模拟算法。

但是我们想要不去模拟就必须要知道每轮冒泡排序改变的是什么东西.

每一轮冒泡 整个序列中 未到位的最大值一定会交换到后面,这个时候考虑一下这个最大值所处的位置 如果为p的话,其应该到的位置为pos 那么说明p+1~pos这些数字对逆序对的贡献减一 (如果这个数字每次都在排头那么我们就可以不用知道序列是什么样子而直接求出来逆序对的变换情况了。

考虑这个最大值p之前的位置会怎么变换 可以发现还是一个较大值被交换到p+1这个位置然后轮到p开始交换。

至此我们不难发现每次冒泡某个数字之前如果有比其大的数字那么这种数字至少减少1.

设b[i]为i之前有多少个比i大的数字.那么进行k轮 每次每个位置上的b[i]都要减减.

答案=ans-$\sum{min(b[i],k)}$ans为初始序列的逆序对个数.这个显然可以利用线段树维护。

考虑修改 ans修改是O(1)的 b[i]的修改也仅限于一个位置 所以这是一个区间求和单点修改的问题.

树状数组好像需要两个才能维护 直接线段树即可.
```
const int MAXN=200010;
int n,m;
int a[MAXN],b[MAXN];
ll c[MAXN],ans;
inline int ask(int x)
{
int cnt=0;
while(x)
{
cnt+=c[x];
x-=x&(-x);
}
return cnt;
}
inline void add(int x,int y)
{
if(!x)return;
while(x>1;
build(pr)return;
if(l=t[p].r)
{
t[p].add+=k;
t[p].sum+=k*(t[p].r-t[p].l+1);
return;
}
spread(p);
long long mid=(t[p].l+t[p].r)>>1;
if(lmid)change(pr)return 0;
if(l=t[p].r)return t[p].sum;
spread(p);
long long mid=(t[p].l+t[p].r)>>1;
long long ans=0;
if(lmid)ans+=ask(pa[x+1])
{
change(1,b[x+1],b[x+1],-1);
--b[x+1];--ans;
swap(a[x],a[x+1]);
swap(b[x],b[x+1]);
}
else
{
change(1,b[x]+1,b[x]+1,1);
++b[x];++ans;
swap(a[x],a[x+1]);
swap(b[x],b[x+1]);
}
}
else
{
x=min(x,n);
ll w=ask(1,1,x);
printf("%lld\n",ans-w);
}
}
return 0;
}
```
ps:发现这个代码中的线段树长的不太好看了吧,这是很早以前的代码了,考试的时候为了节省时间直接粘了个板子.
T3:最小环.

当时写的时候时间不太够用了 因为T1debug时间过长。。

观察子任务有两个点k==1 简单分析一下什么时候形成的环最大 amax-4,amax-2,amax,amax-1,amax-3...这样排列。

这样计算可以发现能把样例算出来,关于证明:交换任意两个数的位置得到的结果不会比原来更优 所以这样排列就是最优的。

还有一个子任务n为偶数且k==2 我们发现此时形成了两个环。把n分一半套用上述k==1的做法即可解决。

对于$k\leq \frac{n}{2}$我们发现n和k互质的时候也只会形成一个环和k==1一样。

我们只需要知道对于一个k来说最终形成的环的个数有多少个 再把数分开单独计算每一个环即可。

可以发现形成环的个数为 gcd(n,k).

$n\leq 200000$ d(n)最多280..每次暴力计算每个环O(n) 所以总复杂度$d(n)\cdot n$.d(n)表示n的因子个数。
```
const ll MAXN=200010;
ll n,m;
ll a[MAXN],sum;
ll ans[MAXN];
inline ll cmp(ll a,ll b){return a>b;}
inline ll gcd(ll a,ll b){return b?gcd(b,a%b):a;}
int main()
{
//freopen("1.in","r",stdin);
//freopen("ring.out","w",stdout);
n=read();m=read();
for(ll i=1;i

最新文章

  1. Linux上如何查看物理CPU个数,核数,线程数
  2. Java监听器
  3. opengl 3.3 tutorial
  4. Activity中与ListActivity中使用listview区别
  5. GCD求最大公约数
  6. Java基础知识强化89:Date类之Data类概述及其方法
  7. Java数据结构漫谈-ArrayList
  8. ecshop 分页小记
  9. I/O和管道
  10. [Ubuntu 16.04 LTS ]屏幕分辨率 Unknown display
  11. 一脸懵逼学习Storm---(一个开源的分布式实时计算系统)
  12. rsync拉取服务器上的代码到本地
  13. linux上安装mysql,tomcat,jdk
  14. Directory 类
  15. Hive 特殊分隔符处理
  16. 页面显示This is the initial start page for the WebDriver server.的解决办法
  17. go语言最快最好运用最广的web框架比较(大多数人不了解的特性)
  18. WebSocket学习记录
  19. TCP状态迁移
  20. BufferedInputStream使用详解

热门文章

  1. BZOJ2200 道路与航线 题解
  2. 权力越大职责越大——C#中的职责链模式
  3. day32 异常处理、网络编程
  4. shell专题(十):Shell工具(重点)
  5. Mysql基础(十一):流程控制结构、分支结构、循环结构
  6. java 面向对象(四十二):反射(六)反射应用三:调用运行时类的指定结构
  7. java 面向对象(九):类的结构:构造器(一)简介;属性赋值顺序;JavaBean的概念
  8. 用Vue实现一个简单的图片轮播
  9. 尝鲜刚发布的 SpringFox 3.0.0,以前造的轮子可以不用了...
  10. CSS栅格布局