博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
解题报告 之 HDU5288 OO' s Sequence
阅读量:6371 次
发布时间:2019-06-23

本文共 2549 字,大约阅读时间需要 8 分钟。

解题报告 之 HDU5288 OO' s Sequence

Description

OO has got a array A of size n ,defined a function f(l,r) represent the number of i (l<=i<=r) , that there's no j(l<=j<=r,j<>i) satisfy a 
imod a 
j=0,now OO want to know
 

Input

There are multiple test cases. Please process till EOF. 
In each test case: 
First line: an integer n(n<=10^5) indicating the size of array 
Second line:contain n numbers a 
i(0<a 
i<=10000) 
 

Output

For each tests: ouput a line contain a number ans.
 

Sample Input

 
5 1 2 3 4 5
 

Sample Output

 
23
 
题目大意:给你n个数的序列(当中数可能反复)。对于每个子区间,求该子区间中没有因子也在该子区间的的个数。再把全部子区间内这种数的数量求和。比方例子中的 1 2 3 4 5,那么子区间[1,2,3]中这种数就是1。2,3三个。然后对于子区间2 3 4,这种数就仅仅有两个,由于4有因子2也在该子区间中。

分析:非常自然的想法是遍历每个子区间,再统计有多少个数,再加起来。但这样做是不可行的。这样就中了题目的陷阱,被那个公式给误导了,所以我们必需要跳出惯性思维。将关注的单位从子区间变到每个数。
考虑一个数。它能被统计多少次取决于什么呢?取决于它在多少个子区间内可以做到没有因子。所以我们非常自然的去关注离他近期的左右两个因子。由于这两个因子范围以外的子区间都没有卵用。

。比方5 5 2 3 3 4 3 2 5 5。那么对于4来说,我们找到左右两个因子2之后,就能够发现从5開始和结束的子区间都不会算到4。由于有2在那里杵着。

至此,问题转化为,找到每个数左右离它近期的因子。然后就行非常easy的知道这个数可以被统计多少次了。那么怎么去寻找左右两边的因子呢?有两种做法,首先介绍我的方法。注意到可能的数字一共仅仅有1e4个。
先从左到右扫描依次更新两个数据,一是这个数最后出现的位置,用loc数组表示,还有一个是这个数左边离它近期的因子的位置则用该数的每个因子(遍历),求全部因子中最后出现的最大值。
然后再从右到左扫描,原理一样的。完毕之后
再遍历序列,对于每个数求它被统计多少次就可以。
#include
#include
#include
#include
#include
#include
using namespace std;typedef long long ll;const int MAXN = 1e5 + 10;const int MAXM = 1e4 + 10;const int INF = 0x3f3f3f3f;const int MOD = 1e9 + 7;int nums[MAXN]; //序列中的数int lb[MAXN], rb[MAXN]; //序列中的数左右离他近期的因子的位置int latest[MAXM];//某个数字最后出现的位置int main(){ int n; while(scanf( "%d", &n ) == 1) { memset( lb, 0, sizeof lb ); memset( rb, INF, sizeof rb ); //reset for(int i = 1; i <= n; i++) { scanf( "%d", &nums[i] ); }//input for(int i = 0; i < MAXM; i++) latest[i] = 0; for(int i = 1; i <= n; i++) { for(int j = 1; j <= sqrt( nums[i] ); j++) {//遍历每一个因子 if(nums[i] % j == 0) { lb[i] = max( lb[i], latest[j] ); lb[i] = max( lb[i], latest[nums[i] / j] ); } } latest[nums[i]] = i; //更新位置。注意要遍历后更新,由于本身也是自己的因子 }// tackle 1 for(int i = 0; i < MAXM; i++) latest[i] = n + 1; for(int i = n; i >= 1; i--) { for(int j = 1; j <= sqrt( nums[i] ); j++) { if(nums[i] % j == 0) { rb[i] = min( rb[i], latest[j] ); rb[i] = min( rb[i], latest[nums[i] / j] ); } } latest[nums[i]] = i; }// tackle 2 同理 ll ans = 0; for(int i = 1; i <= n; i++) { ans = (ans + (i - lb[i])*(rb[i] - i)) % MOD; //统计序列中每一个数被统计的次数。能够理解为范围内左边选一个数的选法*右边选一个数的选法。 } printf( "%lld\n", ans ); } return 0;}
另一种方法是,记录每一个数字出现的位置,每次更新的时候用二分去找距离它近期的因子的位置。可是非常麻烦也更慢。

转载地址:http://bauqa.baihongyu.com/

你可能感兴趣的文章
地图点聚合优化方案
查看>>
Google Chrome 快捷方式
查看>>
备考PMP心得体会
查看>>
vue proxy匹配规则
查看>>
线上应用故障排查之一:高CPU占用
查看>>
Extend Volume 操作 - 每天5分钟玩转 OpenStack(56)
查看>>
IronPython教程
查看>>
squid via检测转发循环
查看>>
计算分页
查看>>
iptables 做nat路由器脚本
查看>>
数据结构(C语言版)第三章:栈和队列
查看>>
Stopping and/or Restarting an embedded Jetty in...
查看>>
Oracle存储过程中的数据集输入参数
查看>>
vsftp 配置
查看>>
VCSA中配置时间和时区,实测至6.5适用
查看>>
高并发IM系统架构优化实践
查看>>
产品经理教你玩转阿里云负载均衡SLB系列(一):快速入门--什么是负载均衡
查看>>
有关linux--进程组、会话、守护进程详解
查看>>
我的友情链接
查看>>
monkeyrunner运行Python脚本来检查apk渠道和验证是否可以调用微信
查看>>