题目大意:

给你两个数n,k求n的全排列的第k小,有多少满足如下条件的数:

首先定义一个幸运数字:只由4和7构成

对于排列p[i]满足i和p[i]都是幸运数字

思路:

对于n,k<=1e9

一眼逆康托展开

什么?你不知到康托展开?

点这里点这里点这里

由于阶乘的增长是非常快的

13的阶乘就大于1e9了

所以说:

对于一个n的权排列 1  2 3 4 ...... n

我们最多动他的后13位就可以得到第k小的排列

我们称之为动n的后x位可以得到第k小的排列(如果这里都取13的话,有的序列是n<13的,会越界)

然后我们对[1,n-x]中的数字统计答案的时候可以数位dp,可以dfs

然后对后x位统计答案,就是裸的逆康托展开了

Code:

ll n, k, x, fac[20], ans;
std::vector<ll> v;
ll suf[20], cnt;
void dfs(ll num, ll top)
{
// cout<<top<<endl;
if(num > top) return ;
if(num <= top && num != 0)ans++;
dfs(num * 10 + 4, top);
dfs(num * 10 + 7, top);
}
int ok(ll x)
{
int flag = 1;
while(x)
{
int yy = x % 10;
// cout<<yy<<"@"<<endl;
if(yy != 4 && yy != 7) flag = 0;
x /= 10;
}
return flag;
}
void re_count()
{
sort(v.begin(), v.end());
for(int i = x ; i >= 1 ; i--)
{
ll pos = k / fac[i - 1];
k = k % fac[i - 1];
suf[++cnt] = v[pos];
v.erase(v.begin() + pos);
} }
int main()
{
fac[0] = 1;
rep(i, 1, 16) fac[i] = i * fac[i - 1];
n = read(), k = read();
k--;
for(int i = 1 ; i <= 16; i++)
{
if(fac[i] >k)
{
x = i;
break;
}
}
// cout<<x<<"#"<<endl;
for(int i = n; i >= n - x + 1; i--) v.push_back(i);
if(n - x < 0)
{
cout << -1;
return 0;
}
dfs(0, n - x); //搜出n-x的幸运数
re_count();
for(int i = n - x + 1; i <= n; i++) if(ok(i) && ok(suf[i - (n - x)])) ans++;
out(ans);
return 0;
}

Codeforces-121C(逆康托展开)的更多相关文章

  1. LightOJ1060 nth Permutation(不重复全排列+逆康托展开)

    一年多前遇到差不多的题目http://acm.fafu.edu.cn/problem.php?id=1427. 一开始我还用搜索..后来那时意外找到一个不重复全排列的计算公式:M!/(N1!*N2!* ...

  2. nyoj 139——我排第几个|| nyoj 143——第几是谁? 康托展开与逆康托展开

    讲解康托展开与逆康托展开.http://wenku.baidu.com/view/55ebccee4afe04a1b071deaf.html #include<bits/stdc++.h> ...

  3. 题解报告:NYOJ 题目143 第几是谁?(逆康托展开)

    描述 现在有"abcdefghijkl”12个字符,将其按字典序排列,如果给出任意一种排列,我们能说出这个排列在所有的排列中是第几小的.但是现在我们给出它是第几小,需要你求出它所代表的序列. ...

  4. HDU1027 Ignatius and the Princess II( 逆康托展开 )

    链接:传送门 题意:给出一个 n ,求 1 - n 全排列的第 m 个排列情况 思路:经典逆康托展开,需要注意的时要在原来逆康托展开的模板上改动一些地方. 分析:已知 1 <= M <= ...

  5. 康托展开&逆康托展开学习笔记

    啊...好久没写了...可能是最后一篇学习笔记了吧 题目大意:给定序列求其在全排列中的排名&&给定排名求排列. 这就是康托展开&&逆康托展开要干的事了.下面依次介绍 一 ...

  6. hdoj 1027 Ignatius and the Princess II 【逆康托展开】

    Ignatius and the Princess II Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K ( ...

  7. 康托展开与逆康托展开模板(O(n^2)/O(nlogn))

    O(n2)方法: namespace Cantor { ; int fac[N]; void init() { fac[]=; ; i<N; ++i)fac[i]=fac[i-]*i; } in ...

  8. DeCantor Expansion (逆康托展开)

    Background\text{Background}Background The \text{The }The Listen&Say Test will be hold on May 11, ...

  9. cf121C. Lucky Permutation(康托展开)

    题意 题目链接 Sol 由于阶乘的数量增长非常迅速,而\(k\)又非常小,那么显然最后的序列只有最后几位会发生改变. 前面的位置都是\(i = a[i]\).那么前面的可以直接数位dp/爆搜,后面的部 ...

随机推荐

  1. Upcoming Browser Behavior Changes & Chrome & SameSite

    Upcoming Browser Behavior Changes & Chrome & SameSite Chrome 80 https://auth0.com/blog/brows ...

  2. useful podcast

    useful podcast front end podcast https://shoptalkshow.com https://stackoverflow.blog/podcast/ SoundC ...

  3. c++ 使用PID获取可执行文件路径

    注意看备注 https://docs.microsoft.com/en-us/windows/win32/api/psapi/nf-psapi-getmodulefilenameexa #includ ...

  4. c++ 读取ASCII

    void ReadASCII(BYTE* addr, size_t offset, char r[]) { size_t i = 0; char c; while (true) { c = *(add ...

  5. uniapp 发起网络请求

    推荐下我写的uni-http 创建http-config.js import Vue from 'vue' const BASE_URL = 'http://xxx.com'; if (process ...

  6. VAST维萨币二月发行,高倍币重现江湖!

    市场长期的历史经验表明,经营盈利能力最好的企业,经常是那些现在的经营方式与5年前甚至10年前几乎完全相同的企业.这个经营模式放到币圈也是一样的,2020年的挖矿是最火的,这个模式现在在市场也同样受用. ...

  7. 创建gitHub账户并配置秘钥

    1. 登录注册地址 https://github.com/ 2.点击注册 Sign up 3.输入邮箱 密码 进行注册 4.注册成功后,登录邮箱验证 .然后通过邮箱和密码登录gitHub.设置 set ...

  8. Docker中配置MySQL并实现远程访问

    Docker配置MySQL容器 拉取MySQL镜像 docker pull mysql:5.6 有可能会因为网络问题失败,重复尝试. 创建容器 docker run -d --name selfdef ...

  9. git配置了公钥,在下载项目时为什么还要输入密码

    配置git地址:https://www.cnblogs.com/lz0925/p/10794616.html 原文链接:https://blog.csdn.net/xiaomengzi_16/arti ...

  10. 【Notes_8】现代图形学入门——几何(基本表示方法、曲线与曲面)

    几何 几何表示 隐式表示 不给出点的坐标,给数学表达式 优点 可以很容易找到点与几何之间的关系 缺点 找某特定的点很难 更多的隐式表示方法 Constructive Solid Geometry .D ...