2/25 floyed+dp+bfs+dfs染色+二分和bfs

本文探讨了三个不同的编程问题,涉及灾后重建的最短路径算法(Floyd-Warshall)、尼克任务的最优化安排(逆序考虑影响),以及动物运动路径计算(COWJOGS)。通过实例展示了如何运用算法解决实际问题,包括动态规划、图论和优先级队列等技术。
摘要由CSDN通过智能技术生成

P1119 灾后重建

隐含条件:询问的t也是从小到大的

#include <bits/stdc++.h>

using namespace std;
const int maxn=1e3+5;
const int inf=0x3f3f3f3f;
int n,m,f[maxn][maxn],a[maxn],q;
void floyed(int k)
{
    for(int i=0;i<n;i++)
    {
        for(int j=0;j<n;j++)
            f[i][j]=min(f[i][j],f[i][k]+f[k][j]);
    }
}
int main()
{
    scanf("%d%d",&n,&m);
    for(int i=0;i<n;i++)
        scanf("%d",&a[i]);
    memset(f,inf,sizeof(f));
    for(int i=0;i<n;i++)
        f[i][i]=0;
    for(int i=1;i<=m;i++)
    {
        int u,v,w;scanf("%d%d%d",&u,&v,&w);
        f[u][v]=f[v][u]=w;
    }
    scanf("%d",&q);
    int now=0;
    for(int i=1;i<=q;i++)
    {
        int x,y,t;scanf("%d%d%d",&x,&y,&t);
        while(a[now]<=t&&now<n)
        {
            floyed(now);now++;
        }
        if(a[x]>t||a[y]>t||f[x][y]==inf)
            cout<<-1<<endl;
        else
            cout<<f[x][y]<<endl;
    }
    return 0;
}

P1280 尼克的任务

降序排列,f[i]表示当前i最多有多少闲暇时间,会受到后面人物的影响,因此要反向考虑。

#include <bits/stdc++.h>

using namespace std;
const int maxn=1e4+5;
const int inf=0x3f3f3f3f;
int f[maxn],n,k,num[maxn],g=1;
struct node
{
    int p,t;
}e[maxn];
bool cmp(node e1,node e2)
{
    return e1.p>e2.p;
}
int main()
{
    scanf("%d%d",&n,&k);
    for(int i=1;i<=k;i++)
    {
        scanf("%d%d",&e[i].p,&e[i].t);
        num[e[i].p]++;
    }
    sort(e+1,e+k+1,cmp);
    for(int i=n;i>=1;i--)
    {
        if(num[i]==0)
            f[i]=f[i+1]+1;
        else
        {
            for(int j=1;j<=num[i];j++) //控制这个时间点该扫描几个任务
            {
                f[i]=max(f[i],f[i+e[g].t]);
                g++;
            }
        }
    }
    cout<<f[1]<<endl;
    return 0;
}

P3111 [USACO14DEC]Cow Jog S

#include <bits/stdc++.h>
#define int long long

using namespace std;
const int maxn=1e5+5;
struct node
{
    int p,sp;
}e[maxn];
int n,t,s[maxn],ans;

signed main()
{
    cin>>n>>t;
    for(int i=1;i<=n;i++)
    {
        cin>>e[i].p>>e[i].sp;
        s[i]=e[i].p+e[i].sp*t;
    }
    ans=1;
    for(int i=n-1;i>=1;i--)
    {
        if(s[i]>=s[i+1])
            s[i]=s[i+1];
        else
            ans++;
    }
    cout<<ans<<endl;
    return 0;
}

P7775 [COCI 2009-2010 #2] VUK

两次bfs

#include <bits/stdc++.h>


using namespace std;
const int inf=0x3f3f3f3f;
const int maxn=1e3+5;
int dx[4]={0,1,0,-1};
int dy[4]={1,0,-1,0};
int dd[maxn][maxn],n,m;
bool tree[maxn][maxn],vis[maxn][maxn];
struct node
{
    int x,y,dis;
}st,ed;
queue<node>q;
void bfs()
{
    while(!q.empty())
    {
        node cur=q.front();q.pop();
        dd[cur.x][cur.y]=cur.dis;
        for(int i=0;i<4;i++)
        {
            int nx=cur.x+dx[i],ny=cur.y+dy[i];
            if(1<=nx&&nx<=n&&1<=ny&&ny<=m&&!tree[nx][ny])
            {
                tree[nx][ny]=1;
                q.push({nx,ny,cur.dis+1});
            }
        }
    }
    return ;
}
struct wolf
{
    int a,b,m_d;
    bool operator < (const wolf &tmp) const
    {
		return m_d<tmp.m_d;
	}
};
int bfs1(int x,int y)
{
	int ans=inf;
	priority_queue<wolf> q;
	vis[x][y]=1;
	q.push({x,y,dd[x][y]});
	while(!q.empty()){
		wolf cur=q.top();
		q.pop();
		if(cur.a==ed.x&&cur.b==ed.y) {//取最小答案
			ans=min(ans,cur.m_d);
		}
		for(int i=0;i<4;i++){
			int nx=cur.a+dx[i];
			int ny=cur.b+dy[i];
			if(1<=nx&&nx<=n&&1<=ny&&ny<=m&&!vis[nx][ny]){
				vis[nx][ny]=1;
				q.push({nx,ny,min(cur.m_d,dd[nx][ny])});
			}
		}
	}
	return ans;//返回答案
}
signed main()
{
    scanf("%d%d",&n,&m);
    for(int i=1;i<=n;i++)
    {
        for(int j=1;j<=m;j++)
        {
            char c;cin>>c;
            if(c=='+')
            {
                tree[i][j]=1;
                q.push({i,j,0});
            }
            else if(c=='V')
                st={i,j};
            else if(c=='J')
                ed={i,j};
        }
    }
    bfs();
    cout<<bfs1(st.x,st.y)<<endl;
    return 0;
}

P3496 [POI2010]GIL-Guilds

存图+染色(要保证联通)

#include <bits/stdc++.h>

using namespace std;
const int maxn=1e6+5;
const int inf=0x3f3f3f3f;
int n,m,head[maxn],cnt;;
bool vis[maxn],p[maxn];
struct node
{
    int to,nxt;
}e[maxn];
void add(int from,int to)
{
    e[++cnt].to=to;
    e[cnt].nxt=head[from];
    head[from]=cnt;
}
void dfs(int u,int color)  //染色原理
{
    vis[u]=1;p[u]=color;
    for(int i=head[u];i;i=e[i].nxt)
    {
        int v=e[i].to;
        if(!vis[v])
            dfs(v,color^1);
    }
}
int main()
{
    scanf("%d%d",&n,&m);
    for(int i=1;i<=m;i++)
    {
        int u,v;scanf("%d%d",&u,&v);
        add(u,v);add(v,u);
        vis[u]=vis[v]=1;
    }
    for(int i=1;i<=n;i++)
    {
        if(!vis[i])
        {
            cout<<"NIE"<<endl;return 0;
        }
    }
    cout<<"TAK"<<endl;
    for(int i=1;i<=n;i++)
        vis[i]=0;
    for(int i=1;i<=n;i++)
    {
        if(!vis[i])
            dfs(i,0);
    }
    for(int i=1;i<=n;i++)
    {
        if(!p[i])
            cout<<"K"<<endl;
        else
            cout<<"S"<<endl;
    }
    return 0;
}

P1902 刺杀大使

二分+bfs

#include <bits/stdc++.h>

using namespace std;
const int maxn=1e3+5;
const int inf=0x3f3f3f3f;
int n,m,p[maxn][maxn],l=inf,r,ans;
bool vis[maxn][maxn];
int dx[4]={0,1,0,-1};
int dy[4]={1,0,-1,0};
int bfs(int x,int y,int mx)
{
    memset(vis,0,sizeof(vis));
    queue<pair<int,int> >q;
    q.push(make_pair(x,y));
    vis[x][y]=1;
    while(!q.empty())
    {
        int xx=q.front().first,yy=q.front().second;
        q.pop();
        for(int i=0;i<4;i++)
        {
            int nx=xx+dx[i],ny=yy+dy[i];
            if(nx<=0||nx>n||ny<=0||ny>m||vis[nx][ny]||p[nx][ny]>mx)
                continue;
            vis[nx][ny]=1;
            if(nx==n)
                return 1;
            q.push(make_pair(nx,ny));
        }
    }
    return 0;
}
int main()
{
    scanf("%d%d",&n,&m);
    for(int i=1;i<=n;i++)
        for(int j=1;j<=m;j++)
    {
        scanf("%d",&p[i][j]);
        l=min(l,p[i][j]);
        r=max(r,p[i][j]);
    }
    while(l<=r)
    {
        int mid=l+r>>1;
        if(bfs(1,1,mid))
            r=mid-1,ans=mid;
        else
            l=mid+1;
    }
    cout<<ans<<endl;
    return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值