打表或者画个图可以看出i>根号n时每个i的贡献值相差很小,可以利用公式优化(函数C)

但是注意不能一整段使用公式,否则复杂度还是会劣化到O(n)(显然对gongxian只能逐步递减)

网上看了不少代码,但是都没有对贡献值边界问题给定明确的判断

所以还是加多一个while循环确定贡献值的开端是前面的n/i没有的

#include<bits/stdc++.h>
using namespace std;
const int maxn = 1e5+11;
typedef long long ll;
ll H(ll n,ll j){
ll ans=0;
for(int i = 1; i <= j; i++) ans+=n/i;
return ans;
}
ll C(ll n,ll gongxian){
return gongxian*(n/gongxian-n/(gongxian+1));
}
int main(){
int T,kase=0; scanf("%d",&T);
while(T--){
ll n; scanf("%lld",&n);
if(n<=20){
printf("Case %d: %lld\n",++kase,H(n,n));
continue;
}
ll gen = sqrt(n);
ll ans1 = H(n,gen);
ll tmp = n/gen;
ll cnt=gen;
while(1){
if(n/gen==n/(cnt+1)){
ans1+=n/gen;
cnt++;
}
else{
cnt++;//start
break;
}
}
ll gongxian = n/cnt;
ll ans2=0;
while(gongxian){
ans2+=C(n,gongxian);
gongxian--;
}
printf("Case %d: %lld\n",++kase,ans1+ans2);
}
return 0;
}

顺便附加对规律观察用的代码

#include<bits/stdc++.h>
#define rep(i,j,k) for(int i = j; i <= k; i++)
#define scan(a) scanf("%d",&a)
using namespace std;
typedef pair<int,int> P;
const int maxn = 1e6+11;
typedef long long ll;
int cnt[maxn],n;
bool C(int n){
memset(cnt,0,sizeof cnt);
rep(i,1,n) cnt[n/i]++;
int maxzero=0,num=0;
rep(i,1,n) if(i>=sqrt(n)&&cnt[i]==1) num++;
// cout<<(int)sqrt(n+0.5)+1<<" "<<cnt[(int)sqrt(n+0.5)+1]<<endl;
// return cnt[(int)sqrt(n+0.5)+1];
cout<<num<<endl;
}
int main(){
// rep(i,1,10000) if(C(i)) cout<<i<<endl;
srand(time(NULL));
int n=rand()%1000000;
C(n);cout<<sqrt(n)<<endl;
}

可以看出[根号n,n]对答案的贡献基本等于根号n

#include<bits/stdc++.h>
using namespace std;
int main(){
int cnt=0;
for(int i = 100; i <= 1e6; i++){
int a=i/(int)sqrt(i),b=sqrt(i);
if(a-b<0) cnt++;
// cout<<i<<" "<<a-b<<endl; }
cout<<cnt<<endl;
}

可以看出a(应该)都是大于等于b

待做同类题目:1257 2956

//BACKUP
#include<bits/stdc++.h>
using namespace std;
const int maxn = 1e5+11;
typedef long long ll;
ll cal(int n,int k){
ll ans=0;
for(int i = 1; i <= n; i++) ans+=k%i;//k-(k/i)*i
return ans;
}
ll G(int n,int k){
ll ans=0;
for(int i = 1; i <= n; i++) ans+=(k/i)*i;
return ans;
}
ll con(ll k,ll i){return i*(k/i-k/(i+1));}
ll sum(ll a1,ll n,ll d){return a1+(n-1)*d;}
int main(){
int n,k;
while(cin>>n>>k){
if(n<=100){
cout<<cal(n,k)<<endl;
continue;
}
ll root=sqrt(n);
// cout<<"root"<<root<<endl;
ll ans1=1ll*k*n;
ll ans2=G(root,k); //
ll ans3=0;
int cnt=root+1; if(k/cnt==0){//防止越界处理
cout<<ans1-G(root,k)<<endl;
continue;
}
while(1){
if(k/root==k/(cnt+1)){
ans3+=(k/(cnt+1))*(cnt+1);
cnt++;
// cout<<k/root<<" "<<k/cnt<<endl;
}
else{
cnt++;
break;
}
}
bool flag=0;
ll lim=k/n; if(lim==0) lim++;
ll val=k/cnt; if(val==0) flag=1;
ll now=cnt;
ll ans4=0;
while(!flag&&val>=lim){
cout<<"NOW "<<now<<endl;
ll LEN = k/val-k/(val+1);
cout<<"LEN "<<LEN<<endl;
cout<<"VAL "<<val<<endl;
ans4+=con(k,val)*(LEN?sum(now,LEN,1):0);
now+=LEN;
val--;
}
cout<<ans1-(ans2-ans3-ans4)<<endl;
}
return 0;
}

LightOJ - 1245 根号n暴力的更多相关文章

  1. LightOJ 1245 数学题,找规律

    1.LightOJ 1245   Harmonic Number (II)   数学题 2.总结:看了题解,很严谨,但又确实恶心的题 题意:求n/1+n/2+....+n/n,n<=2^31. ...

  2. LightOJ 1245 Harmonic Number (II)(找规律)

    http://lightoj.com/volume_showproblem.php?problem=1245 G - Harmonic Number (II) Time Limit:3000MS    ...

  3. lightoj 1245

    lightoj 1245 Harmonic Number (II) 题意:给定一个 n ,求 n/1 + n/2 + …… + n/n 的值(这里的 "/" 是计算机的整数除法,向 ...

  4. LightOJ - 1234 LightOJ - 1245 Harmonic Number(欧拉系数+调和级数)

    Harmonic Number In mathematics, the nth harmonic number is the sum of the reciprocals of the first n ...

  5. LightOJ 1245 - Harmonic Number (II)

    题目链接:http://www.lightoj.com/volume_showproblem.php?problem=1245 题意:仿照上面那题他想求这么个公式的数.但是递归太慢啦.让你找公式咯. ...

  6. LightOj 1245 --- Harmonic Number (II)找规律

    题目链接:http://lightoj.com/volume_showproblem.php?problem=1245 题意就是求 n/i (1<=i<=n) 的取整的和这就是到找规律的题 ...

  7. Harmonic Number (II) LightOJ - 1245 (找规律?。。。)

    题意: 求前n项的n/i  的和 只取整数部分 暴力肯定超时...然后 ...现在的人真聪明...我真蠢 觉得还是别人的题意比较清晰 比如n=100的话,i=4时n/i等于25,i=5时n/i等于20 ...

  8. lightoj 1245 Harmonic Number (II)(简单数论)

    题目链接:http://www.lightoj.com/volume_showproblem.php?problem=1245 题意:求f(n)=n/1+n/2.....n/n,其中n/i保留整数 显 ...

  9. LightOJ 1245 Harmonic Number (II) 水题

    分析:一段区间的整数除法得到的结果肯定是相等的,然后找就行了,每次是循环一段区间,暴力 #include <cstdio> #include <iostream> #inclu ...

随机推荐

  1. 217. Contains Duplicate数组重复元素 123

    [抄题]: Given an array of integers, find if the array contains any duplicates. Your function should re ...

  2. mongo状态查看方法

    列举一些常用的mongodb状态查看方法. 1.mongostat 是mongdb自带的状态检测工具, inserts/s 每秒插入次数 query/s 每秒查询次数 update/s 每秒更新次数 ...

  3. Ubuntu14.04(64位)下gcc-linaro-arm-linux-gnueabihf交叉编译环境搭建

    1. 下载 gcc-linaro-arm-linux-gnueabihf-4.9.tar.gz 下载地址参考:http://blog.csdn.net/lg1259156776/article/det ...

  4. jQuery--后台主机列表编辑

    先看效果: 要求: 全选,反选和取消 编辑模式下的全选,反选和取消 编辑模式下单选进入编辑状态,取消退出编辑状态 表格元素有可编辑,不可编辑,下拉选择 按住ctrl选择下拉框,下面的同列选项都随之改变 ...

  5. 7. Smali基础语法总结

    最近在学习Android 移动安全逆向方面,逆向首先要看懂代码,Android4.4之前一直使用的是 Dalivk虚拟机,而Smali是用于Dalivk的反汇编程序的实现. Smali 支持注解,调试 ...

  6. oracle数据库基本操作

    我们主要学习数据库的一些基本操作,比如如何在数据库创建用户,授权,删除用户,回收权限,为用户加锁或者解锁等一些常用的操作. 首先,我们要知道数据库中创建用户的语句怎么写,看下面: 1.创建用户 cre ...

  7. 编写高质量代码改善C#程序的157个建议——建议4: TryParse比Parse好

    建议4: TryParse比Parse好 如果注意观察除string外的所有基元类型,会发现它们都有两个将字符串转型为本身的方法:Parse和TryParse.以类型double为例,这两个方法最简单 ...

  8. telerik 值得学习 web mvc 桌面 控件大全

    http://www.telerik.com http://www.dxper.net/forum-70-1.html http://www.dxper.net/forum.php

  9. 如何使用 channel

    如何使用 Channel 例子来自于Concurrency is not parallelism Google Search: A fake framework v1.0 var ( Web = fa ...

  10. day07-ip地址管理

    1. ip地址由网络地址位与主机地址位两部分构成. 如:172.16.45.10/16中网路地址为172.16.0.0.主机地址为172.16.45.10. 2. 查看系统网卡信息:ifconfig: ...