题面

传送门:UOJ


Solution

这题的数位DP好蛋疼啊qwq

好吧,我们说回正题。

首先,我们先回忆一下LUCAS定理:

\(C_n^m \equiv C_{n/p}^{m/p} \times C_{n\%p}^{m\%p} (\%p)\)

我们仔细观察这个定理,就可以发现一个事实:LUCAS定理本质上是在对n,m两个数做K进制下的数位分离

所以说,LUCAS定理我们可以这样表示:

\(C_n^m \equiv \prod C_{a_i}^{b_i}\)

(ai与bi为K进制拆分后的两个数的每一位数,若一个数的位数不足另一个数,则以前导零填充

我们要判断一个\(C_n^m\)是否能被K整除,只需要保证其中一个\(C_{a_i}^{b_i}\)能被K整除(即同余K为零)就好。

又因为K为质数,且ai,bi均小于K,所以说我们要使得\(C_{a_i}^{b_i}\)为0,必须有\(b_i\)>\(a_i\)

.

.

所以说,问题就变为了对于有多少个\((i,j)\)使得\(j\)中某一位\(>i\)

这个新的问题显然可以用数位DP来解决。

在这里,我使用记忆化搜索来写(用记忆化可以减少讨论数)

考虑这样设状态:

设\(f[x][0/1][0/1][0/1][0/1][0/1]\)表示填到第\(x\)位,

i是否卡上界n,j是否卡上界m,j是否卡上界 \((j<i)\),上一位是否为前导零(这道题不需要,但是为了模板完整性。我还是写上去了),之前是否有某一位达成需求,之后可以达成的总共的可行方案数

转移非常好讨论,我们只需要注意一下j的上界是两个限制的最小值就好。

我是这样写转移的:

  1. for(int i=0;i<=(limit1==true?l1[to]:K);i++)
  2. {
  3. int t_j=(limit2==true?l2[to]:K);
  4. t_j=(limit3==true?min(i,t_j):t_j);
  5. for(int j=0;j<=t_j;j++)
  6. {
  7. t_ans+=dfs(to+1,limit1==true and i==l1[to],limit2==true and j==l2[to],limit3==true and j==i,zero==true and i==0,OK==true or j>i);
  8. t_ans%=poi;
  9. }
  10. }

这样子写,看起来时间复杂度是

\(O(T*n^2*2^5)\)

但是因为我们会少讨论很多没有意义的情况,所以说能跑得过去。

确切复杂度我不会算qwq

还请dalao们指点


Code

  1. //BZOJ 4737: 组合数问题
  2. //Jan,14th,2019
  3. //LUCAS定理的运用+鬼畜数位DP
  4. #include<iostream>
  5. #include<cstdio>
  6. #include<algorithm>
  7. #include<cstring>
  8. using namespace std;
  9. long long read()
  10. {
  11. long long x=0,f=1; char c=getchar();
  12. while(!isdigit(c)){if(c=='-') f=-1;c=getchar();}
  13. while(isdigit(c)){x=x*10+c-'0';c=getchar();}
  14. return x*f;
  15. }
  16. const int poi=1000000007;
  17. const int N=70;
  18. long long f[N][2][2][2][2][2];//到i位,底数卡不卡n,指数卡不卡m,指数卡不卡底数,zero,OK
  19. int n,l1[N],l2[N],K;
  20. long long dfs(int to,bool limit1,bool limit2,bool limit3,bool zero,bool OK)
  21. {
  22. if(f[to][limit1][limit2][limit3][zero][OK]>=0) return f[to][limit1][limit2][limit3][zero][OK];
  23. long long t_ans=0;
  24. if(to==n+1)
  25. {
  26. if(OK==true)
  27. t_ans=1;
  28. return f[to][limit1][limit2][limit3][zero][OK]=t_ans;
  29. }
  30. for(int i=0;i<=(limit1==true?l1[to]:K);i++)
  31. {
  32. int t_j=(limit2==true?l2[to]:K);
  33. t_j=(limit3==true?min(i,t_j):t_j);
  34. for(int j=0;j<=t_j;j++)
  35. {
  36. t_ans+=dfs(to+1,limit1==true and i==l1[to],limit2==true and j==l2[to],limit3==true and j==i,zero==true and i==0,OK==true or j>i);
  37. t_ans%=poi;
  38. }
  39. }
  40. return f[to][limit1][limit2][limit3][zero][OK]=t_ans;
  41. }
  42. int main()
  43. {
  44. int T=read();K=read();
  45. for(;T>0;T--)
  46. {
  47. n=0;
  48. long long num1=read(),num2=read();
  49. num2=min(num1,num2);//防止m>n
  50. while(num1!=0)
  51. l1[++n]=num1%K,num1/=K;
  52. for(int i=1;i<=n;i++)
  53. l2[i]=num2%K,num2/=K;
  54. reverse(l1+1,l1+1+n);
  55. reverse(l2+1,l2+1+n);
  56. memset(f,0x80,sizeof f);
  57. K--;
  58. dfs(1,true,true,true,true,false);
  59. K++;
  60. printf("%lld\n",f[1][true][true][true][true][false]);
  61. }
  62. return 0;
  63. }

[UOJ 275/BZOJ4737] 【清华集训2016】组合数问题 (LUCAS定理的运用+数位DP)的更多相关文章

  1. UOJ 275. 【清华集训2016】组合数问题

    UOJ 275. [清华集训2016]组合数问题 组合数 $C_n^m $表示的是从 \(n\) 个物品中选出 \(m\) 个物品的方案数.举个例子,从$ (1,2,3)(1,2,3)$ 三个物品中选 ...

  2. UOJ#275. 【清华集训2016】组合数问题 数位dp

    原文链接https://www.cnblogs.com/zhouzhendong/p/UOJ275.html 题解 用卢卡斯定理转化成一个 k 进制意义下的数位 dp 即可. 算答案的时候补集转化一下 ...

  3. uoj#275. 【清华集训2016】组合数问题(数位dp)

    传送门 假设有\(k|{n\choose m}\),因为\(n!\)中质因子\(k\)的次数为\(S(n)=\left\lfloor\frac{n}{k}\right\rfloor+\left\lfl ...

  4. UOJ #269. 【清华集训2016】如何优雅地求和

    UOJ #269. [清华集训2016]如何优雅地求和 题目链接 给定一个\(m\)次多项式\(f(x)\)的\(m+1\)个点值:\(f(0)\)到\(f(m)\). 然后求: \[ Q(f,n,x ...

  5. [UOJ#276]【清华集训2016】汽水

    [UOJ#276][清华集训2016]汽水 试题描述 牛牛来到了一个盛产汽水的国度旅行. 这个国度的地图上有 \(n\) 个城市,这些城市之间用 \(n−1\) 条道路连接,任意两个城市之间,都存在一 ...

  6. UOJ #274. 【清华集训2016】温暖会指引我们前行 [lct]

    #274. [清华集训2016]温暖会指引我们前行 题意比较巧妙 裸lct维护最大生成树 #include <iostream> #include <cstdio> #incl ...

  7. UOJ275 [清华集训2016] 组合数问题 【Lucas定理】【数位DP】

    题目分析: 我记得很久以前有人跟我说NOIP2016的题目出了加强版在清华集训中,但这似乎是一道无关的题目? 由于$k$为素数,那么$lucas$定理就可以搬上台面了. 注意到$\binom{i}{j ...

  8. UOJ #276「清华集训2016」汽水

    为什么你们常数都这么小啊 UOJ #276 题意:在树上找一条链使得|边权平均值$ -k$|尽量小,$ n<=5e4$ $ Solution:$ 首先二分答案$ ans$,即我们需要找一条链使得 ...

  9. uoj#274. 【清华集训2016】温暖会指引我们前行

    http://uoj.ac/problem/274 由于边权互不相同,只需用lct维护带加边的最大生成树 #include<bits/stdc++.h> #define lc ch][0 ...

随机推荐

  1. 栈和队列数据结构的基本概念及其相关的Python实现

    先来回顾一下栈和队列的基本概念: 相同点:从"数据结构"的角度看,它们都是线性结构,即数据元素之间的关系相同. 不同点:栈(Stack)是限定只能在表的一端进行插入和删除操作的线性 ...

  2. 【题解】 P2734 [USACO3.3]游戏 A Game

    \(\color{purple}{Link}\) \(\text{Solution:}\) 考虑区间\([l,r]\)的最优解.显然它可以由\([l+1,r]\)或\([l,r-1]\)转移而来.至此 ...

  3. 实现Excel文件的上传和解析

    前言 本文思维导图 一.需求描述 实现一个页面上传excel的功能,并对excel中的内容做解析,最后存储在数据库中. 二.代码实现 需求实现思路: 先对上传的文件做校验和解析,这里我们通过Excel ...

  4. devops-jenkins-Pipeline实战

    1) 配置gitlab的分支项目 点击红色标记的创建 project 项目 点击Create project创建Pipeline-demo项目 项目创建完成,然后我们创建一个Add README 然后 ...

  5. Linux系统的一些问题

    1.操作系统提供的服务: - 进程调度 - 内存管理 - 磁盘管理 - 网络服务 - 设备管理 - 提供应用程序编程接口 2.shell是什么? shell是一种具有特殊用途的程序,主要用于读取用户输 ...

  6. java读取中文乱码解决方法

    Java读取文本文件(例如csv文件.txt文件等),遇到中文就变成乱码.读取代码如下: List<String> lines=new ArrayList<String>(); ...

  7. 分布式系统中的CAP、ACID、BASE概念

    目录 CAP ACID BASE CAP 分布式系统中,这三个特性只能满足其中两个. 一致性(Consistency):分布式中一致性又分强一致性和弱一致性,强一致性主浊任何时刻任何节点看到的数据都是 ...

  8. 经验分享:Windows10值得推荐的软件,总有一款是你的菜

    今天在知乎上看到有人分享wids10推荐好用的软件:今天小编做了一点点的修改和根据自己的使用情况总结出来转发分享给大家:   1.安全放病毒--火绒[推荐] 2.办公软件--office2019[推荐 ...

  9. 皕杰报表:连接数据库失败,请检查数据源配置(oracle.jdbc.driver.OracleDriver)

    皕杰报表:连接数据库失败,请检查数据源配置(oracle.jdbc.driver.OracleDriver)问题解决: 缺少了classes12.jar 在窗口--首选项--报表运行时配置--添加-- ...

  10. Spring官方都推荐使用的@Transactional事务,为啥我不建议使用!

    GitHub 17k Star 的Java工程师成神之路,不来了解一下吗! GitHub 17k Star 的Java工程师成神之路,真的不来了解一下吗! GitHub 17k Star 的Java工 ...