洛谷 P1993 小K的农场

洛谷 P1993 小K的农场

Description

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

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

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

Input

  • 第一行包括两个整数 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 一样多。

Output

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

Sample Input

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

Sample Output

Yes

Data Size

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

题解:

  • 差分约束。

  • 差分约束就是数形结合。用最短路算法解决不等式组。

    • 农场a比农场b至少多种植了c个单位的作物 -> b + c <= a

    • 农场a比农场b至多多种植了c个单位的作物 -> b + c >= a

    • 农场a与农场b种植的作物数一样多 -> a = b

  • 对于a >= b + c,可以理解为b向a连了一条权值为c的单向边;对于b >= a + (-c),可以理解为a向b连了一条权值为-c的单向边;a=b理解为连了一条权值为0,在a、b之间的无向边。

  • 因为此题的不等式均可以化成x >= y + c。所以如果x的最小值应该是众多"y + c"的最大值。所以如果要求最短路,实则是求最长路

  • 另:洛谷写bfs-spfa会被卡,dfs-spfa不会。但dfs-spfa是玄学复杂度。所以不纠结分数,关键是思想。

#include <iostream>
#include <cstdio>
#include <cstring>
#include <queue>
#define N 50005
using namespace std;

struct E {int next, to, dis;} e[N];
int n, m, num;
int h[N], cnt[N], dis[N];
bool vis[N];

void add(int u, int v, int w)
{
    e[++num].next = h[u];
    e[num].to = v;
    e[num].dis = w;
    h[u] = num;
}

bool spfa()
{
    bool flag = 1;
    queue<int> que;
    memset(dis, -0x3f, sizeof(dis));
    dis[0] = 0, vis[0] = 1, que.push(0);
    while(!que.empty())
    {
        int now = que.front();
        cnt[now]++;
        if(cnt[now] >= n) {flag = 0; break;}
        que.pop();  vis[now] = 0;
        for(int i = h[now]; i != 0; i = e[i].next)
            if(dis[now] + e[i].dis > dis[e[i].to])
            {
                dis[e[i].to] = dis[now] + e[i].dis;
                if(!vis[e[i].to])
                {
                    que.push(e[i].to);
                    vis[e[i].to] = 1;
                }
            }
    }
    return flag;
}

int main()
{
    cin >> n >> m;
    for(int i = 1; i <= m; i++)
    {
        int op; cin >> op;
        if(op == 1)
        {
            int a, b, c;
            cin >> a >> b >> c;
            add(b, a, c);
            continue;
        }
        if(op == 2)
        {
            int a, b, c;
            cin >> a >> b >> c;
            add(a, b, -c);
            continue;
        }
        if(op == 3)
        {
            int a, b;
            cin >> a >> b;
            add(a, b, 0), add(b, a, 0);
            continue;
        }
    }
    for(int i = 1; i <= n; i++) add(0, i, 0);
    if(!spfa()) cout << "No";
    else cout << "Yes";
    return 0;
}

转载于:https://www.cnblogs.com/BigYellowDog/p/11229951.html

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值