题目描述

小$c$是一名$oier$。最近,他发现他的数据结构好像学傻了。因为他在刷题时碰到了一道傻逼数据结构题,强行使用了平衡树来解决,卡着时间$AC$。为此,他被狠狠地嘲讽了一番。于是,小$c$找了大量的数据结构题来做。

昨天,小$c$正在吃寿司,突然发现许多盘寿司围成了一个圆圈,这些寿司中有红色的也有蓝色的。由于小$c$看交错的颜色非常不爽,想通过一些操作,使得所有的红色寿司形成了一块连续的区域,蓝色的寿司也形成了一块连续的区域。如果小$c$每次只可以交换相邻的两盘寿司,那么最少需要多少步才可以达到小$c$的要求呢?由于他做题做多了,脑袋已经有点不清醒了,于是这个问题就交给你了。


输入格式

第一行一个数$T$,表示数据组数。
接下来$T$行,每行一行由$B$和$R$组成的字符串,$B$表示蓝色,$R$表示红色。第$i$个字符描述顺时针数第$i$盘寿司的颜色。注意,最后一盘寿司和第$1$盘寿司是相邻的。


输出格式

对于每组数据,输出一行表示最小的交换次数。


样例

样例输入

1
BBRBBRBBBRRR

样例输出

5


数据范围与提示

样例说明:

以下说明交换的步骤:
交换位置$2$、位置$3$上的寿司;
交换位置$1$、位置$2$上的寿司;
交换位置$6$、位置$7$上的寿司;
交换位置$7$、位置$8$上的寿司;
交换位置$8$、位置$9$上的寿司;

数据范围:

$T\leqslant 10$

$n\leqslant 1,000,000$


题解

算法一:

爆搜,枚举所有走法。可以获得$0$到$20$分不等。

算法二:

注意到每种状态实际上是可以压缩的。所以搜索的时候对于较小的数据把状态记录下来,用最小表示法去个重,对于稍大的数据可以用$IDA∗$搜索,对每个状态进行估价来进行加速。可以获得$20$到$40$分不等。

算法三:

将环上的问题转化为序列上,我们只需要将环复制一遍即可转化为序列。

那么,考虑这样一个问题,对于最优解,一定存在一个点使得所有的交换都不经过这个点(它的左右不交换),下面简称这个点为断点。

这样我们就可以枚举断点,答案即为把每个$B$挪到两端,反之同理。

然后在进行统计答案,比较每一个断点哪个最优。

现在来简单的证明一下断点的存在,来脑补一个环,环上面有很多$R$,我们肯定要将所有的$R$都撸到一起,而这时候,肯定存在“最下面一个点”作为底端,而在撸的时候肯定不会经过这个点。

时间复杂度:$\Theta (n^2)$。

期望得分:$40$分。

算法四:

考虑决策的单调性问题,上面$\Theta (n^2)$的算法中,我们需要再枚举一个点,这个点左端的$R$都移向左端,右端的都移向右端,那么就满足了一个$\bigcup$型函数。

显然可以进行三分求解。

那么当前枚举的区间的代价怎么求呢?

我们需要开五个数组:

  $\alpha.slft[i]$:表示将点$i$左侧所有的$R$都挪到最左端所要付出的代价,而每个$R$挪到最左端的代价就是它左端的$B$的数量。

  $\beta.srht[i]$:表示将点$i$右侧所有的$R$都挪到最右端所要付出的代价。

  $\chi.blft[i]$:表示点$i$左侧有几个$B$,可以感性的理解为$B$的左缀和。

  $\delta.brht[i]$:表示点$i$右侧有几个$B$,可以感性的理解为$B$的右缀和。

  $\epsilon.rlft[i]$:表示点$i$左侧有几个$R$,可以感性的理解为$R$的前缀和。

下面我只讲当前区间$[L,R]$下,将点$V$左侧的$R$都移到$L$所要付出的代价,反之同理:

  先写出来式子:$slft[V]-slft[L-1]-blft[L-1]\times (rlft[V]-rlft[L-1])$。

  来解释一下式子:

  $slft[V]-slft[L-1]$的含义为将$[L,V]$区间中所有的$R$移到最左端所要付出的代价,不过显然我们并不用把他们都移到左端,而是将其移动到$L$,那么我们不需要付出的代价就为$1$到$L$之间的$B$乘上$L$到$V$之间的$R$,即为$blft[L-1]\times (rlft[V]-rlft[L-1])$。

时间复杂度:$\Theta (n\log_{\frac{3}{2}}n)$。

期望得分:$65$分。

算法五:

我们还会发现这样一个问题,在断点不断右移的时候,当前状态的最有决策点肯定在上衣状态的最优决策点的右侧,这样复杂度就成了线性的了。

时间复杂度:$\Theta (n)$。

期望得分:$100$分。


代码时刻

$\Theta (n\log_{\frac{3}{2}}n)$算法:

  1. #include<bits/stdc++.h>
  2. using namespace std;
  3. int n;
  4. char ch[4000001];
  5. long long slft[4000001],srht[4000001],blft[4000001],brht[4000001],rlft[4000001];
  6. long long ans;
  7. long long minn(int x,int y,int z){return min(x,min(y,z));}
  8. long long judge(int l,int v,int r)
  9. {return slft[v]-slft[l-1]-blft[l-1]*(rlft[v]-rlft[l-1])+srht[v+1]-srht[r+1]-brht[r+1]*(rlft[r]-rlft[v]);}//计算答案
  10. long long _doudou(int l,int r)//三分
  11. {
  12. int flagl=l,flagr=r;
  13. while(r-l>2)
  14. {
  15. int midl=(r-l+1)/3+l;
  16. int midr=(r-l+1)*2/3+l;
  17. if(judge(flagl,midl,flagr)>judge(flagl,midr,flagr))l=midl;
  18. else r=midr;
  19. }
  20. return minn(judge(flagl,l+1,flagr),judge(flagl,l,flagr),judge(flagl,r,flagr));
  21. }
  22. void pre_work()//清空
  23. {
  24. for(int i=1;i<=n;i++)ch[n+i]=ch[i];
  25. blft[0]=slft[0]=rlft[0]=srht[2*n+1]=brht[2*n+1]=0;
  26. }
  27. int main()
  28. {
  29. int T;
  30. scanf("%d",&T);
  31. while(T--)
  32. {
  33. scanf("%s",ch+1);
  34. ans=200209230020020923;
  35. n=strlen(ch+1);
  36. pre_work();
  37. for(int i=1;i<=2*n;i++)
  38. {
  39. blft[i]=blft[i-1];
  40. slft[i]=slft[i-1];
  41. rlft[i]=rlft[i-1];
  42. if(ch[i]=='B')blft[i]++;
  43. else
  44. {
  45. rlft[i]++;
  46. slft[i]+=blft[i];
  47. }
  48. }
  49. for(int i=2*n;i;i--)
  50. {
  51. brht[i]=brht[i+1];
  52. srht[i]=srht[i+1];
  53. if(ch[i]=='B')brht[i]++;
  54. else srht[i]+=brht[i];
  55. }
  56. for(int i=1;i<=n;i++)
  57. ans=min(ans,_doudou(i,i+n-1));
  58. printf("%lld\n",ans);
  59. }
  60. return 0;
  61. }

$\Theta (n)$算法:

  1. #include<bits/stdc++.h>
  2. using namespace std;
  3. int n;
  4. char ch[4000001];
  5. long long slft[4000001],srht[4000001],blft[4000001],brht[4000001],rlft[4000001];
  6. long long ans;
  7. long long judge(int l,int v,int r)
  8. {return slft[v]-slft[l-1]-blft[l-1]*(rlft[v]-rlft[l-1])+srht[v+1]-srht[r+1]-brht[r+1]*(rlft[r]-rlft[v]);}
  9. void pre_work()
  10. {
  11. for(int i=1;i<=n;i++)ch[n+i]=ch[i];
  12. blft[0]=slft[0]=rlft[0]=srht[2*n+1]=brht[2*n+1]=0;
  13. }
  14. int main()
  15. {
  16. int T;
  17. scanf("%d",&T);
  18. while(T--)
  19. {
  20. scanf("%s",ch+1);
  21. ans=200209230020020923;
  22. n=strlen(ch+1);
  23. pre_work();
  24. for(int i=1;i<=2*n;i++)
  25. {
  26. blft[i]=blft[i-1];
  27. slft[i]=slft[i-1];
  28. rlft[i]=rlft[i-1];
  29. if(ch[i]=='B')blft[i]++;
  30. else
  31. {
  32. rlft[i]++;
  33. slft[i]+=blft[i];
  34. }
  35. }
  36. for(int i=2*n;i;i--)
  37. {
  38. brht[i]=brht[i+1];
  39. srht[i]=srht[i+1];
  40. if(ch[i]=='B')brht[i]++;
  41. else srht[i]+=brht[i];
  42. }
  43. int lft=1;
  44. for(int i=1;i<=n;i++)
  45. {
  46. while(lft!=n+i&&judge(i,lft,i+n-1)>=judge(i,lft+1,i+n-1))lft++;
  47. ans=min(ans,judge(i,lft,i+n-1));
  48. }
  49. printf("%lld\n",ans);
  50. }
  51. return 0;
  52. }

rp++

[CSP-S模拟测试]:寿司(暴力)的更多相关文章

  1. [2018冬令营模拟测试赛(二十一)]Problem A: Decalcomania

    [2018冬令营模拟测试赛(二十一)]Problem A: Decalcomania 试题描述 输入 见"试题描述" 输出 见"试题描述" 输入示例 见&quo ...

  2. [考试反思]1003csp-s模拟测试58:沉淀

    稳住阵脚. 还可以. 至少想拿到的分都拿到了,最后一题的确因为不会按秩合并和线段树分治而想不出来. 对拍了,暴力都拍了.挺稳的. 但是其实也有波折,险些被卡内存. 如果内存使用不连续或申请的内存全部使 ...

  3. [考试反思]0814NOIP模拟测试21

    前两名是外校的240.220.kx和skyh拿到了190的[暴力打满]的好成绩. 我第5是170分,然而160分就是第19了. 在前一晚上刚刚爆炸完毕后,心态格外平稳. 想想前一天晚上的挣扎: 啊啊啊 ...

  4. [考试反思]0729NOIP模拟测试10

    安度因:哇哦. 安度因:谢谢你. 第三个rank1不知为什么就来了.迷之二连?也不知道哪里来的rp 连续两次考试数学都占了比较大的比重,所以我非常幸运的得以发挥我的优势(也许是优势吧,反正数学里基本没 ...

  5. csp-s模拟测试98

    csp-s模拟测试98 $T1$??不是我吹我轻松手玩20*20.$T2$装鸭好像挺可做?$T3$性质数据挺多提示很明显? $One$ $Hour$ $Later$ 这$T1$什么傻逼题真$jb$难调 ...

  6. csp-s模拟测试96

    csp-s模拟测试96 $T1$一眼慢速乘,$T2$稍证一手最优性尝试用神奇数据结构优化,无果,弃.$T3$暴力+信仰. 100 03:16:38 95 03:16:56 35 03:17:10 23 ...

  7. csp-s模拟测试94

    csp-s模拟测试94 一场简单题,打爆了.$T1$脑抽分解质因数准备分子分母消,想了半天发现$jb$互质直接上天,果断码了高精滚蛋.$T2$无脑手玩大样例,突然灵光一闪想到映射到前$K$大小的区间, ...

  8. csp-s模拟测试93

    csp-s模拟测试93 自闭场. $T1$想到$CDQ$,因为复杂度少看见一个$0$打了半年还用了$sort$直接废掉,$T2$,$T3$直接自闭暴力分都没有.考场太慌了,心态不好. 02:07:34 ...

  9. csp-s模拟测试92

    csp-s模拟测试92 关于$T1$:最短路这一定建边最短路. 关于$T2$:傻逼$Dp$这一定线段树优化$Dp$. 关于$T3$:最小生成树+树P+换跟一定是这样. 深入(?)思考$T1$:我是傻逼 ...

随机推荐

  1. UrlConnection发送http请求 中文乱码解决

    中文乱码 DataOutputStream dos = new DataOutputStream(httpConn.getOutputStream()); //dos.writeBytes(jsonD ...

  2. new 和 malloc 的区别 及使用

    Malloc: 定义上:malloc  memory allocation 动态内存分配 是c中的一个函数 使用方法: extern void *malloc(unsigned int num_byt ...

  3. [转帖]Oracle dba_objects和all_objects 最大的区别

    Oracle dba_objects和all_objects 最大的区别 原创 Oracle 作者:maohaiqing0304 时间:2015-08-14 15:07:18  9281  0   链 ...

  4. C++基础-类和对象

    本文为 C++ 学习笔记,参考<Sams Teach Yourself C++ in One Hour a Day>第 8 版.<C++ Primer>第 5 版.<代码 ...

  5. 洛谷 P1541 乌龟棋 & [NOIP2010提高组](dp)

    传送门 解题思路 一道裸的dp. 用dp[i][j][k][kk]表示用i个1步,j个2步,k个3步,kk个4步所获得的最大价值,然后状态转移方程就要分情况讨论了(详见代码) 然后就是一开始统计一下几 ...

  6. [Codeforces 1246B] Power Products (STL+分解质因数)

    [Codeforces 1246B] Power Products (STL+分解质因数) 题面 给出一个长度为\(n\)的序列\(a_i\)和常数k,求有多少个数对\((i,j)\)满足\(a_i ...

  7. npm publish 失败可能的原因记录

    npm 发布个人包时,遇到不少坑,总结如下(可能不全): 1.npm版本过低,处理:npm install -g npm update 2.可能权限原因,处理:npm publish --access ...

  8. PCIeのType0与Type1型配置请求与BAR(基地址寄存器)

    PCIe中存在两种配置空间Type0&type1,TYPE0对应非桥设备(Endpoint),Type1对应桥设备(Root和Switch端口中的P2P桥)因为Root每个端口总都含有一个P2 ...

  9. Hive常用非交互式命令

    [hadoop@hadoop hive-0.13.1]$ bin/hive -help usage: hive -d,--define <key=value> Variable subsi ...

  10. VB里面的字体颜色

    颜色常数颜色常数 值 描述vbBlack &H0 黑色vbRed &HFF 红色vbGreen &HFF00 绿色vbYellow &HFFFF 黄色vbBlue &a ...