对于任意输入的正整数n,请编程求出具有n个不同因子的最小正整数m。

Solution

(乍一看很简单却搞了好久?我真是太菜了)

根据因子个数计算公式

若 \(m = \prod p_i^{q_i}\), 则 \(n = \prod (q_i + 1)\)

设 \(f[i][j]\) 为只包含前 \(j\) 个质因数,因子个数为 \(i\) 的最小的数

转移类似背包: \(f[i][j]=min_{k|i} (f[i/k][j-1] \cdot p_j^{k-1})\)

这样直接做是 \(O(n \sqrt n \log n)\) ,考虑到需要枚举的 \(i\) 有且仅有 \(n\) 的因数,而约数个数的一个宽上界是 \(O(\sqrt n)\),复杂度就压缩到了 \(O(n \log n)\)

使用高精度直接 dp 可能会复杂度爆炸,所以我们对数一下

#include <bits/stdc++.h>
using namespace std; struct Biguint {
int a[100005], len; Biguint() {
memset(a, 0, sizeof a);
len = 0;
} void read() {
string str;
cin >> str;
memset(a, 0, sizeof a);
len = str.length();
for (int i = 0; i < str.size(); i++)
a[i] = str[str.length() - i - 1] - '0';
} void print() {
for (int i = len - 1; i >= 0; i--) {
cout << a[i];
}
} bool operator < (const Biguint& obj) {
const int* b = obj.a;
if (this->len == obj.len) {
for (int i = len - 1; i>=0; --i)
if (a[i] != b[i]) return a[i] < b[i];
return false;
}
else return this->len < obj.len;
} bool operator > (const Biguint& obj) {
const int* b = obj.a;
if (this->len == obj.len) {
for (int i = len - 1; i>=0; --i)
if (a[i] != b[i]) return a[i] > b[i];
return false;
}
else return this->len > obj.len;
} bool operator != (const Biguint& obj) {
return (*this < obj) | (*this > obj);
} bool operator == (const Biguint& obj) {
return !((*this < obj) | (*this > obj));
} bool operator <= (const Biguint& obj) {
return (*this) < obj || (*this) == obj;
} bool operator >= (const Biguint& obj) {
return (*this) > obj || (*this) == obj;
} Biguint operator += (const Biguint& obj) {
const int* b = obj.a;
if (obj.len > len) len = obj.len;
for (int i = 0; i < len; i++) {
a[i] += b[i];
if (a[i] >= 10) a[i + 1] += a[i] / 10, a[i] %= 10;
}
if (a[len]) ++len;
while (a[len - 1] >= 10)
a[len] += a[len - 1] / 10, a[len - 1] %= 10, ++len;
return *this;
} Biguint operator + (const Biguint& obj) {
Biguint ret;
ret += *this;
ret += obj;
return ret;
} Biguint operator -= (const Biguint& obj) {
const int* b = obj.a;
for (int i = 0; i < len; i++) {
a[i] -= b[i];
if (a[i] < 0) a[i + 1]--, a[i] += 10;
}
while (a[len - 1] == 0 && len > 0) --len;
return *this;
} Biguint operator -(const Biguint& obj) {
Biguint ret;
ret += *this;
ret -= obj;
return ret;
} Biguint operator *= (int b) {
for (int i = 0; i < len; i++)
a[i] *= b;
for (int i = 0; i < len; i++)
a[i + 1] += a[i] / 10, a[i] %= 10;
++len;
while (a[len - 1] >= 10)
a[len] += a[len - 1] / 10, a[len - 1] %= 10, ++len;
while (a[len - 1] == 0 && len > 0) --len;
return *this;
} Biguint operator * (int b) {
Biguint ret;
ret = *this;
ret *= b;
return ret;
} Biguint operator * (const Biguint& obj) {
const int* b = obj.a;
Biguint ret;
for (int i = 0; i < len; i++)
for (int j = 0; j < obj.len; j++)
ret.a[i + j] += a[i] * b[j];
for (int i = 0; i < len + obj.len; i++)
ret.a[i + 1] += ret.a[i] / 10, ret.a[i] %= 10;
ret.len = len + obj.len;
++ret.len;
while (ret.a[ret.len - 1])
ret.a[ret.len] += ret.a[ret.len - 1] / 10, ret.a[ret.len - 1] %= 10, ++ret.len;
while (ret.a[ret.len - 1] == 0 && ret.len > 0) --ret.len;
return ret;
} }; ostream& operator << (ostream& os, Biguint num)
{
for (int i = num.len - 1; i >= 0; --i)
os << num.a[i];
if (num.len == 0) os << "0";
return os;
} istream& operator >> (istream& is, Biguint& num)
{
string str;
is >> str;
memset(num.a, 0, sizeof num.a);
num.len = str.length();
for (int i = 0; i < str.length(); i++)
num.a[i] = str[str.length() - i - 1] - '0';
return is;
} const int N = 500005;
const int p[21] = {0, 2, 3, 5, 7, 11, 13, 17, 19, 23, 29,
31, 37, 41, 43, 47, 53, 59, 61, 67, 71};
int n,g[50005][21],h[50005][21],ii[50005],top;
double lp[21]={},f[50005][21]; signed main() {
ios::sync_with_stdio(false);
cin>>n;
for(int i=1;i<=n;i++) if(n%i==0) ii[++top]=i;
for(int i=1;i<=20;i++) lp[i]=log(p[i]);
for(int i=0;i<=50000;i++) for(int j=0;j<=20;j++) f[i][j]=1e9;
f[1][0]=1;
for(int _i=2;_i<=top;_i++) {
int i=ii[_i];
int sq=sqrt(i);
for(int j=1;j<=20;j++) {
for(int k=1;k<=sq;k++) {
if(i%k==0) {
if(f[i][j]>f[i/k][j-1]+(k-1)*lp[j]) {
f[i][j]=f[i/k][j-1]+(k-1)*lp[j];
g[i][j]=i/k;
h[i][j]=k-1;
}
}
}
for(int u=1;u<=sq;u++) {
int k=i/u;
if(i%k==0) {
if(f[i][j]>f[i/k][j-1]+(k-1)*lp[j]) {
f[i][j]=f[i/k][j-1]+(k-1)*lp[j];
g[i][j]=i/k;
h[i][j]=k-1;
}
}
}
}
}
int pos=n;
Biguint ans;
ans.len=1;
ans.a[0]=1;
for(int j=20;j;--j) {
int i=pos;
for(int k=1;k<=h[i][j];k++) ans*=p[j];
pos=g[i][j];
}
cout<<ans;
}

[HNOI2001] 求正整数 - 背包dp,数论的更多相关文章

  1. BZOJ 1225: [HNOI2001] 求正整数( dfs + 高精度 )

    15 < log250000 < 16, 所以不会选超过16个质数, 然后暴力去跑dfs, 高精度计算最后答案.. ------------------------------------ ...

  2. 高精度+搜索+质数 BZOJ1225 [HNOI2001] 求正整数

    // 高精度+搜索+质数 BZOJ1225 [HNOI2001] 求正整数 // 思路: // http://blog.csdn.net/huzecong/article/details/847868 ...

  3. bzoj1225 [HNOI2001] 求正整数

    1225: [HNOI2001] 求正整数 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 762  Solved: 313[Submit][Statu ...

  4. luogu P1128 [HNOI2001]求正整数 dp 高精度

    LINK:求正整数 比较难的高精度. 容易想到贪心不过这个贪心的策略大多都能找到反例. 考虑dp. f[i][j]表示前i个质数此时n的值为j的最小的答案. 利用高精度dp不太现实.就算上FFT也会T ...

  5. 【BZOJ】1225: [HNOI2001] 求正整数

    http://www.lydsy.com/JudgeOnline/problem.php?id=1225 题意:给一个数n,求一个最小的有n个约数的正整数.(n<=50000) #include ...

  6. [HNOI2001]求正整数

    题目描述 对于任意输入的正整数n,请编程求出具有n个不同因子的最小正整数m. 例如:n=4,则m=6,因为6有4个不同整数因子1,2,3,6:而且是最小的有4个因子的整数. 输入输出格式 输入格式: ...

  7. BZOJ 1225: [HNOI2001] 求正整数 高精度+搜索+质数

    题意:给定n求,有n个因子的最小正整数. 题解:水题,zcr都会,我就不说什么了. 因数个数球求法应该知道,将m分解质因数,然后发现 a1^p1*a2^p2....an^pn这样一个式子, (1+p1 ...

  8. P1128 [HNOI2001]求正整数

    传送门 rqy是我们的红太阳没有它我们就会死 可以考虑dp,设\(dp[i][j]\)表示只包含前\(j\)个质数的数中,因子个数为\(i\)的数的最小值是多少,那么有转移方程 \[f[i][j]=m ...

  9. BZOJ1222: [HNOI2001]产品加工(诡异背包dp)

    Time Limit: 15 Sec  Memory Limit: 162 MBSubmit: 907  Solved: 587[Submit][Status][Discuss] Descriptio ...

随机推荐

  1. WPF 原生绑定和命令功能使用指南

    WPF 原生绑定和命令功能使用指南 魏刘宏 2020 年 2 月 21 日 如今,当谈到 WPF 时,我们言必称 MVVM.框架(如 Prism)等,似乎已经忘了不用这些的话该怎么使用 WPF 了.当 ...

  2. MySQL锁与事务隔离级别

    一.概述 1.锁的定义 锁是计算机协调多个进程或线程并发访问某一资源的机制. 在数据库中,除了传统的计算资源(如CPU.RAM.IO等)的争用以外,数据也是一种供需要用户共享的资源.如何保证数据并发访 ...

  3. Vue整合d3.v5.js制作--柱状图(rect)

    先上效果图: 图中柱状图变成纯蓝色是鼠标滑动过的颜色(颜色可改,本人配色能力十分的强,建议直接用默认设置即可 ( ᖛ ̫ ᖛ )ʃ)) 1.环境说明 Vue版本:"vue": &q ...

  4. Git操作:绑定上传已存在的仓库到Github

    之前使用github都是创建一个全新的仓库,然后clone下来用,但如果我已经有一个正在使用的仓库,想要绑定上传已存在的仓库到github,怎么做呢?其实在github创建仓库的时候会提示: …or ...

  5. 算法训练 最大获利 注意数据规模(long long)

    资源限制 时间限制:1.0s   内存限制:256.0MB 问题描述 Chakra是一位年轻有为的企业家,最近他在进军餐饮行业.他在各地开拓市场,共买下了N个饭店.在初期的市场调研中,他将一天划分为M ...

  6. mysql必知必会--过 滤 数 据

    使用 WHERE 子句 数据库表一般包含大量的数据,很少需要检索表中所有行.通常只 会根据特定操作或报告的需要提取表数据的子集.只检索所需数据需要 指定搜索条件(search criteria),搜索 ...

  7. EPEL添加与删除

    EPEL简介 EPEL的全称叫 Extra Packages for Enterprise Linux,由Fedora社区打造,如它的全称,这是一个为红帽系列及衍生发行版如CentOS.Fedora提 ...

  8. Android(安卓)全套开发资料视频+源码

    最近看到这么一张图,我觉得对于IT界的人来说应该很有感触. 也许这意味着今年是996的元年吧,但是那又怎么样?即便它虐我们千百遍,我们还是得微笑着面对它.So,今天分享一些整理的Android开发相关 ...

  9. 【46】谷歌 Inception 网络简介Inception(2)

    Inception 网络(Inception network) 在上节笔记中,你已经见到了所有的Inception网络基础模块.在本节笔记中,我们将学习如何将这些模块组合起来,构筑你自己的Incept ...

  10. windows下XAMPP集成环境中,MySQL数据库的使用

    https://jingyan.baidu.com/article/d169e186467a44436611d8b1.html