BZOJ 2165 大楼

BZOJ 2165 大楼

​ 题目连接:https://remmina.github.io/BZPRO/JudgeOnline/2165.html


​ 很明显,答案是具有单调性的。如果乘坐次数少了肯定不行,多了肯定行。我们第一反应一定是二分。但是这道题并不是二分。因为我们如果二分答案的话,那为什么不直接一步一步做上去,每做一次就检查一下答案是否满足,这样甚至时间复杂度是优于二分的。那么我们就想暴力,问题就变成了怎么加速暴力。答案具有单调性,而且需要我们去加速。显然倍增的优化就呼之欲出了。确定了基本想法。下面的问题就是怎么去倍增。

​ 由于我们可以到达 \(n\) 个房间,每做一次的决策也是多样化的。如果只是预处理 w[x] 数组表示从 1 开始走 \(2^x\) 步能到达的最高高度。我们发现,由于决策的多样化,我们并不好处理出这样的数组。注意到既然决策是多种多样的,我们何不将所有的决策都倍增出来?注意到 \(n\) 的值域并不大,只有 100。我们可以用矩阵来表示每一种决策所导致的结果。具体点说,我们让一个矩阵 v[i][j] 表示从 \(i\)\(j\)最大权值。如果不能到达的话 v[i][j]=-1。我们设 w[i] 为一个矩阵他表示的是最多走\(2^i\)步所能产生的矩阵。转移式为:

\[w_i=w_{i-1}\times w_{i-1} \]

​ 乘法是什么意思,这里的乘法并不是真正矩阵的乘法,而是一种类似于 floyd 的思想。我们定义矩阵的乘法如下。

\[a\times b=res,res_{i,j}=\max(res_{i,j},a_{i,k}+b_{k,j})\;(1\le i,j,k\le n,a_{i,k}\not=-1,b_{k,j}\not=-1) \]

​ 上界问题,我们就一直做 \(w_i=w_{i-1}\times w_{i-1}\) 这个操作。直到 \(w_{i}\) 满足条件。(在本题中满足条件的矩形 \(v\)\(max(v_{1,i})\ge m\;(1\le i\le n)\))然后我们运用 倍增LCA 的思想,从大到小枚举 \(i\)。如果目前的矩阵 \(tmp\times w_i\) 不满足条件,我们就让 \(tmp\) 乘上一个 \(w_i\)。然后 \(ans\) 加上一个 \(2^i\)。最后再让 \(ans+1\)。总的来说,时间复杂度为 \(O(n^3\times up)\)\(up\) 表示上界,最大不会超过 \(\log_2(10^{18})\) 。详情见代码。

代码如下:

#include<bits/stdc++.h>
#define ll long long
#define R register
using namespace std;
const int MAXN = 105;
int n;
ll m;
struct matrix
{
    ll v[MAXN][MAXN];
    matrix()
    {
        memset(v,-1,sizeof v);
    }
    matrix operator *(const matrix &x)
    {
        matrix ans;
        for(R int k=1;k<=n;++k) for(R int i=1;i<=n;++i) for(R int j=1;j<=n;++j)
            if(v[i][k]!=-1&&x.v[k][j]!=-1)
                ans.v[i][j]=max(v[i][k]+x.v[k][j],ans.v[i][j]);
        for(int i=1;i<=n;++i)
            for(int j=1;j<=n;++j)
                ans.v[i][j]=min(m,max(ans.v[i][j],max(v[i][j],x.v[i][j])));
        return ans;
    }
}w[61];
bool judge(matrix a)
{
    for(R int i=1;i<=n;++i)
        if(a.v[1][i]>=m) return true;
    return false;
}
int main()
{
    int T;
    scanf("%d",&T);
    while(T--)
    {
        scanf("%d %lld",&n,&m);
        for(R int i=1;i<=n;++i)
            for(R int j=1;j<=n;++j)
            {
                scanf("%lld",&w[0].v[i][j]);
                if(!w[0].v[i][j]) w[0].v[i][j]=-1;
            }
        ll up=1,ans=0;
        for(int i=1;;++i,++up)
        {
            w[i]=w[i-1]*w[i-1];
            if(judge(w[i])) break;
        }
        for(R int i=1;i<=up;++i)
            w[i]=w[i-1]*w[i-1];
        bool flag=0;matrix tmp;
        for(int i=up;i>=0;--i)
        {
            matrix res;
            if(flag) res=tmp*w[i];
            else res=w[i];
            if(!judge(res))
            {
                tmp=res;
                flag=1;
                ans+=(1ll<<(ll)(i));
            }
        }
        printf("%lld\n",ans+1);
    }
    return 0;
}

文末提一嘴,校内OJ上卡常,所以这个代码跑了4400ms(4000ms为AC),只有80分,理论上是能过的,所以在OJ上开了O2。不开O2的代码在这份的基础上卡卡常就能过,我比较懒所以开了O2

posted @ 2021-07-01 19:48  夜空之星  阅读(39)  评论(0编辑  收藏  举报