解题思路:

注意事项:

参考代码:

#include<bits/stdc++.h>

/*

其实这一题自己最大的收获是数学等价,来回跳2x次等价于单向跳2x次,证明:

假设可以来回跳2x次,那么按回来的跳的石头正着跳就行,=>可以单向跳2x次

假设可以单向跳2x次,那么从对岸按偶数次的跳的石头来跳就行=>可以来回跳2x次

所以可以来回跳2x次<=>可以单向跳2x次


这道题自己暴力的算法70分,时间复杂度为O(logn*(n+x*n))=O(logn*n*x)

优化:

1.使用了并查集;2.改变了一个青蛙一个青蛙跳的思想,而是一起跳;时间复杂度为O(n*logn),new online judge能得100分

*/

using namespace std;

#define maxn 100010

#define maxx 1e9+10

int n,x;//n是河的宽度,x是天数,显然石头数为n-1

int pare[maxn];

int h[maxn],tmph[maxn];

int fnum[maxn];//记录每块石头上的青蛙数

int findroot(int a){

    if(a!=pare[a]){

        pare[a]=findroot(pare[a]);//压缩路径

    }

    return pare[a];

}

void Union(int x,int y){

    y=findroot(y);

    x=findroot(x);

    pare[y]=x;//以x为根

}

bool isok(int abi){//ability

    for(int i=1;i<=n-1;i++){

        tmph[i]=h[i];

    }

    for(int i=0;i<=n;i++){

        pare[i]=i;

    }

    memset(fnum,0,sizeof(fnum));

    fnum[0]=x;//出发处有x只青蛙

    for(int i=0;i<=n;i++){

        if(i+abi>=n){

            return 1;//所有青蛙都能到对岸

        }

        while(fnum[i]){//一直跳到前i块石头没有青蛙,也就是第i块石头没有青蛙,因为i是从0开始循环的

            int root=findroot(i+abi);

            if(root<=i)return 0;//已经没有能跳的石头

            int jumpnum=min(tmph[root],fnum[i]);

            fnum[i]-=jumpnum;

            tmph[root]-=jumpnum;//跳上去就高度下降和跳走后高度下降本质上一样,高度都表示了这个石头的承载量

            fnum[root]+=jumpnum;

            if(!tmph[root]){

                pare[root]=findroot(root-1);//Union

            }

        }

    }

    /* for(int i=1;i<=x;i++){

        int pos=0;

        int minpos=0;

        while(pos<n){//当还没到终点时

            int flag=0;

            int tmp=pos;

            for(int j=pos+abi;j>minpos;j--){

                if(j>=n||tmph[j]){

                    tmph[pos]--;

                    pos=j;

                    flag=1;

                    break;

                }

            }

            minpos=tmp+abi;//小于minpos的地方不会跳到,这就是一种去重

            if(!flag){//没有成功往前跳

                return 0;

            }

        }

    }

    return 1;*/

}

int findmin(int l,int r){//寻找最小的满足条件的跳跃能力,每个区间必然有至少一个满足条件

    if(l==r)return l;//找到了答案

    int mid=l+((r-l)>>1);//完全等价于mid=(l+r)/2,自己找几个例子验证就明白了

    if(isok(mid))return findmin(l,mid);

    else return findmin(mid+1,r);

}

int main(){

    scanf("%d%d",&n,&x);

    x=x<<1;

    for(int i=1;i<=n-1;i++){

        scanf("%d",&h[i]);

    }

    printf("%d\n",findmin(1,n));

    system("pause");

    return 0;

}


点赞(0)
 

0.0分

2 人评分

C语言网提供由在职研发工程师或ACM蓝桥杯竞赛优秀选手录制的视频教程,并配有习题和答疑,点击了解:

一点编程也不会写的:零基础C语言学练课程

解决困扰你多年的C语言疑难杂症特性的C语言进阶课程

从零到写出一个爬虫的Python编程课程

只会语法写不出代码?手把手带你写100个编程真题的编程百练课程

信息学奥赛或C++选手的 必学C++课程

蓝桥杯ACM、信息学奥赛的必学课程:算法竞赛课入门课程

手把手讲解近五年真题的蓝桥杯辅导课程

评论列表 共有 0 条评论

暂无评论