<更新提示>

<第一次更新>


<正文>

Description

给定正整数n,m,问有多少个正整数满足:

(1) 不含前导0;

(2) 是m的倍数;

(3) 可以通过重排列各个数位得到n。

\(n\leq10^{20},m\leq100\)

Input Format

一行两个整数n,m。

Output Format

一行一个整数表示答案对998244353取模的结果。

Sample Input

1 1

Sample Output

1

解析

大致题意:给定一个位数不超过\(20\)位的正整数\(n\),求\(n\)的互异排列中有多少种情况恰为\(m\)的倍数,不能存在前导\(0\)。

假如\(n\leq10^{16},m\leq20\),那显然可以直接状压\(dp\)来计数,设\(f[S][i]\)代表\(n\)中取数情况为\(S\),得到的数模\(m\)余\(i\)的方案总数,可以枚举一个\(j\)来转移,时间复杂度\(O(2^{lg_n}lg_nm)\)。

当\(n\leq10^{20},m\leq100\)时,空间时间就都不行了。其实,在数字\(n\)中,很可能有多个一样的数字,而之前的状态我们把每一位数字都当做不一样的来处理,这样既费空间,又还需要额外判断重复状态。所以我们需要一种既能具体唯一表示状态,又不会有重复的状压方法。

对于这题来说,有一种很合适的状压方法:变进制状压。我们把\(0-9\)每一种数字各用了几次压成一个状态,例如\(0\)出现了一次,那么在状态中代表的权值就是\(1\),同理\(2\)个\(0\)代表的权值就是\(2\),假设有\(cnt_0\)个\(0\),那么权值就分别是\(1-cnt_0\),但是一个\(1\)代表的权值就是\(cnt_0+1\)了,一个\(1\)和一个\(0\)代表的权值是\(cnt_0+2\),两个\(1\)代表的权值是\(2cnt_0+1\),以此类推,这样可以以每一个数出现的次数来不重不漏地表示每一个状态。

显然,当每一个数字出现的次数尽可能平均时,代表的状态权值最大,实际测试可以发现最大权值不超过\(60000\),这样利用之前的转移方法就可以通过本题了,时间空间都没有问题。

\(Code:\)

#include <bits/stdc++.h>
using namespace std;
const int Mod = 998244353 , SIZE = 25 , Maxbit = 12 , M = 120 , MaxS = 60000;
char num[SIZE];
int m,len,cnt[Maxbit],f[MaxS][M];
int base[Maxbit],full[Maxbit],now[Maxbit];
inline void input(void)
{
scanf("%s",num+1);
len = strlen( num + 1 );
scanf("%d",&m);
}
inline void init(void)
{
for (int i=1;i<=len;i++)
cnt[ num[i] - '0' ] ++;
base[0] = 1 , full[0] = cnt[0];
for (int i=1;i<=9;i++)
{
base[i] = full[i-1] + 1;
full[i] = full[i-1] + base[i] * cnt[i];
}
}
inline void dp(void)
{
for (int i=1;i<=9;i++)
if ( cnt[i] )
f[base[i]][i%m] = 1;
for (int S=1;S<=full[9];S++)
{
int S_ = S;
for (int i=9;i>=0;i--)
if ( cnt[i] )
now[i] = cnt[i] - (S_/base[i]) , S_ %= base[i];
for (int i=0;i<=9;i++)
{
if ( !now[i] ) continue;
for (int j=0;j<m;j++)
{
int k = ( (j*10) % m + i ) % m;
f[ S + base[i] ][k] = ( f[ S + base[i] ][k] + f[S][j] ) % Mod;
}
}
}
}
int main(void)
{
input();
init();
dp();
printf("%d\n",f[full[9]][0]);
return 0;
}

<后记>

『数 变进制状压dp』的更多相关文章

  1. HDU 3001 Travelling 3进制状压dp

    题意:10个点,若干条边,边有花费,每个点最多走两次,求走过所有点,花费最少 分析:因为每个点最多走两次,所以联想到3进制,然后枚举状态,就行了(我也是照着网上大神的代码写的) #include &l ...

  2. T2963 贪吃蛇【BFS,四进制状压,A*】

    Online Judge:未知 Label:BFS,四进制状压,暴力,A*,哈希,玄学. 题目描述 给定一个n*m的地图和蛇的初始位置,地图中有些位置有石头,蛇不能经过.当然蛇也不能爬到地图之外. 每 ...

  3. ZRDay6A. 萌新拆塔(三进制状压dp)

    题意 Sol 这好像是我第一次接触三进制状压 首先,每次打完怪之后吃宝石不一定是最优的,因为有模仿怪的存在,可能你吃完宝石和他打就GG了.. 因此我们需要维护的状态有三个 0:没打 1:打了怪物 没吃 ...

  4. hdu 3001 Travelling 经过所有点(最多两次)的最短路径 三进制状压dp

    题目链接 题意 给定一个\(N\)个点的无向图,求从任意一个点出发,经过所有点的最短路径长度(每个点至多可以经过两次). 思路 状态表示.转移及大体思路 与 poj 3311 Hie with the ...

  5. HDU 3001 三进制状压DP

    N个城市,M条道路,每条道路有其经过的代价,每一个城市最多能够到达两次,求走全然部城市最小代价,起点随意. 三进制状压.存储每一个状态下每一个城市经过的次数. 转移方程: dp[i+b[k]][k]= ...

  6. 『Exclusive Access 2 dilworth定理 状压dp』

    Exclusive Access 2 Description 给出 N 个点M 条边的无向图,定向得到有向无环图,使得最长路最短. N ≤ 15, M ≤ 100 Input Format 第一行一个 ...

  7. UVA 10817 - Headmaster's Headache(三进制状压dp)

    题目:https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&category=20&pag ...

  8. Travelling (三进制+状压dp)

    题目链接 #include <bits/stdc++.h> using namespace std; typedef long long ll; inline ll read(){ ,f= ...

  9. HDU - 3001 Travelling(三进制状压dp)

    Travelling After coding so many days,Mr Acmer wants to have a good rest.So travelling is the best ch ...

随机推荐

  1. LCD编程_画点线圆

    上篇博客中进行了lcd的简单测试,这篇博客将进行更加复杂的测试——画点.画线.画圆.画线和画圆是在画点的基础上实现的,因此本篇博客重点实现画点操作. 先抛出这样的一个问题,已知: (x,y)的坐标: ...

  2. MySQL小测试(2)

    2.创建一个stu表,字段有:自增主键id,不为空姓名,默认值性别(枚举类型),无限制身高 create table stu( id int primary key auto_increment, n ...

  3. java Atomic compareAndSet部分原理分析

    以AtomicLong的compareAndSet方法举例.先说结论:如果CPU支持,则基于CPU指令(CMPXCHG8)实现:否则使用ObjectLocker锁实现. 分析过程如下: 该方法在jdk ...

  4. nginx 请求限制

    1.nginx 请求限制 1.连接频率限制 - limit_conn_module 2.请求频率限制 - limit_req_module 连接限制的语法 请求限制的语法 limit_conn_zon ...

  5. 网络编程 TCP协议:三次握手,四次回收,反馈机制 socket套接字通信 粘包问题与解决方法

    TCP协议:传输协议,基于端口工作 三次握手,四次挥手 TCP协议建立双向通道. 三次握手, 建连接: 1:客户端向服务端发送建立连接的请求 2:服务端返回收到请求的信息给客户端,并且发送往客户端建立 ...

  6. (二)STM32开发例程

    1控制LED和继电器 除了 PA11和PA12其他都可以 void setup() { pinMode(PC13, OUTPUT); pinMode(PC14, OUTPUT); pinMode(PC ...

  7. 在Hadoop-3.1.2上安装HBase-2.2.1

    目录 目录 1 1. 前言 3 2. 缩略语 3 3. 安装规划 3 3.1. 用户规划 3 3.2. 目录规划 4 4. 相关端口 4 5. 下载安装包 4 6. 修改配置文件 5 6.1. 修改策 ...

  8. vue动态添加路由,跳转页面时,页面报错路由重复:vue-router.esm.js?8c4f:16 [vue-router] Duplicate named routes definition: { name: "Login", path: "/login" }

    之前用了一个vue-element-admin做了一个小项目,里面用到了动态添加路由,动态展示侧边栏, 当我切换页面时,控制台总是警告提示路由重复,连续跳转几次页面后,控制台就被这些警告占满了, 于是 ...

  9. Gamma阶段项目展示

    Gamma阶段项目展示 一. 团队成员介绍 姓名 Gamma职责 个人博客 张圆宁 PM,后端 个人博客 王文珺 后端 个人博客 牛宇航 后端 个人博客 申化文 后端 个人博客 汪慕澜 测试,部署 个 ...

  10. Failed to contact the endpoint at http://controller:35357/ for discovery. Fallback to using that endpoint as the base url.

    问题描述 openstack安装过程中,执行 openstack domain create --description "Domain" example 报错如下: Failed ...