51nod 1061 最复杂的数V2
题目链接
题面简述
求\([1, n]\)中约数个数最多的数。
\(n \le 10^{200}\)
题解
首先,答案一定是一个反素数。
什么是反素数?
一个正整数\(x\)是反素数的充要条件是:\([1, x - 1]\)中的整数的约数个数都小于\(x\)的约数个数。
反素数有什么性质?
把一个反素数分解成\(p_1^{a_1}p_2^{a_2}...p_n^{a_n}\)的形式,则\(a_1 \ge a_2 \ge ... \ge a_n\)。
除\(1\)以外,一个反素数\(x\)一定可以由一个小于\(x\)的反素数乘上一个质数得来。(证明大概比较显然?感性理解一下 = =)
如果我们能求出\(10^{200}\)以内所有反质数,询问时直接lower_bound即可。
由此设计出算法:
- 维护一个堆,一开始只放\(1\)进去; 
- 每次取堆中最小的数(堆顶元素); 
- 判断这是不是一个反素数:根据定义,如果比它小的反素数中有约数个数与它相等或更多的,则它不是反素数。维护当前已经出堆的反素数的约数个数的最大值,如果当前堆顶元素的约数个数不比它大,则它不是反素数。 
- 枚举各个合法的素数(要求满足性质1)和它相乘,如果没有超出\(10^{200}\)则也压到堆中。 
这样就能求出\(10^{200}\)以内所有反质数了(实际上只有3810个!)
这道题需要高精度,好在只需要高精度乘低精度即可。
AC代码
#include <cstdio>
#include <cmath>
#include <cstring>
#include <algorithm>
#include <iostream>
#include <queue>
#define space putchar(' ')
#define enter putchar('\n')
using namespace std;
typedef long long ll;
template <class T>
void read(T &x){
    char c;
    bool op = 0;
    while(c = getchar(), c < '0' || c > '9')
	if(c == '-') op = 1;
    x = c - '0';
    while(c = getchar(), c >= '0' && c <= '9')
	x = x * 10 + c - '0';
    if(op) x = -x;
}
template <class T>
void write(T x){
    if(x < 0) putchar('-'), x = -x;
    if(x >= 10) write(x / 10);
    putchar('0' + x % 10);
}
const int N = 1000005, maxp = 100;
int T, prime[N], pcnt, tot;
bool np[N];
char s[256];
struct big {
    ll n[256], l;
    big(){
	memset(n, 0, sizeof(n));
	n[1] = l = 1;
    }
    void out(){
	for(int i = l; i; i--)
	    write(n[i]);
    }
    bool operator < (const big &b) const {
	if(l != b.l) return l < b.l;
	for(int i = l; i; i--)
	    if(n[i] != b.n[i]) return n[i] < b.n[i];
	return 0;
    }
    bool operator > (const big &b) const {
	if(l != b.l) return l > b.l;
	for(int i = l; i; i--)
	    if(n[i] != b.n[i]) return n[i] > b.n[i];
	return 0;
    }
    big operator * (ll x) const {
	big c;
	c.l = l;
	for(int i = 1; i <= l; i++)
	    c.n[i] = n[i] * x;
	while(x) c.l++, x /= 10;
	for(int i = 1; i <= c.l; i++)
	    c.n[i + 1] += c.n[i] / 10, c.n[i] %= 10;
	while(!c.n[c.l]) c.l--;
	return c;
    }
} MAX, cur;
struct data{
    big num;
    int cnt[maxp];
    data(){
	memset(cnt, 0, sizeof(cnt));
    }
    data(big x){
	num = x;
	memset(cnt, 0, sizeof(cnt));
    }
    bool operator < (const data &b) const{
	return num > b.num;
    }
    data operator * (int p) const{
	data c;
	c.num = num * prime[p];
	memcpy(c.cnt, cnt, sizeof(cnt));
	c.cnt[p]++;
	return c;
    }
} lst[4005];
priority_queue <data> que;
void euler(int n){
    np[0] = np[1] = 1;
    for(int i = 2; i <= n; i++){
	if(!np[i]) prime[++pcnt] = i;
	for(int j = 1; j <= pcnt && i * prime[j] <= n; j++){
	    np[i * prime[j]] = 1;
	    if(i % prime[j] == 0) break;
	}
    }
}
int main(){
    euler(1000000);
    MAX.l = 201;
    MAX.n[201] = MAX.n[1] = 1;
    cur.n[1] = 0;
    que.push(data());
    while(!que.empty()){
	data u = que.top();
	que.pop();
	big d;
	for(int p = 1; u.cnt[p]; p++)
	    d = d * (u.cnt[p] + 1);
	if(!(cur < d)) continue;
	lst[++tot] = u;
	cur = d;
	for(int p = 1; p < maxp && (p == 1 || u.cnt[p - 1]); p++)
	    if((p == 1 || u.cnt[p] < u.cnt[p - 1])){
		data v = u * p;
		if(v.num < MAX) que.push(v);
	    }
    }
    reverse(lst + 1, lst + tot + 1);
    read(T);
    while(T--){
	scanf("%s", s + 1);
	big n;
	n.l = strlen(s + 1);
	for(int i = 1; i <= n.l; i++)
	    n.n[n.l - i + 1] = s[i] - '0';
	data v = data(n);
	data u = *lower_bound(lst + 1, lst + tot + 1, v);
	big d;
	for(int p = 1; u.cnt[p]; p++)
	    d = d * (u.cnt[p] + 1);
	u.num.out(), space, d.out(), enter;
    }
    return 0;
}
51nod 1061 最复杂的数V2的更多相关文章
- 【51nod】1061 最复杂的数 V2
		题解 我是榜上最后一名= = 可能高精度用vector太慢了吧--什么破题= = 这道题很简单,如果高精度熟练代码--也很简单--然而,参数调了好久 我们发现质数的指数一定是,质数越小,指数越大,这个 ... 
- 51Nod 1084:矩阵取数问题 V2(多维DP)
		1084 矩阵取数问题 V2 基准时间限制:2 秒 空间限制:131072 KB 分值: 80 难度:5级算法题 收藏 关注 一个M*N矩阵中有不同的正整数,经过这个格子,就能获得相应价值的奖励 ... 
- 51nod 1218 最长递增子序列 V2——LIS+思路(套路)
		题目:http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1218 自己怎么连这种 喜闻乐见的大水题 都做不出来了…… 好像见过 ... 
- 51nod 1218 最长递增子序列 V2(dp + 思维)
		题目链接:https://www.51nod.com/onlineJudge/questionCode.html#!problemId=1218 题解:先要确定这些点是不是属于最长递增序列然后再确定这 ... 
- 51nod 1053 最大M子段和 V2
		N个整数组成的序列a[1],a[2],a[3],…,a[n],将这N个数划分为互不相交的M个子段,并且这M个子段的和是最大的.如果M >= N个数中正数的个数,那么输出所有正数的和. 例如:-2 ... 
- 51nod 1132 覆盖数字的数量 V2
		http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1132 题意是给定a,b,l,r求[l,r]内有几个整数可以表示成ax+b ... 
- 51nod 1479 小Y的数论题
		一脸不可做题~~~233333 T<=100000,所以一定要logn出解啦. 但是完全没有头绪*&#……%*&……()……#¥*#@ 题解: 因为2^p+2^p=2^(p+1) ... 
- 51nod 1060 最复杂的数
		把一个数的约数个数定义为该数的复杂程度,给出一个n,求1-n中复杂程度最高的那个数. 例如:12的约数为:1 2 3 4 6 12,共6个数,所以12的复杂程度是6.如果有多个数复杂度相等,输出最 ... 
- 51nod 1060 最复杂的数  反素数
		1060 最复杂的数 基准时间限制:1 秒 空间限制:131072 KB 把一个数的约数个数定义为该数的复杂程度,给出一个n,求1-n中复杂程度最高的那个数. 例如:12的约数为:1 2 3 4 6 ... 
随机推荐
- 将当前的Ubuntu系统封装成为可以安装(发布)的iso镜像
			将当前的Ubuntu系统封装成为可以安装(发布)的iso镜像 在使用以上方法安装依赖的时候xresprobe 会找不到安装地址,采用下面的方式: Package xresprobe is not in ... 
- Centos7 64位 --  glibc-2.29 编译升级方法(已成功)
			某软件出现漏洞,需要升级解决(忘了哪个)结果提示glibc版本过低. 懵懂无知的我以为glibc想其他软件一样编译升级一下就好.. 结果? 重装系统! 说真的,如非必要(或学习),请勿升级 glibc ... 
- pandas:解决groupby().apply()方法打印两次
			对于以下dataframe执行dataframe.groupby(['name', 'course']).apply(lambda x: test(x)) 操作 其中test(x)函数为: def t ... 
- 如何看待P2P领域的羊毛党?
			本文来自网易云社区 不利:不利的影响应该是显而易见的,前面的题主也有解释过.总结来说,不利的影响主要是两点: a. 对平台毛利和资金的损害.一般来说,优惠活动本是一个用户只能享用一次,但如果注册多个账 ... 
- .Net Core Cookie-Based认证与授权
			.Net Core的其中一种认证与授权模式是基于Cookie的,首先我们先创建一个.Net Core MVC 项目: 然后增加对页面访问的权限控制,对要访问的页面Conytroller增加Author ... 
- 忘记mysql数据库root密码
			找到配置文件my.ini ,然后将其打开,可以选择用记事本打开,查找的方法如下: 打开后,搜索mysqld关键字 找到后,在mysqld下面添加skip-grant-tables,保存退出. PS: ... 
- 第二次作业 --- 我对QQ的评测
			腾讯QQ(简称“QQ”)是腾讯公司开发的一款基于Internet的即时通信(IM)软件.腾讯QQ支持在线聊天.视频通话.点对点断点续传文件.共享文件.网络硬盘.自定义面板.QQ邮箱等多种功能,并可与多 ... 
- beta阶段测试基本概况报告
			文件地址 测试基本信息 Bitmap 测试 ... 
- Linux内核分析-创建新进程的过程
			分析Linux内核创建一个新进程的过程 task_struct结构体分析 struct task_struct{ volatile long state; //进程的状态 unsigned long ... 
- Particle filter for visual tracking
			Kalman Filter Cons: Kalman filtering is inadequate because it is based on the unimodal Gaussian dist ... 
