http://www.lydsy.com/JudgeOnline/problem.php?id=1005 (题目链接)

题意

  给出标号为1到N的点,以及某些点最终的度数,允许在任意两点间连线,可产生多少棵度数满足要求的树?

Solution

  prufer编码。关于prufer,有一篇博客写得很好,还运用组合数学求出了公式,可惜代码是java。这道题还要写高精度,高精度除法太麻烦了,因为组合数一定是整数,所以我们选择将n!分解质因数,最后相加减就可以了。如何分解质因数呢,我当然不会去暴力分解,一次考试中就是暴力分解TLE了= =。

  若P为质数,则n!可分解为 一个数×P^x。其中 
  

  并且P^t小于n,所以就很好做了。

代码

  1. // bzoj1005
  2. #include<algorithm>
  3. #include<iostream>
  4. #include<cstring>
  5. #include<cstdlib>
  6. #include<cstdio>
  7. #include<cmath>
  8. #include<map>
  9. #define inf 2147483640
  10. #define LL long long
  11. #define free(a) freopen(a".in","r",stdin);freopen(a".out","w",stdout);
  12. using namespace std;
  13. inline LL getint() {
  14. LL x=0,f=1;char ch=getchar();
  15. while (ch>'9' || ch<'0') {if (ch=='-') f=-1;ch=getchar();}
  16. while (ch>='0' && ch<='9') {x=x*10+ch-'0';ch=getchar();}
  17. return x*f;
  18. }
  19.  
  20. const int maxn=10010;
  21. int ans[maxn*100],np[maxn],a[maxn],d[maxn];
  22. int l=1,n,sum;
  23.  
  24. void pls(int x,int f) {
  25. for (int i=2;i<=x;i++)
  26. if (!np[i]) for (int j=i;j<=x;j*=i) a[i]+=f*x/j;
  27. }
  28. void mul(int x) {
  29. for (int i=1;i<=l;i++) ans[i]*=x;
  30. for (int i=1;i<=l;i++) {
  31. ans[i+1]+=ans[i]/10;
  32. ans[i]%=10;
  33. }
  34. while (ans[l+1]>0) {l++;ans[1+l]+=ans[l]/10;ans[l]%=10;}
  35. }
  36. int main() {
  37. for (int i=2;i<=1000;i++) if (!np[i]) {
  38. int x=i+i;
  39. while (x<=1000) np[x]=1,x+=i;
  40. }
  41. scanf("%d",&n);
  42. if (n==1) {
  43. int x;scanf("%d",&x);
  44. if (!x) printf("1");
  45. else printf("0");
  46. return 0;
  47. }
  48. int cnt=0;ans[1]=1;
  49. for (int i=1;i<=n;i++) {
  50. scanf("%d",&d[i]);
  51. if (!d[i]) {printf("0");return 0;}
  52. if (d[i]>0) {d[i]--;sum+=d[i];}
  53. else cnt++;
  54. }
  55. if (sum>n-2) {printf("0");return 0;}
  56. pls(n-2,1);
  57. pls(n-2-sum,-1);
  58. for (int i=1;i<=n;i++) if (d[i]) pls(d[i],-1);
  59. for (int i=2;i<=1000;i++)
  60. for (int j=1;j<=a[i];j++) mul(i);
  61. for (int i=1;i<=n-2-sum;i++) mul(cnt);
  62. for (int i=l;i>=1;i--) printf("%d",ans[i]);
  63. return 0;
  64. }

  

【bzoj1005】 HNOI2008—明明的烦恼的更多相关文章

  1. bzoj1005 [HNOI2008]明明的烦恼

    1005: [HNOI2008]明明的烦恼 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 3032  Solved: 1209 Description ...

  2. bzoj1005: [HNOI2008]明明的烦恼(prufer+高精度)

    1005: [HNOI2008]明明的烦恼 题目:传送门 题解: 毒瘤题啊天~ 其实思考的过程还是比较简单的... 首先当然还是要了解好prufer序列的基本性质啦 那么和1211大体一致,主要还是利 ...

  3. [BZOJ1005] [HNOI2008] 明明的烦恼 (prufer编码)

    Description 自从明明学了树的结构,就对奇怪的树产生了兴趣......给出标号为1到N的点,以及某些点最终的度数,允许在任意两点间连线,可产生多少棵度数满足要求的树? Input 第一行为N ...

  4. 【prufer编码+组合数学】BZOJ1005 [HNOI2008]明明的烦恼

    Description 自从明明学了树的结构,就对奇怪的树产生了兴趣...... 给出标号为1到N的点,以及某些点最终的度数,允许在任意两点间连线,可产生多少棵度数满足要求的树? Solution 这 ...

  5. BZOJ1005 HNOI2008明明的烦恼(prufer+高精度)

    每个点的度数=prufer序列中的出现次数+1,所以即每次选一些位置放上某个点,答案即一堆组合数相乘.记一下每个因子的贡献分解一下质因数高精度乘起来即可. #include<iostream&g ...

  6. BZOJ1005:[HNOI2008]明明的烦恼(组合数学,Prufer)

    Description 自从明明学了树的结构,就对奇怪的树产生了兴趣......给出标号为1到N的点,以及某些点最终的度数,允许在任意两点间连线,可产生多少棵度数满足要求的树? Input 第一行为N ...

  7. [bzoj1005][HNOI2008][明明的烦恼] (高精度+prufer定理)

    Description 自从明明学了树的结构,就对奇怪的树产生了兴趣......给出标号为1到N的点,以及某些点最终的度数,允许在任意两点间连线,可产生多少棵度数满足要求的树? Input 第一行为N ...

  8. bzoj1005: [HNOI2008]明明的烦恼 prufer序列

    https://www.lydsy.com/JudgeOnline/problem.php?id=1005 给出标号为1到N的点,以及某些点最终的度数,允许在任意两点间连线,可产生多少棵度数满足要求的 ...

  9. [bzoj1005][HNOI2008]明明的烦恼-Prufer编码+高精度

    Brief Description 给出标号为1到N的点,以及某些点最终的度数,允许在 任意两点间连线,可产生多少棵度数满足要求的树? Algorithm Design 结论题. 首先可以参考这篇文章 ...

  10. [BZOJ1005][HNOI2008]明明的烦恼 数学+prufer序列+高精度

    #include<cstdio> #include<cstring> #include<algorithm> using namespace std; int N; ...

随机推荐

  1. 使用友盟进行apk的自动更新

    [声明] 欢迎转载,但请保留文章原始出处→_→ 生命壹号:http://www.cnblogs.com/smyhvae/ 文章来源:http://www.cnblogs.com/smyhvae/p/4 ...

  2. AC日记——积木大赛 洛谷 P1969

    题目描述 春春幼儿园举办了一年一度的“积木大赛”.今年比赛的内容是搭建一座宽度为n的大厦,大厦可以看成由n块宽度为1的积木组成,第i块积木的最终高度需要是hi. 在搭建开始之前,没有任何积木(可以看成 ...

  3. Resharper团队协作之TODO

    TODO 需求 首先我想跟大家分享一下我们团队的代码检查流程. 1. 项目经理随时会检查成员的代码,如果发现有不符合规范的代码,会在注释里面加todo.比如,假设leo的代码不符合规范,那么项目经理就 ...

  4. MongoDB学习(四)客户端工具备份数据库

    在上一篇MongoDB学习(三)中讲解了如何在服务器端进行数据的导入导出与备份恢复,本篇介绍下如何利用客户端工具来进行远程服务器的数据备份到本地. 以客户端工具MongoVUE为例来进行讲解: 1.首 ...

  5. maven总结2

    依赖 maven版本:apache-maven-3.1.1    IDE: springsource   默认支持maven集成             若使用的是eclipse,则需要先安装m2ec ...

  6. Linux下smokeping网络监控环境部署记录

    smokeping是一款监控网络状态和稳定性的开源软件(它是rrdtool的作者开发的),通过它可以监控到公司IDC的网络状况,如延时,丢包率,是否BGP多线等:smokeping会向目标设备和系统发 ...

  7. getLovParameter

    else if (pageContext.isLovEvent()) { StHelper.handleLovEvent(pageContext, webBean); } public static ...

  8. Caffe学习系列(21):caffe图形化操作工具digits的安装与运行

    经过前面一系列的学习,我们基本上学会了如何在linux下运行caffe程序,也学会了如何用python接口进行数据及参数的可视化. 如果还没有学会的,请自行细细阅读: caffe学习系列:http:/ ...

  9. 在matlab中进行地理坐标和像素坐标的相互转换

    clc;close all;clear; %地理坐标和像素坐标的相互转换 [pic,R]=geotiffread('boston.tif'); %读取带地理坐标信息的tif影像 [m,n,~]=siz ...

  10. Android调用蓝牙打印机

    首先需要一个jar包,bluesdk,请自行百度. 具体排版样式跟网络打印机打印排版样式实现一样,这里不多叙述,只贴一个实现方法代码.蓝牙打印机使用前需要先跟手机配对,可以保存在本地,记录下地址,这里 ...