反素数 -- 数学

反素数就是区间内约数个数最多的那个数。

在ACM题目里,

一般是求约数最多而且数字最小的那个数,【1--n】

二是求约数刚好等于n的最小的那个数

三是求区间里的最小反素数【beign,end】

1和3有区别吗?有,1可以加速,3只能暴力

先说下思路

思路 : 官方题解 : 

(1)此题最容易想到的是穷举,但是肯定超时。

(2)我们可以知道,计算约数的个数和质因数分解有着很大的联系: 若Q的质因数分解为:Q=p1^k1*p2^k2*…*pm^km(p1…pm为素数,k1…km≥1),则Q有(k1+1)(k2+1)…(km+1)个约数。但是质因数分解的时间复杂度很高,所以也会超时。

(3)通过以上的公式,我们可以“突发奇想”:为何不能把质因数分解的过程反过来呢? 这个算法就是枚举每一个素数。初始时让m=1,然后从最小的素数2开始枚举,枚举因子中包含0个2、1个2、2个2…k个2,直至m*2^k大于区间的上限N。在这个基础上枚举3、5、7……的情况,算出现在已经得到的m的约数个数,同时与原有的记录进行比较和替换。直至所有的情况都被判定过了。 这个算法的优化:如果p1*p2*p3*……*pk>N(pi表示第i个素数),那么只要枚举到p k-1,既不浪费时间,也不会遗漏。

(4)以上的算法还不是最好的,还可以继续优化。 我们看以下的例子: 6=2*3 10=2*5 6和10的质因数分解“模式”完全相同,所以它们的约数个数是相同的。但是由于3<5,所以6<10。 12=2^2*3 18=3^2*2 12和18的质因数分解“模式”完全相同,所以它们的约数个数是相同的。但是由于12的质因数分解中2的指数大于3的指数,18的质因数分解中3的指数大于2的指数,所以12<18。 根据以上的举例,我们也可以对(3)中的算法进行一个改进:可以在枚举时进行一个优化,使得枚举到的数字中2的指数不小于3的指数,3的指数不小于5的指数……这样我们就能够得到质因数分解“模式”相同的最小数(证明略)。再对于每一个得到的数进行比较和记录。这个算法的优化力度极大,效率几乎达到了极限。

 

每个思路都很好理解,所以

http://codeforces.com/problemset/problem/27/E

这是签到题了,约数刚好等于n,那么最需dfs的时候判断即可,用第4这个方法的思路,time 30ms

#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <algorithm>
using namespace std;
#define inf (0x3f3f3f3f)
typedef long long int LL;#include <iostream>
#include <sstream>
#include <vector>
#include <set>
#include <map>
#include <queue>
#include <string>
LL pr, mx, BEGIN, END = 1e18;
const int maxn=50+20;
int prime[maxn];//这个记得用int,他保存的是质数,可以不用开maxn那么大
bool check[maxn];
int total;
int n;
void initprime() {for (int i=2; i<=maxn-20; i++) {if (!check[i]) { //是质数了prime[++total]=i;//只能这样记录,因为后面要用
        }for (int j=1; j<=total; j++) { //质数或者合数都进行的if (i*prime[j]>maxn-20) break;check[i*prime[j]]=1;if (i%prime[j]==0) break;//关键,使得它只被最小的质数筛去。例如i等于6的时候。//当时的质数只有2,3,5。6和2结合筛去了12,就break了//18留下等9的时候,9*2=18筛去
        }}return ;
}
void dfs(LL curnum, int cnt, int depth, int up) {if (depth > total) return ; // 越界了,用不到那么多素数if ((cnt > mx || cnt == mx && pr > curnum) && cnt == n) {pr = curnum;mx = cnt;}for (int i = 1; i <= up; ++i) { //枚举有多少个prime[depth]if (END / curnum < prime[depth]) return ;if ((BEGIN - 1) / curnum == END / curnum) return ; //区间不存在这个数的倍数curnum *= prime[depth]; //一路连乘上去dfs(curnum, cnt * (i + 1), depth + 1, i); // 2^2 * 3, 3最多2个
    }
}void work() {cin >> n;dfs(1, 1, 1, 64);cout << pr << endl;return ;
}int main() {
#ifdef localfreopen("data.txt","r",stdin);
#endifinitprime();work();return 0;
}
View Code

http://vjudge.net/problem/11177

求1--n里最小反素数,思路一样

#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <algorithm>
using namespace std;
#define inf (0x3f3f3f3f)
typedef long long int LL;#include <iostream>
#include <sstream>
#include <vector>
#include <set>
#include <map>
#include <queue>
#include <string>
LL pr, mx, BEGIN, END;
const int maxn=50+20;
int prime[maxn];
bool check[maxn];
int total;
void initprime() {for (int i=2; i<=maxn-20; i++) {if (!check[i]) { prime[++total]=i;}for (int j=1; j<=total; j++) { if (i*prime[j]>maxn-20) break;check[i*prime[j]]=1;if (i%prime[j]==0) break;}}return ;
}void dfs(LL curnum, int cnt, int depth, int up) {if (depth > total) return ; // if ((cnt > mx || cnt == mx && pr > curnum) && curnum >= BEGIN) {pr = curnum;mx = cnt;}for (int i = 1; i <= up; ++i) { if (END / curnum < prime[depth]) return ;if ((BEGIN - 1) / curnum == END / curnum) return ; //
        curnum *= prime[depth]; //
        dfs(curnum, cnt * (i + 1), depth + 1, i);}
}void work() {pr = 0, mx = 0, BEGIN = 1;cin >> END;dfs(1, 1, 1, 64);cout << pr << " " << mx << endl;return ;
}int main() {
#ifdef localfreopen("data.txt","r",stdin);
#endifinitprime();int t;cin >> t;while (t--) {work();}return 0;
}
View Code

难题是这个

http://acm.njupt.edu.cn/acmhome/problemdetail.do?&method=showdetail&id=1203

求解区间内[begin,end]最小反素数

用4思路啊,不行,为啥,因为4只关心约数个数,没考虑过区间问题,很明显2^2*3  4的思路限制了3的个数最多只能是2,

举个例子[150,150]你怎么破?150 = 2*3*5*5,这种情况,在4中是不可能的,所以只能用3了

 

首先,数据范围是n<=1e9,数据太大,如何快速算出来呢?我们注意到,如果是暴力算的话,最快的方法就是分解质因子,然后组合式计算啦。但是在算18和30的约数的时候,他们的

gcd(18,30)=6,其实是被重复算了的,那么我们思维反过来一下,把分解质因数变成用质因数去组合,使得变成区间内的数,这样一来,我们在2*3的时候,*3就得到了18,*5就得到了30,能省掉一定的时间。但是还是会TLE。假如我们现在枚举到的数是now,会不会它的约数根本就没可能存在于区间里呢?也就是[begin,end]根本就没这些约数。[7,11]内是不会存在6的倍数的。如果[1,begin-1]中6的约数和[1,end]中6的约数相同,说明什么?新加进去的区间[begin,end]根本就没6的约数,这里可以剪枝。还是TLE!!可行性剪枝,如果一个数是now,现在枚举一个新的质数去乘以它,去结合成新的数字,那么如果它无论组成什么其他数字,因子个数都不会超过当前最优值mx呢?怎么判断呢?放缩咯,假如现在是2*3,重新去匹配一个新的素数5,那么,我就要看,当前2*3还能再乘多少个3呢?我记作q,那么这个新的匹配,最理想的情况下因子个数会多2­­q­倍,为什么呢?把那些3,全部替换成5*7*11*13这样来算的话,就是有2q个了。别以为这样没用,当你搜[1,1e9]的时候,你枚举到8000w,再去枚举5那些是没用的,根本就不可能,这里能剪很多。

其实我们还有一个根本的问题没解决,那就是预处理素数到多大,还有万一它是大素数呢?

想着预处理多少,要看数据,预处理出来的最大质数,primeMax‑2是要大过1e9才行的。为什么呢?因为你只有这样,才能防止它数据是两个大质数相乘的形式[primeMax2,primeMax2]。这里的因子个数是3,你枚举不到这个primeMax的话,就只能得到2。

还有那个大素数,没什么怕的,如果当前那个数now,幻想它乘以一个大质数,还是在end的范围的话,就看看*2和mx谁大咯。乘以一个大素数也才加一倍因子数。其实乘以一个小的质因子的话,因子数会更多,这里主要是判断只有一个大素数的特殊情况。枚举不到那个大素数那里的。

#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <algorithm>
using namespace std;
#define inf (0x3f3f3f3f)
typedef long long int LL;#include <iostream>
#include <sstream>
#include <vector>
#include <set>
#include <map>
#include <queue>
#include <string>
LL pr, mx, BEGIN, END = 1e18;
const int maxn=1e6+20;
int prime[maxn];//这个记得用int,他保存的是质数,可以不用开maxn那么大
bool check[maxn];
int total;
int n;
void initprime() {for (int i=2; i<=maxn-20; i++) {if (!check[i]) { //是质数了prime[++total]=i;//只能这样记录,因为后面要用
        }for (int j=1; j<=total; j++) { //质数或者合数都进行的if (i*prime[j]>maxn-20) break;check[i*prime[j]]=1;if (i%prime[j]==0) break;//关键,使得它只被最小的质数筛去。例如i等于6的时候。//当时的质数只有2,3,5。6和2结合筛去了12,就break了//18留下等9的时候,9*2=18筛去
        }}return ;
}
LL mypow (LL a, LL b) {LL ans = 1;while (b) {if (b & 1) {ans *= a;}a *= a;b >>= 1;}return ans;
}
void dfs (int cur,int cnt,LL now,LL from) {LL t=from*(cnt+1);//现在一共拥有的因子数if (now>=BEGIN && t>mx || t==mx && pr>now && now >= BEGIN) { //有得换了mx=t;pr=now;}for (int i=cur; i<=total; ++i) { //枚举每一个素数LL temp = now*prime[i];if (END / now < prime[i]) return ; //这个数超出范围了if (i == cur) { //没有变,一直都是用这个数.2^kdfs(cur,cnt+1,temp,from);//唯一就是from没变,一直都是用着2,不是新质数} else { //枚举新质数了。LL k = (cnt+1)*from; //现在有K个因子LL q=(LL)(log(END/now) / log(prime[cur])); //2*3插入5时,用的是3来放缩LL add = k*mypow(2,q);if (add < mx) return ; //这里等于mx不return,可以输出minprif ((BEGIN-1)/now == END/now) return; //不存在now的倍数if (END/now > prime[total]) { //试着给他乘上一个大素数 [999991,999991]if ( k*2 > mx ) { //乘以一个大素数,因子数*2pr = END;//如果只有一个大素数[1e9+7,le9+7]那么,就是端点值//否则,是2*3*5*bigprime的话,结果不是最优的,mx = k*2;}}dfs(i,1,temp,k);}}return;
}void work() {cin >> BEGIN >> END;dfs(1,0,1,1);cout << mx << endl;
//    cout << pr << " " << mx << endl;return ;
}int main() {
#ifdef localfreopen("data.txt","r",stdin);
#endifinitprime();work();return 0;
}
View Code

//cur:当前枚举质数的下标,不用返回来枚举了。

//cnt:分解质因式时:拥有(当前下标那个)素数多少个

//now:当前枚举的那个数字,就是所有质因子相乘得到的数子

//from: 假如:2*2*3*5*7,然后枚举3,记录的是2*2,枚举5,记录的是2*2*3,

//如果是枚举相同的数,则不用变,因为它记录的是上一个不同的质因子一共拥有的因子数。

//所以乘上(cnt+1),就是包括上现在这个质因子一共拥有的因子数了。

dfs(1,0,1,1); //刚开始的时候,下标从1开始,拥有这个素数0个,当前数字最少也是1吧

 

转载于:https://www.cnblogs.com/liuweimingcprogram/p/5877411.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/373889.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

编程挑战系统的输入和输出详细说明

在高校俱乐部线上编程挑战中&#xff0c;一道题目的所有测试数据是放在一个文本文件中&#xff0c;选手将一道题目的程序提交给评判系统运行&#xff0c;程序从该文件中读取测试数据&#xff0c;再把运行结果输出到另一个文本文件中。系统把输出文件与标准答案比对&#xff0c;…

上传文件---未能找到路径“D:\MyProject\Files\”的一部分

C# 使用控件FileUpload 上传文件&#xff0c;简单实例&#xff1a; protected void btnUpload_Click(object sender, EventArgs e){string path Server.MapPath("~/Files/");if (fileUpload.HasFile true){string filename fileUpload.FileName.ToLower();fileUpl…

使用SPANN方式将Spring&Quartz与自定义注释集成

在上一篇文章中 &#xff0c;我们演示了如何在Spring容器中创建和配置带批注的Quartz作业。 我们使用了一个类级别的注释将一些元数据添加到实现Quartz Job的bean中。 批注定义了作业的名称&#xff0c;组及其cron表达式。 后来&#xff0c;大部分代码专用于处理该批注&#xf…

python opencv旋转_Python opencv实现与rotatedrect类似的矩形旋转,pythonopencv,RotatedRect

本文原理&#xff1a;先旋转矩形到指定角度&#xff0c;然后提取矩形外轮廓&#xff0c;从而获取旋转后的矩形坐标点。#&#xff01;/usr/bin/env python3# -*- coding: utf-8 -*-# Author: tcy# Date: 2020-5-2 21:00:53# Version:V1.01# Last Modified by: tcy shanghai song…

关于string转整数

又是leetcode的easy级别题&#xff0c;很基本的题目&#xff0c;却漏考虑很多情况&#xff0c;动手前一定要考虑清楚呀&#xff01;&#xff01;&#xff01; 就当做锻炼写作能力吧&#xff0c;先上题目&#xff01; 将文本转换成整数&#xff0c;注意一下几点&#xff1a; 1.文…

数字三角形——递归、递推、记忆化搜索

数字三角形 描述: 有一个由非负整数组成的三角形&#xff0c;第一行只有一个数&#xff0c;除了最下行之外每个数的左下方和右下方各有一个数。 问题&#xff1a; 从第一行的数开始&#xff0c;每次可以往左下或右下走一格&#xff0c;直到走到最下行…

Java 7功能概述

前面我们讨论了所有未纳入Java 7的内容&#xff0c;然后回顾了将其纳入Java 7的有用的Fork / Join框架 。 今天的帖子将带我们了解Project Coin的每个功能-一系列小的语言增强功能&#xff0c;这些功能虽然不是开创性的&#xff0c;但是对于任何能够使用JDK 7的开发人员来说都是…

缓存技术

提升系统性能的主要方式之一就是缓存。它可以挡掉大部分的数据库访问的冲击&#xff0c;如果没有它&#xff0c;系统很可能会因为数据库不可用导致整个系统崩溃。 但是缓存带来了另外一些棘手的问题&#xff1a; 数据的一致性和实时性。 例如&#xff0c;数据库中的数据状态已经…

水晶报表分组分栏_web报表可视化设计器工具推荐

古往今来&#xff0c;信息就是决胜的关键。在科技时代的今天亦是如此。企业的数据管理在帮助企业加强管控、提高竞争力等方面具有不可或缺的作用。这就不得不说到报表工具。企业想要将储存于各种商业信息系统中的数据转化成有用的信息&#xff0c;最终帮助决策者做出更快、更好…

嵌套矩形——DAG上的动态规划

有向无环图&#xff08;DAG,Directed Acyclic Graph&#xff09;上的动态规划是学习动态规划的基础。很多问题都可以转化为DAG上的最长路、最短路或路径计数问题。 题目描述&#xff1a; 有n个矩形&#xff0c;每个矩形可以用两个整数a,b描述&#xff0c;表示它的长和宽。矩形…

Twisted

Twisted定义Twisted是一个基于事件驱动的网络引擎框架网络框架&#xff0c;别人预先定义好的一个框架&#xff08;一个项目&#xff09;&#xff0c;如.net某个web框架有25个class&#xff0c;从BeginRequest依次执行类里的process方法&#xff0c;程序员自己定义一个类&#x…

从Spring到Java EE 6

我最近在一个非常复杂的项目中工作&#xff0c;其中融合了许多Java EE 6技术&#xff08;例如JPA&#xff0c;JAXB&#xff0c;JMS&#xff0c;JTA&#xff0c;JAX-RS等&#xff09;。 出于生产力和计划方面的原因&#xff0c;将原型应用程序设计为独立的纯Spring应用程序。 当…

Centos 6.5 搭建php环境(nginx+mariadb+php7)

1.mariaDb vim /etc/yum.repos.d/MariaDB.repo [mariadb] name MariaDB baseurl http://yum.mariadb.org/5.5/centos5-x86 gpgkeyhttps://yum.mariadb.org/RPM-GPG-KEY-MariaDB gpgcheck1#如果服务器已经安装了MariaDB-Galera-server包&#xff0c;你可能需要在安装MariaDB-s…

MAC itunes无法验证服务器s.mzstatic/itunes无法更新服务器解决方案

打开host文件&#xff1a; 一、用终端打开&#xff1a; sudo vi /etc/hosts 输入完这行命令后需要输入电脑密码&#xff0c;然后确认&#xff0c;进入host文件 然后按i键进入编辑模式&#xff0c;在最后一行添加&#xff1a;23.214.233.166 s.mzstatic.com 如下图 添加完后&…

硬币问题——固定终点的最长路和最短路

问题描述&#xff1a; 有n种硬币&#xff0c;面值分别为V1,V2...,Vn,每种都有无限多。给定非负整数S&#xff0c;可以选用多少个硬币&#xff0c;使得面值之和恰好为S&#xff1f;输出硬币数目的最小值和最大值。0 < n < 100, 0 < S < 10000, 1 < Vi < S。 …

读取nas_NAS怎么玩?除了存放小姐姐,它竟然还有这些功能

自从有了电脑&#xff0c;就一直在折腾"存储那点事儿"&#xff0c;说到底&#xff0c;电脑的本质就是存储&#xff0c;而自己弄家用存储方面的东西算下来也有几年了。单机的硬盘存储比较简单&#xff0c;但是随着家里各种设备的增多&#xff0c;各个设备间的文件共享…

ZK Web框架思想

我曾多次被要求提出一些有关ZK的意见。 因此&#xff0c;根据我作为ZK用户4年的经验&#xff0c;以下是一些想法&#xff1a; 总体开发人员经验&#xff0c;社区和文档 “就是这样” ZK提供的大多数东西都能很好地工作&#xff0c;并且如果您以前开发过任何桌面Java应用程序&…

OC第一讲:类和对象

今天终于开始进行OC的学习了 一.首先讲了NSLog NSLog是oc里面的输出语句&#xff0c;其用法和printf差不多&#xff0c;但是还是有差别的 1&#xff0c;NSLog是自动换行的&#xff0c;不用像printf那样还需要加\n&#xff1b; 2&#xff0c;NSLog在引号面前需要添加符号&#x…

【转载】关于 Google Chrome 中的全屏模式和 APP 模式

【来源于】新浪微博&#xff1a;阿博 http://www.cnblogs.com/abel/p/3235839.html 全屏模式&#xff1a;kiosk 默认全屏打开一个网页呢&#xff0c;只需要在快捷方式中加上 --kiosk [url] 就可以了。 关于全屏模式&#xff1a; 1、全屏模式下&#xff0c;广告插件&#xff08;…

PL/SQL Developer跑在Oracle 64位数据库上初始化错误

安装完Oracle(64位)、PL/SQL Developer后运行PL/SQL出现如下的错误&#xff1a; 网上查资料说&#xff0c;我的PL/SQL Developer与ORACLE不兼容&#xff0c;即PL/SQL不支持64位的ORACLE&#xff0c;因此得下一个32位的ORCALE客户端并配置相应的参数&#xff1a; 解决步骤小记&a…