计蒜客信息学 3 月普及组模拟赛(B 找规律 C拓扑dp D二分+bfs)

题目链接

做出了A,C,D不会B

A水题

B积木游戏

计蒜客信息学 3 月普及组模拟赛(B 找规律 C拓扑dp D二分+bfs)_第1张图片

做法:

计蒜客信息学 3 月普及组模拟赛(B 找规律 C拓扑dp D二分+bfs)_第2张图片

喵了个咪的  还不能用java大整数,因为只除了一个个位数的整数,那就手撸c++大数除数吧

参考代码来自:传送门

#include
using namespace std;
char a1[1000010];
int a[1000010], lena, i, res[1000010], s;//res[110]存储每一位商 s表示余数 
int main(){
	long long n;
	for(int j=1; j<=3; j++){
		gets(a1);
		lena=strlen(a1);
		//高精求余数 
		for(i=0; i

 

C-箱子

表示以前做过一模一样的,每个箱子有六种形态,然后n方  dp一下就可以了

dp[i]代表 最后一个是i这个箱子的最大层数

#include
#define rep(i,a,b) for(int i=a;i<=(b);++i)
#define mem(a,x) memset(a,x,sizeof(a))
#define pb push_back
#define pi pair
#define mk make_pair
using namespace std;
typedef long long ll;
ll gcd(ll a,ll b) { return b?gcd(b,a%b):a;}
const int N=1e3+10;
int n,len;
struct node
{
    int x,y;
    ll h;
}a[N],dp[N];
bool cmp(node a,node b)
{
    if(a.x!=b.x) return a.x>x>>y>>z;
        a[++len]={x,y,z};
        a[++len]={x,z,y};
        a[++len]={y,x,z};
        a[++len]={y,z,x};
        a[++len]={z,x,y};
        a[++len]={z,y,x};
	}
	sort(a+1,a+1+len,cmp);
	ll ans=0;
	for(int i=1;i<=len;++i){
        for(int j=0;j

 

D-明跑

计蒜客信息学 3 月普及组模拟赛(B 找规律 C拓扑dp D二分+bfs)_第3张图片

二分思路很好想,二分这个威胁度,将怪兽周围长度全部设置不能通过,最后bfs一下能否到达终点即可

#include
#define rep(i,a,b) for(int i=a;i<=(b);++i)
#define mem(a,x) memset(a,x,sizeof(a))
#define pb push_back
#define pi pair
#define mk make_pair
using namespace std;
typedef long long ll;
ll gcd(ll a,ll b) { return b?gcd(b,a%b):a;}
const int N=1e3+10;
int vis[N][N],n,m,len,vs[N][N];
struct node
{
    int x,y,w;
}a[N*N];
int dir[4][2]={1,0,0,1,-1,0,0,-1};
int bfs()
{
    memset(vs,0,sizeof(vs));
    queueque;
    que.push({1,1,0});
    vs[1][1]=1;
    while(que.size()){
        node now=que.front();que.pop();
        if(now.x==n&&now.y==m) return 1;
        for(int i=0;i<4;++i){
            int x=now.x+dir[i][0];
            int y=now.y+dir[i][1];
            if(x<1||y<1||x>n||y>m) continue;
            if(vis[x][y])continue;
            if(vs[x][y])continue;
            vs[x][y]=1;
            que.push({x,y,0});

        }
    }
    return 0;
}
int cal(int mid)
{
    queueque;
    memset(vis,0,sizeof(vis));
    for(int i=1;i<=len;++i){
        que.push({a[i].x,a[i].y,mid});
        vis[a[i].x][a[i].y]=mid;
    }
    while(que.size()){
        node now=que.front();que.pop();
        if(now.w==1) continue;
        for(int i=0;i<4;++i){
            int x=now.x+dir[i][0];
            int y=now.y+dir[i][1];
            if(x<1||y<1||x>n||y>m) continue;
            if(now.w-1>vis[x][y]){
                vis[x][y]=now.w-1;
                que.push({x,y,vis[x][y]});
            }
        }
    }
    return bfs();
}
int main()
{
	cin>>n>>m;
	rep(i,1,n)rep(j,1,m){int x;scanf("%d",&x);if(x==1) a[++len]={i,j};}

	rep(i,1,len) vis[a[i].x][a[i].y]=1;

	if(vis[1][1]||vis[n][m]||!bfs()){
        printf("0");
        return 0;
	}

	int l=1,r=n+m,ans=0;
	while(l<=r){
        int mid=l+r>>1;
        if(cal(mid)) l=mid+1,ans=mid;
        else r=mid-1;
	}
	printf("%d\n",ans);
}

 

你可能感兴趣的:(计蒜客题解)