洛谷 CF448D Multiplication Table

2019-09-04 07:07:37来源:博客园 阅读 ()

新老客户大回馈,云服务器低至5折

洛谷 CF448D Multiplication Table

目录

  • 题目
  • 思路
  • $Code$

题目

CF448D Multiplication Table

思路

二分答案。这个矩阵的每一排都是递增的,所以二分$ans$,去计算有多少个数等于$ans$,有多少个数小于$ans$,如果小于$ans$的数不多于$k-1$个并且小于等于$ans$的数不少于$k$个,那么当前$ans$就是答案。
Q:如何计算小于$ans$的数的个数?
A:
$$\sum_{i=1}^{n}min(\lfloor \frac{ans-1}{i} \rfloor,m)$$
Q:如何计算等于$ans$的数的个数?
A:当$i|ans$并且$ans/i$小于$m$时个数加一。

$Code$

#include<iostream>
#include<cstring>
#include<string>
#include<algorithm>
#include<cstdio>
#define ll long long
using namespace std;
ll n,m,k;
inline void read(ll &T){
    ll x=0;bool f=0;char c=getchar();
    while(c<'0'||c>'9'){if(c=='-')f=!f;c=getchar();}
    while(c>='0'&&c<='9'){x=x*10+c-'0';c=getchar();}
    T=f?-x:x;
}
inline void write(ll x){
    if(x<0) putchar('-'),write(-x);
    else{
        if(x/10) write(x/10);
        putchar(x%10+'0');
    }
}

int main(){
    read(n),read(m),read(k);
    ll l=1,r=n*m;
    while(l<=r){
        ll mid=(l+r)>>1,sum=0,tmp=0;
        for(int i=1;i<=n;++i){
            sum+=min((mid-1)/i,m);
            if(mid%i==0&&mid/i<=m) tmp++;
        }
        if(sum<=k-1&&sum+tmp>=k){
            write(mid);
            return 0;
        }
        if(sum+tmp<=k-1) l=mid+1;
        else r=mid-1;
    }
    return 0;
}

原文链接:https://www.cnblogs.com/poi-bolg-poi/p/11450808.html
如有疑问请与原作者联系

标签:

版权申明:本站文章部分自网络,如有侵权,请联系:west999com@outlook.com
特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系,版权归原作者所有

上一篇:矩阵乘法(三):根据要求构造矩阵进行快速幂运算

下一篇:了解CAdoSqlserver