题目链接

51nod 1061

题面简述

求\([1, n]\)中约数个数最多的数。

\(n \le 10^{200}\)

题解

首先,答案一定是一个反素数

什么是反素数?

一个正整数\(x\)是反素数的充要条件是:\([1, x - 1]\)中的整数的约数个数都小于\(x\)的约数个数。

反素数有什么性质?

  1. 把一个反素数分解成\(p_1^{a_1}p_2^{a_2}...p_n^{a_n}\)的形式,则\(a_1 \ge a_2 \ge ... \ge a_n\)。

  2. 除\(1\)以外,一个反素数\(x\)一定可以由一个小于\(x\)的反素数乘上一个质数得来。(证明大概比较显然?感性理解一下 = =)

如果我们能求出\(10^{200}\)以内所有反质数,询问时直接lower_bound即可。

由此设计出算法:

  • 维护一个堆,一开始只放\(1\)进去;

  • 每次取堆中最小的数(堆顶元素);

  • 判断这是不是一个反素数:根据定义,如果比它小的反素数中有约数个数与它相等或更多的,则它不是反素数。维护当前已经出堆的反素数的约数个数的最大值,如果当前堆顶元素的约数个数不比它大,则它不是反素数。

  • 枚举各个合法的素数(要求满足性质1)和它相乘,如果没有超出\(10^{200}\)则也压到堆中。

这样就能求出\(10^{200}\)以内所有反质数了(实际上只有3810个!)

这道题需要高精度,好在只需要高精度乘低精度即可。

AC代码

  1. #include <cstdio>
  2. #include <cmath>
  3. #include <cstring>
  4. #include <algorithm>
  5. #include <iostream>
  6. #include <queue>
  7. #define space putchar(' ')
  8. #define enter putchar('\n')
  9. using namespace std;
  10. typedef long long ll;
  11. template <class T>
  12. void read(T &x){
  13. char c;
  14. bool op = 0;
  15. while(c = getchar(), c < '0' || c > '9')
  16. if(c == '-') op = 1;
  17. x = c - '0';
  18. while(c = getchar(), c >= '0' && c <= '9')
  19. x = x * 10 + c - '0';
  20. if(op) x = -x;
  21. }
  22. template <class T>
  23. void write(T x){
  24. if(x < 0) putchar('-'), x = -x;
  25. if(x >= 10) write(x / 10);
  26. putchar('0' + x % 10);
  27. }
  28. const int N = 1000005, maxp = 100;
  29. int T, prime[N], pcnt, tot;
  30. bool np[N];
  31. char s[256];
  32. struct big {
  33. ll n[256], l;
  34. big(){
  35. memset(n, 0, sizeof(n));
  36. n[1] = l = 1;
  37. }
  38. void out(){
  39. for(int i = l; i; i--)
  40. write(n[i]);
  41. }
  42. bool operator < (const big &b) const {
  43. if(l != b.l) return l < b.l;
  44. for(int i = l; i; i--)
  45. if(n[i] != b.n[i]) return n[i] < b.n[i];
  46. return 0;
  47. }
  48. bool operator > (const big &b) const {
  49. if(l != b.l) return l > b.l;
  50. for(int i = l; i; i--)
  51. if(n[i] != b.n[i]) return n[i] > b.n[i];
  52. return 0;
  53. }
  54. big operator * (ll x) const {
  55. big c;
  56. c.l = l;
  57. for(int i = 1; i <= l; i++)
  58. c.n[i] = n[i] * x;
  59. while(x) c.l++, x /= 10;
  60. for(int i = 1; i <= c.l; i++)
  61. c.n[i + 1] += c.n[i] / 10, c.n[i] %= 10;
  62. while(!c.n[c.l]) c.l--;
  63. return c;
  64. }
  65. } MAX, cur;
  66. struct data{
  67. big num;
  68. int cnt[maxp];
  69. data(){
  70. memset(cnt, 0, sizeof(cnt));
  71. }
  72. data(big x){
  73. num = x;
  74. memset(cnt, 0, sizeof(cnt));
  75. }
  76. bool operator < (const data &b) const{
  77. return num > b.num;
  78. }
  79. data operator * (int p) const{
  80. data c;
  81. c.num = num * prime[p];
  82. memcpy(c.cnt, cnt, sizeof(cnt));
  83. c.cnt[p]++;
  84. return c;
  85. }
  86. } lst[4005];
  87. priority_queue <data> que;
  88. void euler(int n){
  89. np[0] = np[1] = 1;
  90. for(int i = 2; i <= n; i++){
  91. if(!np[i]) prime[++pcnt] = i;
  92. for(int j = 1; j <= pcnt && i * prime[j] <= n; j++){
  93. np[i * prime[j]] = 1;
  94. if(i % prime[j] == 0) break;
  95. }
  96. }
  97. }
  98. int main(){
  99. euler(1000000);
  100. MAX.l = 201;
  101. MAX.n[201] = MAX.n[1] = 1;
  102. cur.n[1] = 0;
  103. que.push(data());
  104. while(!que.empty()){
  105. data u = que.top();
  106. que.pop();
  107. big d;
  108. for(int p = 1; u.cnt[p]; p++)
  109. d = d * (u.cnt[p] + 1);
  110. if(!(cur < d)) continue;
  111. lst[++tot] = u;
  112. cur = d;
  113. for(int p = 1; p < maxp && (p == 1 || u.cnt[p - 1]); p++)
  114. if((p == 1 || u.cnt[p] < u.cnt[p - 1])){
  115. data v = u * p;
  116. if(v.num < MAX) que.push(v);
  117. }
  118. }
  119. reverse(lst + 1, lst + tot + 1);
  120. read(T);
  121. while(T--){
  122. scanf("%s", s + 1);
  123. big n;
  124. n.l = strlen(s + 1);
  125. for(int i = 1; i <= n.l; i++)
  126. n.n[n.l - i + 1] = s[i] - '0';
  127. data v = data(n);
  128. data u = *lower_bound(lst + 1, lst + tot + 1, v);
  129. big d;
  130. for(int p = 1; u.cnt[p]; p++)
  131. d = d * (u.cnt[p] + 1);
  132. u.num.out(), space, d.out(), enter;
  133. }
  134. return 0;
  135. }

51nod 1061 最复杂的数V2的更多相关文章

  1. 【51nod】1061 最复杂的数 V2

    题解 我是榜上最后一名= = 可能高精度用vector太慢了吧--什么破题= = 这道题很简单,如果高精度熟练代码--也很简单--然而,参数调了好久 我们发现质数的指数一定是,质数越小,指数越大,这个 ...

  2. 51Nod 1084:矩阵取数问题 V2(多维DP)

    1084 矩阵取数问题 V2  基准时间限制:2 秒 空间限制:131072 KB 分值: 80 难度:5级算法题  收藏  关注 一个M*N矩阵中有不同的正整数,经过这个格子,就能获得相应价值的奖励 ...

  3. 51nod 1218 最长递增子序列 V2——LIS+思路(套路)

    题目:http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1218 自己怎么连这种 喜闻乐见的大水题 都做不出来了…… 好像见过 ...

  4. 51nod 1218 最长递增子序列 V2(dp + 思维)

    题目链接:https://www.51nod.com/onlineJudge/questionCode.html#!problemId=1218 题解:先要确定这些点是不是属于最长递增序列然后再确定这 ...

  5. 51nod 1053 最大M子段和 V2

    N个整数组成的序列a[1],a[2],a[3],…,a[n],将这N个数划分为互不相交的M个子段,并且这M个子段的和是最大的.如果M >= N个数中正数的个数,那么输出所有正数的和. 例如:-2 ...

  6. 51nod 1132 覆盖数字的数量 V2

    http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1132 题意是给定a,b,l,r求[l,r]内有几个整数可以表示成ax+b ...

  7. 51nod 1479 小Y的数论题

    一脸不可做题~~~233333 T<=100000,所以一定要logn出解啦. 但是完全没有头绪*&#……%*&……()……#¥*#@ 题解: 因为2^p+2^p=2^(p+1) ...

  8. 51nod 1060 最复杂的数

    把一个数的约数个数定义为该数的复杂程度,给出一个n,求1-n中复杂程度最高的那个数.   例如:12的约数为:1 2 3 4 6 12,共6个数,所以12的复杂程度是6.如果有多个数复杂度相等,输出最 ...

  9. 51nod 1060 最复杂的数 反素数

    1060 最复杂的数 基准时间限制:1 秒 空间限制:131072 KB 把一个数的约数个数定义为该数的复杂程度,给出一个n,求1-n中复杂程度最高的那个数. 例如:12的约数为:1 2 3 4 6 ...

随机推荐

  1. .NET小笔记-NPOI读取excel内容到DataTable

    下载比较新的NPOI组件支持excel2007以上的,把.dll添加引用 引入命名空间 using NPOI.HSSF.UserModel;using NPOI.SS.UserModel;using ...

  2. Flutter - TabBar导航栏切换后,状态丢失

    上一篇讲到了 Flutter - BottomNavigationBar底部导航栏切换后,状态丢失 里面提到了TabBar,这儿专门再写一下吧,具体怎么操作,来不让TabBar的状态丢失.毕竟大家99 ...

  3. vsftpd虚拟账户配置

    1. 概述 FTP是文件传输协议,在内外网的文件传输中使用广泛. 本篇博客主要介绍FTP服务器的部署和测试. 2. 软件环境部署 查看系统是否安装FTP软件(vsftpd),执行命令:rpm -qa ...

  4. LB层到Real Server之间访问请求的响应时间及HTTP状态码监控及报警设置

    为了监控到各业务的访问质量,基于LB层的Nginx日志,实现LB层到Real Server之间访问请求的响应时间(即upstream_response_time)及HTTP状态码(即upstream_ ...

  5. http 概念

    什么是回调? 什么是同步/异步? 什么是I/O? 什么是单线/多线程? 什么是阻塞/非阻塞? 什么是事件? 什么是事件驱动? 什么是基于事件驱动的回调? 什么是事件循环?

  6. pl/sql破解方法

    转载源:http://blog.csdn.net/oscar999/article/details/2123803 打开注册表在run下输入regedit删除1.HKEY_CURRENT_USER/S ...

  7. C#获取当月第一天和最后一天

    当月第一天0时0分0秒: DateTime.Now.AddDays(1 - DateTime.Now.Day).Date 当月最后一天23时59分59秒: DateTime.Now.AddDays(1 ...

  8. 『编程题全队』"Gugua"事务管理系统项目宣传文案

    一.项目简介 1.项目简介 Gugua是为了解决有事务管理需要的人群的痛苦, 他们需要 一个便利和高效的个人和团体事务管理平台,但是现有的方案并没有很好地解决这些需求,我们有独特的办法是提供跨平台的软 ...

  9. 常用的查询DOM的方法

    查询body的方法=========== document.body // document.getElementsTagname("body")[0]; 查询html的方法=== ...

  10. Oracle 数据表误删恢复 Flashback

    1. 前提条件. recyclebin 参数打开. 验证参数是否打开: SHOW PARAMETER RECYCLEBIN 2. 如果参数没有打开的话 需要打开,并且重启一下数据库方法为 alter ...