zoukankan      html  css  js  c++  java
  • poj 3259

    SPFA加邻接矩阵勉强过了

      1 #include <iostream>
      2 #include <string>
      3 #include <vector>
      4 #include <cstdlib>
      5 #include <cmath>
      6 #include <map>
      7 #include <algorithm>
      8 #include <list>
      9 #include <ctime>
     10 #include <set>
     11 #include <string.h>
     12 #include <queue>
     13 #include <cstdio>
     14 using namespace std;
     15 typedef int typep;
     16 int maxData = 500000;
     17 #define N 505
     18 #define CLR(arr, what) memset(arr, what, sizeof(arr))
     19 int path[N][N];
     20 int d[N];
     21 int n;
     22 //double maxData = 0;
     23 
     24 bool relax(typep u, typep& v, typep w) {
     25     if (v > (u + w)) {
     26         v = u + w;
     27         return true;
     28     } else
     29         return false;
     30 }
     31 
     32 bool bellman_ford(int s) {
     33     //int n = d.size();
     34     int i, j, k;
     35     bool judge;
     36 
     37     for (i = 0; i < n - 1; i++) {
     38         for (j = 0; j < n; j++) {
     39             for (k = 0; k < n; k++) {
     40                 if (path[j][k] != maxData && d[j] != maxData)
     41                     judge = relax(d[j], d[k], path[j][k]);
     42             }
     43         }
     44     }
     45 
     46     for (j = 0; j < n; j++) {
     47         for (k = 0; k < n; k++) {
     48             if (path[j][k] != maxData && d[j] != maxData) {
     49                 judge = relax(d[j], d[k], path[j][k]);
     50                 if (judge) {
     51                     return true;
     52                 }
     53             }
     54 
     55         }
     56     }
     57 
     58     return false;
     59 }
     60 
     61 
     62 int cnt[N];//记录顶点入队列次数
     63 bool final[N];//记录顶点是否在队列中,SPFA算法可以入队列多次
     64 bool SPFA(int s) {
     65     queue<int> myqueue;
     66     int i;
     67     CLR(final,0);
     68     CLR(cnt,0);
     69     final[s] = true;
     70     cnt[s]++; //源点的入队列次数增加
     71     myqueue.push(s);
     72     int topint;
     73     bool judge;
     74     while (!myqueue.empty()) {
     75         topint = myqueue.front();
     76         myqueue.pop();
     77         final[topint] = false;
     78         for (i = 0; i < n; ++i) {
     79             if (d[topint] < maxData) {
     80                 judge = relax(d[topint], d[i], path[topint][i]);
     81                 if (judge) {
     82                     if (!final[i]) { //判断是否在当前的队列中
     83                         final[i] = true;
     84                         cnt[i]++;
     85                         if (cnt[i] >= n) //当一个点入队的次数>=n时就证明出现了负环。
     86                             return true;
     87                         myqueue.push(i);
     88                     }
     89                 }
     90             }
     91         }
     92     }
     93     return false;
     94 }
     95 
     96 int main() {
     97     int f, m, w, u, v, c, t;
     98     cin >> f;
     99     for (int abc = 0; abc < f; abc++) {
    100         scanf("%d%d%d", &t, &m, &w);
    101         n = t + 1;
    102         CLR(path, maxData);
    103         for (int i = 0; i < t; i++) {
    104             path[0][i + 1] = 0;
    105         }
    106         for (int i = 0; i < m; i++) {
    107             scanf("%d %d %d", &u, &v, &c);
    108             path[u][v] = min(path[u][v], c);
    109             path[v][u] = min(path[v][u], c);
    110         }
    111         for (int i = 0; i < w; i++) {
    112             scanf("%d %d %d", &u, &v, &c);
    113             path[u][v] = min(path[u][v], 0 - c);
    114         }
    115         int judge = false;
    116         CLR(d, maxData);
    117         d[0] = 0;
    118         //judge = bellman_ford(0);
    119         judge = SPFA(0);
    120         if (judge) {
    121             cout << "YES" << endl;
    122         } else {
    123             cout << "NO" << endl;
    124         }
    125     }
    126     return 0;
    127 }

    对于这种稀疏矩阵,邻接表的方法更快,附代码

    #include <iostream>
    #include <string>
    #include <vector>
    #include <cstdlib>
    #include <cmath>
    #include <map>
    #include <algorithm>
    #include <list>
    #include <ctime>
    #include <set>
    #include <string.h>
    #include <queue>
    #include <cstdio>
    using namespace std;
    typedef int typep;
    const int E = 10000;
    const int V = 600;
    #define typec int                       // type of cost
    const typec inf = 1000000; // max of cost
    int n, m, pre[V], edge[E][3];
    typec dist[V];
    int relax(typec u, typec v, typec c) {
        if (dist[v] > dist[u] + c) {
            pre[v] = u;
            dist[v] = dist[u] + c;
            return 1;
        }
        return 0;
    }
    int bellman(int src) {
        int i, j;
        for (i = 0; i < n; ++i) {
            dist[i] = inf;
            pre[i] = -1;
        }
        dist[src] = 0;
        bool flag;
        for (i = 1; i < n; ++i) {
            flag = false; //  优化
            for (j = 0; j < m; ++j) {
                if (1 == relax(edge[j][0], edge[j][1], edge[j][2]))
                    flag = true;
            }
            if (!flag)
                break;
        }
        for (j = 0; j < m; ++j) {
            if (1 == relax(edge[j][0], edge[j][1], edge[j][2]))
                return 0; //  有负圈
        }
        return 1;
    }
    inline void addedge(int u, int v, typec c) {
        edge[m][0] = u;
        edge[m][1] = v;
        edge[m][2] = c;
        m++;
    }
    int main() {
        int f, mt, w, u, v, c, t;
        cin >> f;
        for (int abc = 0; abc < f; abc++) {
            scanf("%d%d%d", &t, &mt, &w);
            n = t + 1;
            m = 0;
            for (int i = 0; i < t; i++) {
                addedge(0, i + 1, 0);
            }
            for (int i = 0; i < mt; i++) {
                scanf("%d %d %d", &u, &v, &c);
                addedge(u, v, c);
                addedge(v, u, c);
            }
            for (int i = 0; i < w; i++) {
                scanf("%d %d %d", &u, &v, &c);
                addedge(u, v, -c);
            }
            int judge = false;
            judge = bellman(0);
            if (!judge) {
                cout << "YES" << endl;
            } else {
                cout << "NO" << endl;
            }
        }
        return 0;
    }

    from kakamilan

  • 相关阅读:
    Oracle列转行函数Listagg以及pivot查询示例
    Java执行操作系统命令
    JUnit学习之hamcrest、testSuite介绍及测试原则
    JUnit学习之JUnit的基本介绍
    Oracle笔记-Multitable INSERT 的用法
    Oracle 11g 新特性 --SQL Plan Management 说明
    Java习惯用法总结
    将Eclipse的Java Project转换为Dynamic Web Project
    在Eclipse中调试web项目
    在Eclipse中添加Servlet-api.jar的方法
  • 原文地址:https://www.cnblogs.com/kakamilan/p/3075889.html
Copyright © 2011-2022 走看看