题面

A

N

T

O

N

\rm ANTON

ANTON 的基因由

A

,

N

,

T

,

O

\rm A,N,T,O

A,N,T,O 四种碱基排列组成。

A

N

T

O

N

\rm ANTON

ANTON 的身体很智能,一旦发现基因序列被改变了,就会通过尽量少的次数交换相邻两个碱基,来还原回开始的基因序列。

现在给出

A

N

T

O

N

\rm ANTON

ANTON 的基因序列,长为

N

N

N ,求一个打乱后的该序列,满足还原时需要交换操作最少次数最多,输出任意一个符合条件的序列。

1

N

1

0

5

1\leq N \leq10^5

1≤N≤105.

题解

我们先考虑最少操作次数怎么求。根据做题经验,有一个贪心思路:把相同类型的碱基不交叉地一一配对,然后把每个初始位置的目标位置拿出来组成一个序列(因此这应该是个1~N的排列),最小操作次数即该排列的逆序对数。

那么,我们不妨把打乱后的序列也这样一一对应回去,每个位置上记录初始位置的编号。容易发现,对于同一种碱基的所有位置,初始位置的编号一定是单增的。在满足这个条件的情况下,我们可以想想怎么最大化逆序对数。

这里有一个结论:一定存在一种逆序对最多的方案,满足同种碱基都相邻。

  • 证明:我们用调整法,假设序列中有这么一段两边都有个 A(省略号中间没有 A):A......A,前面一个的初始位置为

    i

    i

    i ,后面一个的初始位置为

    j

    j

    j,此时

    i

    <

    j

    i<j

    i<j。令省略号中间比

    i

    i

    i 小的数量为

    a

    a

    a ,比

    j

    j

    j 大的数量为

    b

    b

    b 。如果

    a

    b

    a\geq b

    a≥b,把右边的 A 移到省略号左边,逆序对的变化量为

    a

    b

    0

    a-b\geq0

    a−b≥0 。如果

    a

    <

    b

    a<b

    a<b ,把左边的 A 移到省略号右边,逆序对的变化量为

    b

    a

    >

    0

    b-a>0

    b−a>0。两种使 A 相邻的方法总有一个不会让逆序对数减小,得证。

那么,由于只有四种碱基,全排列有

M

!

=

4

!

=

24

M!=4!=24

M!=4!=24 种,最终的序列也只有这么多种,其中一定存在一个答案序列,我们暴力枚举找它。复杂度

O

(

M

!

n

log

n

)

O(M!n\log n)

O(M!nlogn)。

还有一种

O

(

n

)

O(n)

O(n) 的做法。交换次数不一定要求逆序对,还可以求初始与目标的距离。由于我们知道同种碱基要相邻,可以通过预处理前缀和等方式,快速判断每种碱基段在某个位置对答案贡献为多少,这样一来,枚举全排列时就不用每次

O

(

n

)

O(n)

O(n) 求答案,而是常数时间。总时间复杂度就主要只有输入和预处理了。

CODE

O

(

24

n

log

n

)

O(24n\log n)

O(24nlogn)

#include<set>
#include<queue>
#include<cmath>
#include<vector>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
#define MAXN 400005
#define ENDL putchar('\n')
#define LL long long
#define DB double
#define lowbit(x) ((-x) & (x))
#define INF 0x3f3f3f3f
LL read() {
LL f = 1,x = 0;char s = getchar();
while(s < '0' || s > '9') {if(s=='-')f = -f;s = getchar();}
while(s >= '0' && s <= '9') {x=x*10+(s-'0');s = getchar();}
return f * x;
}
int n,m,i,j,s,o,k;
char s0[10] = "ANTO";
char ss[MAXN];
int c[MAXN];
void addc(int x,int y){while(x<=n)c[x]+=y,x+=lowbit(x);}
int sum(int x){int s=0;while(x>0)s+=c[x],x-=lowbit(x);return s;}
void INIT() {for(int i = 1;i <= n;i ++) c[i] = 0;}
LL as = 0,CN;
char D[MAXN],e[MAXN];
void ADD(int x) {
D[CN --] = ss[x];
as += sum(x-1); addc(x,1);
return ;
}
int main() {
int T = read();
while(T --) {
scanf("%s",ss + 1);
n = strlen(ss + 1);
for(int i = 1;i <= n;i ++) e[i] = ss[i];
LL ans = -1;
for(int a=0;a<4;a++) {
for(int b=0;b<4;b++) if(b!=a) {
for(int c=0;c<4;c++) if(c!=a&&c!=b) {
for(int d=0;d<4;d++) if(d!=a&&d!=b&&d!=c) {
INIT();
CN = n;as = 0;
for(int i = n;i > 0;i --) if(ss[i] == s0[d]) ADD(i);
for(int i = n;i > 0;i --) if(ss[i] == s0[c]) ADD(i);
for(int i = n;i > 0;i --) if(ss[i] == s0[b]) ADD(i);
for(int i = n;i > 0;i --) if(ss[i] == s0[a]) ADD(i);
if(as > ans) {
for(int i = 1;i <= n;i ++) e[i] = D[i];
ans = as;
}
}
}
}
}
for(int i = 1;i <= n;i ++) putchar(e[i]);ENDL;
}
return 0;
}

[CF1526D] Kill Anton(逆序对,搜索)的更多相关文章

  1. 剑指Offer(三十五):数组中的逆序对

    剑指Offer(三十五):数组中的逆序对 搜索微信公众号:'AI-ming3526'或者'计算机视觉这件小事' 获取更多算法.机器学习干货 csdn:https://blog.csdn.net/bai ...

  2. Codeforces Round #329 (Div. 2) B. Anton and Lines 逆序对

    B. Anton and Lines Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/contest/593/pr ...

  3. CF785CAnton and Permutation(分块 动态逆序对)

    Anton likes permutations, especially he likes to permute their elements. Note that a permutation of  ...

  4. 洛谷 P1521 求逆序对

    题目描述 我们说(i,j)是a1,a2,…,aN的一个逆序对当且仅当i<j且ai>a j.例如2,4,1,3,5的逆序对有3个,分别为(1,3),(2,3),(2,4).现在已知N和K,求 ...

  5. leetcode_315_逆序对问题

    题目描述 本题来自于Leetcode的算法题库第315题,具体题目描述如下: 给定一个nums整数数组 ,按要求返回一个counts新数组 .数组 counts 有该性质: counts[i]的值是 ...

  6. 【CQOI2011】动态逆序对 BZOJ3295

    Description 对于序列A,它的逆序对数定义为满足i<j,且Ai>Aj的数对(i,j)的个数.给1到n的一个排列,按照某种顺序依次删除m个元素,你的任务是在每次删除一个元素之前统计 ...

  7. CH Round #72 奇数码问题[逆序对 观察]

    描述 你一定玩过八数码游戏,它实际上是在一个3*3的网格中进行的,1个空格和1~8这8个数字恰好不重不漏地分布在这3*3的网格中. 例如:5 2 81 3 _4 6 7 在游戏过程中,可以把空格与其上 ...

  8. POJ3928Ping pong[树状数组 仿逆序对]

    Ping pong Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 3109   Accepted: 1148 Descrip ...

  9. NOIP2013火柴排队[逆序对]

    题目描述 涵涵有两盒火柴,每盒装有 n 根火柴,每根火柴都有一个高度. 现在将每盒中的火柴各自排成一列, 同一列火柴的高度互不相同, 两列火柴之间的距离定义为: ∑(ai-bi)^2 其中 ai 表示 ...

随机推荐

  1. vue组件data函数

    vue组件data通常定义为一个函数并return一个对象,对象中定义的就是组件数据,当然定义数据还有props.computed等方式. data如果直接定义为对象data: {message: ' ...

  2. 介绍python和库文件管理

    一.Python 特点 1.易于学习:Python有相对较少的关键字,结构简单,和一个明确定义的语法,学习起来更加简单. 2.易于阅读:Python代码定义的更清晰. 3.易于维护:Python的成功 ...

  3. 2021.06.19【NOIP提高B组】模拟 总结

    T1 题意:有 \(n\) 个点,有 \(m\) 条边,每次加入一条到图中 问每个点的度数大于零且都是偶数的子图的个数 考试直接判断两点是否出现,出现则更新 其实只要改成并查集判断即可 原理:其实就是 ...

  4. 使用PowerShell压缩和解压ZIP包

    更新记录 本文迁移自Panda666原博客,原发布时间:2021年7月13日. 解压ZIP包 使用PowerShell的Expand-Archive命令.PowerShell官方文档地址. 命令格式: ...

  5. 浅析DispatchProxy动态代理AOP

    浅析DispatchProxy动态代理AOP(代码源码) 最近学习了一段时间Java,了解到Java实现动态代理AOP主要分为两种方式JDK.CGLIB,我之前使用NET实现AOP切面编程,会用Fil ...

  6. MySQL 千万数据库深分页查询优化,拒绝线上故障!

    文章首发在公众号(龙台的技术笔记),之后同步到博客园和个人网站:xiaomage.info 优化项目代码过程中发现一个千万级数据深分页问题,缘由是这样的 库里有一张耗材 MCS_PROD 表,通过同步 ...

  7. RocketMQ事务消息机制

    1.half消息对消费者不可见,用于确定MQ服务正常. 2.MQ响应half消息. 3.生产者执行本地事务. 4.生产者发送具体消息+本地事务状态,MQ根据本地事务状态执行Commit或者Rollba ...

  8. Leetcode----<Diving Board LCCI>

    题解如下: public class DivingBoardLCCI { /** * 暴力解法,遍历每一种可能性 时间复杂度:O(2*N) * @param shorter * @param long ...

  9. 【RPA之家转载】苏桦 华为RPA 企业财务实践:RPA与AI结合,实现百万级票据、合同处理自动化

    [RPA之家转载]苏桦 华为RPA 企业财务实践:RPA与AI结合,实现百万级票据.合同处理自动化 看到大会的主题,说每一位开发者都了不起,说白了我也非常的感触,因为我自己本身也是一个开发者,我从01 ...

  10. PHP生成图形验证码

    在建站过程中,很多时候都会需要用户验证验证码等操作,比如:注册.登录.发表评论.获取资源等等,一方面可以验证当前用户的行为是否是爬虫.机器人等情况,给网站数据统计产生影响:另一方面可以防止用户大量刷取 ...