由于每一个操作的逆操作都存在,可以看作将$a_{i}$全部变为0的代价

先考虑第一个问题,即对于确定的$a_{i}$如何处理

如果仅能用第2种操作,定义点$i$的代价为以$i$为左端点或以$i-1$为右端点的的操作数,考虑一个代价的意义,即改变$i-1$和$i$的差值,因此$ans\ge C\sum_{i=0}^{n}\frac{|a_{i}-a_{i+1}|}{2}$(每一个操作会被算两次)

(对应的方案只要确保每一次操作减少两对“相邻两数的差值”)

加入第1种操作,由于操作与顺序无关,不妨先使用第1种操作、后使用第2种操作,假设第1种操作产生的序列为$b_{i}$,则答案为$C\sum_{i=0}^{n}|b_{i}-b_{i+1}|+\sum_{i=1}^{n}|a_{i}-b_{i}|$(换言之,即找到任意$b_{i}$,最小化该值)

令$f[i][j]$表示前$i$个数$b_{i}=j$的最小值,转移为$f[i][j]=\min(f[i-1][k]+|k-j|\times C)+2|a_{i}-j|$(这里都乘了2避免小数)

归纳$f[i][j]$具有凸性,转移可以看作找到斜率绝对值大于等于$C$的部分(由于凸性,斜率单调递增,即前后两个区间),将这一部分斜率对$\pm C$取max或min

同时,这也就证明了忽略$2|a_{i}-j|$后$f[i][j]$也具有凸性,而后者具有凸性,因此和也有凸性

我们直接维护斜率,更严谨的,令$f'[i][j]=f[i][j]-f[i][j-1]$,每一次转移相当于执行以下操作:1.将所有数对$-C$取max,对$C$取min;2.对$[1,a_{i}]$区间减2,对$(a_{i},+\infty)$区间加2

对于$f[i][0]$直接考虑上面的转移,即$f[i][0]=2a_{i}+\min_{k}f[i-1][k]+kC$

简单化简,有$f[i][0]=2a_{i}+f[i-1][0]+\min_{k}\sum_{j=1}^{k}f'[i-1][j]+C$,而由于$f'[i-1][j]+C$单调递增,后者也可以看作$\sum_{j}\min(f'[i-1][j]+C,0)$

答案即求$f[n+1][0]$,可以看作不断累加,即要求出每一次后面的式子对答案的贡献

由于位置之间相互独立,可以单独去算每一个数的贡献,先对区间离散化(对于$(a_{i-1},a_{i}]$这个区间的贡献显然都相同),再用$dp[i][j]$表示经过前$i$个点后值为$j$的方案数,复杂度为$o(n^{2}k^{2}C)$

再进一步的,只关心$a_{i}$与枚举的位置的关系,因此可以降为$o(n^{2}kC)$

 1 #include<bits/stdc++.h>
2 using namespace std;
3 #define N 105
4 #define mod 1000000007
5 vector<pair<int,int> >v;
6 int n,c,k,x,ans,mi[N],sz[N],sf[N],f[N][N];
7 int main(){
8 scanf("%d%d%d",&n,&c,&k);
9 mi[0]=1;
10 for(int i=1;i<=n;i++)mi[i]=1LL*mi[i-1]*k%mod;
11 for(int i=1;i<=n;i++)
12 for(int j=1;j<=k;j++){
13 scanf("%d",&x);
14 v.push_back(make_pair(x,i));
15 ans=(ans+x)%mod;
16 }
17 ans=2LL*ans*mi[n-1]%mod;
18 sort(v.begin(),v.end());
19 for(int i=1;i<=n;i++)sf[i]=k;
20 for(int i=-1,nex=0;i<(int)v.size();i=nex){
21 while ((i>=0)&&(nex<v.size())&&(v[nex].first==v[i].first)){
22 sz[v[nex].second]++;
23 sf[v[nex++].second]--;
24 }
25 x=v[nex].first;
26 if (i>=0)x-=v[i].first;
27 memset(f,0,sizeof(f));
28 f[0][2*c+2]=1;
29 for(int j=0;j<=n;j++)
30 for(int l=0;l<=2*c+4;l++){
31 if (l<2)ans=(ans+1LL*x*(mod+l-2)%mod*f[j][l]%mod*mi[n-j])%mod;
32 int ll=min(max(l,2),2*c+2);
33 f[j+1][ll+2]=(f[j+1][ll+2]+1LL*sz[j+1]*f[j][l])%mod;
34 f[j+1][ll-2]=(f[j+1][ll-2]+1LL*sf[j+1]*f[j][l])%mod;
35 }
36 }
37 ans=1LL*ans*(mod+1)/2%mod;
38 printf("%d",ans);
39 }

[atAGC049E]Increment Decrement的更多相关文章

  1. STL——increment/decrement/dereference操作符

    increment/dereference操作符在迭代器的实现上占有非常重要的地位,因为任何一个迭代器都必须实现出前进(increment,operator++)和取值(dereference,ope ...

  2. increment/decrement/dereference

    #include <vector> #include <deque> #include <algorithm> #include <iostream> ...

  3. 【M6】区别increment/decrement操作符的前置(prefix)和后置(postfix)形式

    1.考虑++(--的情况是一样的),前置是累加然后取出,后置是取出然后累加. 2.重载方法根据形参表的不同区分,问题来了,前置和后置形式都没有形参,因此没法区分.怎么办? 对于后置增加一个形参int, ...

  4. increment/decrement/dereference操作符

    标题以上分别对于++/--/* #include <iostream> #include <cstddef> using namespace std; class INT { ...

  5. chrome快捷键

    常用的为: Ctrl + ]   下一个面板 Ctrl + [   上一个面板 Ctrl + Alt + [ 去上一个历史记录的面板Ctrl + Alt + ]      去下一个历史记录的面板 Ct ...

  6. vue.js组件化开发实践

    前言 公司目前制作一个H5活动,特别是有一定统一结构的活动,都要码一个重复的轮子.后来接到一个基于模板的活动设计系统的需求,便有了下面的内容.借油开车. 组件化 需求一到,接就是怎么实现,技术选型自然 ...

  7. Redis 对比 Memcached 并在 CentOS 下进行安装配置

    了解一下 Redis Redis 是一个开源.支持网络.基于内存.键值对的 Key-Value 数据库,使用 ANSI C 编写,并提供多种语言的 API ,它几乎没有上手难度,只需要几分钟我们就能完 ...

  8. 移动web 公用样式

    /*! normalize.css v3.0.2 | MIT License | github.com/necolas/normalize */ /** * 1. Set default font f ...

  9. Mifare系列5-存储结构(转)

    文/闫鑫原创转载请注明出处http://blog.csdn.net/yxstars/article/details/38081521 Mifare S50把1K字节的容量分为16个扇区(Sector0 ...

随机推荐

  1. Sequence Model-week1编程题2-Character level language model【RNN生成恐龙名 LSTM生成莎士比亚风格文字】

    Character level language model - Dinosaurus land 为了构建字符级语言模型来生成新的名称,你的模型将学习不同的名字,并随机生成新的名字. 任务清单: 如何 ...

  2. rabbitmq生产者消息确认

    在使用 RabbitMQ 的时候,有时候当我们生产者发送一条消息到 RabbitMQ 服务器后,我们 生产者想知道消息是否到达了 RabbitMQ 服务器上.这个时候我们应该如何处理? 针对上述问题, ...

  3. echart3 力引导布局实现节点的提示和折叠

    最近在项目中需要开发一个图表来显示人员的各种属性,类似于一种树形的结构进行显示数据.如果多个人员有同一个属性,那么需要将相同的属性进行连线,即关联起来.即形成一个关系图,由于我自身对echarts稍微 ...

  4. 如何洗白xi校长?(初稿)

    看看咱们太子殿下,谁还敢黑全世界最好的太子殿下 我们不如来考虑一下如何给校长洗白. 第一当然是买断热搜了.买断热搜可以阻止消息进一步传播.当然这种操作学校再8月18日晚就已经做过了.8月18日该条消息 ...

  5. C语言中都有哪些常见的数据结构你都知道几个?

    上次在面试时被面试官问到学了哪些数据结构,那时简单答了栈.队列/(ㄒoㄒ)/~~其它就都想不起来了,今天有空整理了一下几种常见的数据结构,原来我们学过的数据结构有这么多~ 首先,先来回顾下C语言中常见 ...

  6. 关于linux的fork的一点学习总结

    最近操作系统的实验要用到fork,于是去搜索了一下资料,很幸运地在博客中找到一篇深度好文: http://blog.csdn.net/jason314/article/details/5640969 ...

  7. 重学STM32---(十)之CAN通信(二)

    目录 前言 程序编写 主代码 测试 前言   CAN协议是非常难的,但是在stm32中却是简单的,只需要我们配置寄存器即可,,,即使这样,我在学习的时候也遇到了许多困难 程序编写 1.开时钟,不用说 ...

  8. 第08课 OpenGL 混合

    混合: 在这一课里,我们在纹理的基础上加上了混合,它看起具有透明的效果,当然解释它不是那么容易,当希望你喜欢它. 简单的透明OpenGL中的绝大多数特效都与某些类型的(色彩)混合有关.混色的定义为,将 ...

  9. Vue3学习(十)之 页面、菜单、路由的使用

    一.前言 好几天没更文了,周末真的太冷了,在家躺了一天不爱动.今天给暖气了,相对不那么冷了,就可以继续更文了. 由文章标题不难看出,就是实现点击菜单跳转的意思,我写的很直白了,哈哈. 二.实现点击菜单 ...

  10. 20191310李烨龙Linux C语言编程基础

    Linux C语言编程基础 任务详情 0. 基于Ubuntu或OpenEuler完成下面的任务(OpenEuler有加分) 1. 选择教材第二章的一节进行编程基础练习(2.10,2.11,2.12,2 ...