嵊州D2T2

八月惊魂

这是一个远古时期的秘密,至今已无人关心。

这个世界的每个时代可以和一个 1 ∼ n 的排列一一对应。

时代越早,所对应的排列字典序就越小。

我们知道,公爵已经是 m 个时代前的人物了。

并且通过翻阅古籍,我们得知了公爵所在时代所对应的排列。

那么我们的时代所对应的排列是什么?

希望以此能寻回我们失落的文明……

Input

第一行一个正整数 n。

第二行一个正整数 m。

第三行 n 个整数,表示公爵所在时代对应的排列。

Output 一行 n 个整数,表示我们所在的时代对应的排列。

Examples

august.in august.out

5

3

1 2 3 4 5

1 2 4 5 3

Notes

对于所有数据,满足 n ≤ 10000 , m ≤ 100。

Task1[30%]

n ≤ 10

Task2[60%]

n ≤ 50

Task3[100%]

无特殊限制


Solve!

#include <cstdio>
#include <algorithm>
using namespace std;
int main(){
//freopen("august.in","r",stdin);
//freopen("august.out","w",stdout);
int n,m; scanf("%d%d",&n,&m);
int line[n];
for(int i=;i<n;i++) scanf("%d",&line[i]);
while(m--) next_permutation(line , line + n);
for(int i=;i<n;i++) {printf("%d ",line[i]);}
return ;
}

这篇博客园,之所以我要先code,是因为我连题目都看不懂

时代越早,所对应的排列字典序就越小。

是吗?字典序是什么呀~

还要一个一个推吗?怎么推?

于是我就搁着了。。。

后来

再看std(题解)

next_permutation(line , line + n);

这是什么函数呀?

原来是STL里面的呀。。。

真的没看过呀!!!o(╥﹏╥)o......

next_permutation全排列函数是一个十分好用而且强大的函数。

要想更好的了解这个函数可以看https://blog.csdn.net/howardemily/article/details/68064377

排列(Arrangement),简单讲是从N个不同元素中取出M个,按照一定顺序排成一列,通常用A(M,N)表示。

当M=N时,称为全排列(Permutation)

从数学角度讲,全排列的个数A(N,N)=(N)*(N-1)*...*2*1=N!,但从编程角度,如何获取所有排列?那么就必须按照某种顺序逐个获得下一个排列,通常按照升序顺序(字典序)获得下一个排列。

例如对于一个集合A={1,2,3,},首先获取全排列a1: 1,2,3,;然后获取下一个排列a2: 1,3,2,;

按此顺序,A的全排列如下:

a1: 1,2,3;  a2: 1,3,2;  a3: 2,1,3;  a4: 2,3,1;  a5: 3,1,2;  a6: 3,2,1;  共6种。

对于给定的任意一种全排列,如果能求出下一个全排列的情况,那么求得所有全排列情况就容易了。

好在STL中的algorithm已经给出了一种健壮、高效的方法,下面进行介绍。

next_permutation()

a)从后向前查找第一个相邻元素对(i,j),并且满足A[i] < A[j]。

易知,此时从j到end必然是降序。可以用反证法证明,请自行证明。

b)在[j,end)中寻找一个最小的k使其满足A[i]<A[k]。

由于[j,end)是降序的,所以必然存在一个k满足上面条件;并且可以从后向前查找第一个满足A[i]<A[k]关系的k,此时的k必是待找的k。

c)将i与k交换。

此时,i处变成比i大的最小元素,因为下一个全排列必须是与当前排列按照升序排序相邻的排列,故选择最小的元素替代i。

易知,交换后的[j,end)仍然满足降序排序。因为在(k,end)中必然小于i,在[j,k)中必然大于k,并且大于i。

d)逆置[j,end)

由于此时[j,end)是降序的,故将其逆置。最终获得下一全排序。

注意:如果在步骤a)找不到符合的相邻元素对,即此时i=begin,则说明当前[begin,end)为一个降序顺序,即无下一个全排列,STL的方法是将其逆置成升序。

next_permutation()拓展

如何获取所有全排列情况?

STL中的代码非常精妙,利用next_permutation的返回值,判断是否全排列结束(否则将死循环)。对于给定的一个数组,打印其所有全排列只需如下:

 void all_permutation(int arr[], int n)
{
sort(arr,arr+n); // 先用sort排序
do{//用do……while保证至少做一遍
for(int i=; i<n; printf("%d ",arr[i++]));
printf("\n");
}while(next_permutation(arr,arr+n));//判断也是计算
}

All Permutation


请循其本

所以,我们的代码呢,只要一行(核心部分)呢!

 #include <cstdio>
#include <algorithm>
using namespace std;
int main(){
//freopen("august.in","r",stdin);
//freopen("august.out","w",stdout);
int n,m; scanf("%d%d",&n,&m);
int line[n];
for(int i=;i<n;i++) scanf("%d",&line[i]);
while(m--) next_permutation(line , line + n);
for(int i=;i<n;i++) {printf("%d ",line[i]);}
return ;
}

别忘记加头文件哈!

#include <algorithm>

OK!

嵊州D2T2 八月惊魂 全排列 next_permutation()的更多相关文章

  1. 全排列next_permutation()用法和构造函数赋值

    全排列next_permutation()用法 在头文件aglorithm里 就是1~n数组的现在的字典序到最大的字典序的依次增加.(最多可以是n!种情况) int a[n]; do{ }while( ...

  2. 关于全排列 next_permutation() 函数的用法

    这是一个c++函数,包含在头文件<algorithm>里面,下面是基本格式. 1 int a[]; 2 do{ 3 4 }while(next_permutation(a,a+n)); 下 ...

  3. 全排列 ( next_permutation)

    SLT: C++的STL有一个函数可以方便地生成全排列,这就是next_permutation 在C++ Reference中查看了一下next_permutation的函数声明: #include ...

  4. 全排列 next_permutation 用法

    给一个正整数n,让你求它的全排列 先介绍一个函数,iota(a,a+n,1) 用法就是把a数组的第0位到第n-1位依次赋为1,2,.....n: 然后是next_permutation(a,a+4)函 ...

  5. 排列2(全排列next_permutation 注意格式)

    排列2 Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submiss ...

  6. 蓝桥杯 方格填数 DFS 全排列 next_permutation用法

    如下的10个格子(参看[图1.jpg]) 填入0~9的数字.要求:连续的两个数字不能相邻.(左右.上下.对角都算相邻) 一共有多少种可能的填数方案? 请填写表示方案数目的整数.注意:你提交的应该是一个 ...

  7. 全排列 next_permutation() 函数的用法

    在头文件<algorithm>里面有如下代码: int a[]; do { } while(next_permutation(a,a+n)); 可产生1~n的全排列有如下代码: #incl ...

  8. 全排列 next_permutation() 函数的使用

    看来看去还是这篇博客比较简洁明了 https://www.cnblogs.com/My-Sunshine/p/4985366.html 顺便给出牛客网的一道题,虽然这道题用dfs写出全排列也能做,题意 ...

  9. STL中关于全排列next_permutation以及prev_permutation的用法

    这两个函数都包含在algorithm库中.STL提供了两个用来计算排列组合关系的算法,分别是next_permutation和prev_permutation. 一.函数原型 首先我们来看看这两个函数 ...

随机推荐

  1. Linux IO模式

    原文地址:https://segmentfault.com/a/1190000003063859 同步IO和异步IO,阻塞IO和非阻塞IO分别是什么,到底有什么区别?不同的人在不同的上下文下给出的答案 ...

  2. SQL logic error or missing database no such table: xxx

    原文:SQL logic error or missing database no such table: xxx System.Data.SQLite.SQLiteException (0x8000 ...

  3. Debian下编译安装驱动模块

    在Linux下,我们常有需要自己来编译安装驱动模块的情况,例如要安装显卡驱动,要安装无线驱动,有的时候某个程序的安装使用与需要安装相应的驱动模块.Debian Linux下的生活本已十分简单,使用ap ...

  4. Upgrade a Non-CDB To a PDB on CDB

    .Stop the cluster database and start database on one node with read noly [oracle@raca1 admin]$ srvct ...

  5. ps 专题

    ps p 22763  -L -o pcpu,pid,tid,time,tname,cmd,pmem,rss --sort rss  按rss排序 ps p 26653 -L -o pcpu,tid ...

  6. 卸载win10内置windows app的方法

    原文:卸载win10内置windows app的方法 2015年,微软推出了windows10操作系统,其以漂亮的界面.良好的操作方式.方便的推送升级迅速获得了好多人的好评,因此,好多同学都换了win ...

  7. 我写的一个Qt 显示二维码( QR Code)的控件(可以去掉对 libpthread 的依赖,而且编译出的库文件可以在 vc2010 的release 模式下使用)

    最近一个项目需要显示二维码,所以花了点时间(只用了一个晚上,写的很不完善),写了个显示二维码的控件.当然这个控件用到了些开源的代码,比如qrencode,所以我也打算把我的代码开源. 我的代码参考了 ...

  8. Docker Explanation and Apache Image

    https://blog.sajjan.com.np/2017/02/05/docker-getting-started-containers-ubuntu/ https://blog.sajjan. ...

  9. 很幽默的讲解六种Socket IO模型 Delphi版本(自己Select查看,WM_SOCKET消息通知,WSAEventSelect自动收取,Overlapped I/O 事件通知模型,Overlapped I/O 完成例程模型,IOCP模型机器人)

    很幽默的讲解六种Socket IO模型(转)本文简单介绍了当前Windows支持的各种Socket I/O模型,如果你发现其中存在什么错误请务必赐教. 一:select模型 二:WSAAsyncSel ...

  10. 判断一个窗口是否被挂起(发WM_NULL消息,或者调用IsHungAppWindow API进行测试)

    判断一个窗口是否被挂起了(就是没有响应了),在多窗口编程了经常会用到,在给别的窗口发消息前,为了目的窗口能确定收到消息,常常在之前先检测窗口是否被挂起了,我们以前常用的方式的是使用下面的方法: //  ...