差分约束

一.讲解:

https://blog.csdn.net/dragon60066/article/details/80245797

二.洛谷1993

题目:
小K在MC里面建立很多很多的农场,总共n个,以至于他自己都忘记了每个农场中种植作物的具体数量了,他只记得一些含糊的信息(共m个),以下列三种形式描述:

农场a比农场b至少多种植了c个单位的作物,
农场a比农场b至多多种植了c个单位的作物,
农场a与农场b种植的作物数一样多。

但是,由于小K的记忆有些偏差,所以他想要知道存不存在一种情况,使得农场的种植作物数量与他记忆中的所有信息吻合。

输入格式:

第一行包括两个整数 n 和 m,分别表示农场数目和小 K 记忆中的信息数目。

接下来 m 行:

如果每行的第一个数是 1,接下来有 3 个整数 a,b,c,表示农场 a 比农场 b 至少多种植

了 c 个单位的作物。

如果每行的第一个数是 2,接下来有 3 个整数 a,b,c,表示农场 a 比农场 b 至多多种植

了 c 个单位的作物。如果每行的第一个数是 3,家下来有 2 个整数 a,b,表示农场 a 终止的

数量和 b 一样多。

输出格式:

如果存在某种情况与小 K 的记忆吻合,输出“Yes”,否则输出“No”。

输入样例#1:

3 3
3 1 2
1 1 3 1
2 2 3 2

输出样例#1:

Yes

说明

对于 100% 的数据保证:1 ≤ n,m,a,b,c ≤ 10000。

思路:

对于至少的,a-b>=c 可以写成: b-a<=-c
对于至多的,a-b<=c
对于相等的,直接加一条权值为0的边即可.

代码:

#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <iostream>
#include <algorithm>
#include <queue>
#include <vector>
#define re register
using namespace std;
const int maxn = 200005;
const int maxm = 1000005;
const int inf=0x3f3f3f3f;
int n,m,st,en,dis[maxn];///距离数组
int head[maxn],cnt;///head->是总边数中的第几边,cnt->总边数
int vis[maxn];///判断是否存入需更新数组
int times[maxn];///判断负环

inline int read() {
    int X=0,w=1; char c=getchar();
    while (c<'0'||c>'9') { if (c=='-') w=-1; c=getchar(); }
    while (c>='0'&&c<='9') X=(X<<3)+(X<<1)+c-'0',c=getchar();
    return X*w;
}

struct edge
{
    int u,v,w,ne;
}ed[maxm];                                                          ///定义边

void init()
{
    for(int i=0;i<=n;i++)
        head[i]=-1,dis[i]=inf,vis[i]=0,times[i]=0;///初始化
    cnt=0;
}                                                                ///初始化

void addedge(int u,int v,int w)
{
    ed[cnt].u=u;
    ed[cnt].v=v;
    ed[cnt].w=w;
    ed[cnt].ne=head[u];///对于u点,上一条边的编号
    head[u]=cnt++;///这条边的编号,更新head
}                                                                ///加边

struct cmp
{
    bool operator()(int a,int b)
    {
        return dis[a]>dis[b];
    }
};
bool spfa(int st)
{
    priority_queue<int,vector<int>,cmp>q;

    q.push(st);///放入队列
    vis[st]=1;///标记
    dis[st]=0;///定义初始点的距离为0

    while(!q.empty())
    {
        int u=q.top();
        q.pop();///弹出正在更新的数
        vis[u]=0;///取消标记
        for(int i=head[u];i!=-1;i=ed[i].ne)///i等于u最后一条边的编号,然后等于上一条边的编号
        {
            int v=ed[i].v;
            if(dis[v]>dis[u]+ed[i].w)  ///已存的下一个点到st的距离  与  (st到此点距离 + 此点到下一点距离) 比较
            {                          ///如果已存的大,说明此时的路径为最短路径,   然后更新!!!
                dis[v]=dis[u]+ed[i].w;
                if(!vis[v])///不在队列中
                {
                    vis[v]=1;
                    q.push(v);///标记并放入队列
                    if(++times[v]>n)
                    {
                        return 0;
                    }///判负环
                }
            }
        }
    }
    return 1;
}                                                                 ///定义算法                                                              ///定义算法

int main()
{
     n=read(),m=read();///>>en;///n个点,m条路径,st->初始值,en->结束值
    init();///初始化
    for(int i=1;i<=n;i++)
    {
        addedge(0,i,0);
    }

    for(int i=0;i<m;i++)
    {
        int u,v,w,p;
         p=read();
        if(p==1)
        {
            u=read(),v=read(),w=read();
            addedge(u,v,-w);
        }
        else if(p==2)
        {
             v=read(),u=read(),w=read();
            addedge(u,v,w);
        }
        else if(p==3)
        {
             u=read(),v=read();
            addedge(u,v,0);
            addedge(v,u,0);
        }
    }
     if(spfa(0))cout << "Yes";
     else cout << "No";
    return 0;
}

三.HDU3592

题意:
题意简述:

当排队等候喂食时,奶牛喜欢和它们的朋友站得靠近些。FJ有N(2<=N<=1000)头奶牛,编号从1到N,沿一条直线站着等候喂食。奶牛排在队伍中的顺序和它们的编号是相同的。因为奶牛相当苗条,所以可能有两头或者更多奶牛站在同一位置上。即使说,如果我们想象奶牛是站在一条数轴上的话,允许有两头或更多奶牛拥有相同的横坐标。

一些奶牛相互间存有好感,它们希望两者之间的距离不超过一个给定的数L。另一方面,一些奶牛相互间非常反感,它们希望两者间的距离不小于一个给定的数D。给出ML条关于两头奶牛间有好感的描述,再给出MD条关于两头奶牛间存有反感的描述。(1<=ML,MD<=10000,1<=L,D<=1000000)

你的工作是:如果不存在满足要求的方案,输出-1;如果1号奶牛和N号奶牛间的距离可以任意大,输出-2;否则,计算出在满足所有要求的情况下,1号奶牛和N号奶牛间可能的最大距离。

代码:

#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <iostream>
#include <algorithm>
#include <queue>
#include <vector>
#define re register
using namespace std;
const int maxn = 200005;
const int maxm = 1000005;
const int inf=0x3f3f3f3f;
int n,x,t,y,st,en,dis[maxn];///距离数组
int head[maxn],cnt;///head->是总边数中的第几边,cnt->总边数
int vis[maxn];///判断是否存入需更新数组
int times[maxn];///判断负环

inline int read() {
    int X=0,w=1; char c=getchar();
    while (c<'0'||c>'9') { if (c=='-') w=-1; c=getchar(); }
    while (c>='0'&&c<='9') X=(X<<3)+(X<<1)+c-'0',c=getchar();
    return X*w;
}

struct edge
{
    int u,v,w,ne;
}ed[maxm];                                                          ///定义边

void init()
{
    for(int i=0;i<=n;i++)
        head[i]=-1,dis[i]=inf,vis[i]=0,times[i]=0;///初始化
    cnt=0;
}                                                                ///初始化

void addedge(int u,int v,int w)
{
    ed[cnt].u=u;
    ed[cnt].v=v;
    ed[cnt].w=w;
    ed[cnt].ne=head[u];///对于u点,上一条边的编号
    head[u]=cnt++;///这条边的编号,更新head
}                                                                ///加边

struct cmp
{
    bool operator()(int a,int b)
    {
        return dis[a]>dis[b];
    }
};
int spfa(int st)
{
    priority_queue<int,vector<int>,cmp>q;

    q.push(st);///放入队列
    vis[st]=1;///标记
    dis[st]=0;///定义初始点的距离为0

    while(!q.empty())
    {
        int u=q.top();
        q.pop();///弹出正在更新的数
        vis[u]=0;///取消标记
        for(int i=head[u];i!=-1;i=ed[i].ne)///i等于u最后一条边的编号,然后等于上一条边的编号
        {
            int v=ed[i].v;
            if(dis[v]>dis[u]+ed[i].w)  ///已存的下一个点到st的距离  与  (st到此点距离 + 此点到下一点距离) 比较
            {                          ///如果已存的大,说明此时的路径为最短路径,   然后更新!!!
                dis[v]=dis[u]+ed[i].w;
                if(!vis[v])///不在队列中
                {
                    vis[v]=1;
                    q.push(v);///标记并放入队列
                    if(++times[v]>n)
                    {
                        return -1;
                    }///判负环
                }
            }
        }
    }
    if(dis[n]==inf)return -2;
    else return dis[n];
}                                                                 ///定义算法                                                              ///定义算法

int main()
{
    t=read();
  while(t--)
  {
      n=read(),x=read(),y=read();
    init();///初始化
    int u,v,w;
    for(int i=0;i<x;i++)
    {
        u=read(),v=read(),w=read();
        addedge(u,v,w);
    }
    for(int i=0;i<y;i++)
    {
        v=read(),u=read(),w=read();
        addedge(u,v,-w);
    }
    printf("%d\n",spfa(1));
  }

    return 0;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值