zoukankan      html  css  js  c++  java
  • Warshall算法求传递闭包及具体实现

    传递闭包

    在数学中,在集合 X 上的二元关系 R 的传递闭包是包含 R 的 X 上的最小的传递关系。

    例如,如果 X 是(生或死)人的集合而 R 是关系“为父子”,则 R 的传递闭包是关系“x 是 y 的祖先”。再比如,如果 X 是空港的集合而关系 xRy 为“从空港 x 到空港 y 有直航”,则 R 的传递闭包是“可能经一次或多次航行从 x 飞到 y”。

    Warshall算法

    Warshall在1962年提出了一个求关系的传递闭包的有效算法。其具体过程如下,设在n个元素的有限集上关系R的关系矩阵为M:
    (1)置新矩阵A=M;
    (2)置k=1;
    (3)对所有i如果A[i,k]=1,则对j=1..n执行:
    A[i,j]←A[i,j]∨A[k,j];
    (4)k增1;
    (5)如果k≤n,则转到步骤(3),否则停止。
    所得的矩阵A即为关系R的传递闭包t(R)的关系矩阵。
     要说跟floyd算法有什么关系的话...就是warshall求传递闭包他...他没有记录路径长度只是判断顶点之间是否直接或间接连通(个人理解)
    代码实现:
    #include <bits/stdc++.h>
    using namespace std;
    /* freopen("k.in", "r", stdin);
    freopen("k.out", "w", stdout); */
    // clock_t c1 = clock();
    // std::cerr << "Time:" << clock() - c1 <<"ms" << std::endl;
    //#pragma comment(linker, "/STACK:1024000000,1024000000")
    #define de(a) cout << #a << " = " << a << endl
    #define rep(i, a, n) for (int i = a; i <= n; i++)
    #define per(i, a, n) for (int i = n; i >= a; i--)
    #define ls ((x) << 1)
    #define rs ((x) << 1 | 1)
    typedef long long ll;
    typedef unsigned long long ull;
    typedef pair<int, int> PII;
    typedef pair<double, double> PDD;
    typedef pair<ll, ll> PLL;
    typedef vector<int, int> VII;
    #define inf 0x3f3f3f3f
    const ll INF = 0x3f3f3f3f3f3f3f3f;
    const ll MAXN = 1e6 + 7;
    const ll MAXM = 1e5 + 7;
    const ll MOD = 1e9 + 7;
    const double eps = 1e-6;
    const double pi = acos(-1.0);
    int Mat[20][20]; //
    void Print_Mat(int n)
    {
        for (int i = 0; i < n; i++)
        {
            for (int j = 0; j < n; j++)
                cout << Mat[i][j] << " ";
            cout << endl;
        }
        return;
    }
    void Warshall(int n)
    {
        for (int k = 0; k < n; k++)
            for (int i = 0; i < n; i++)
                for (int j = 0; j < n; j++)
                    if (Mat[i][k] && Mat[k][j])
                        Mat[i][j] = 1;
    }
    int main()
    {
        int n;
        cout << "输入矩阵阶数" << endl;
        while (cin >> n)
        {
            memset(Mat, 0, sizeof(Mat));
            cout << "输入矩阵M:" << endl;
            for (int i = 0; i < n; i++)
                for (int j = 0; j < n; j++)
                    cin >> Mat[i][j];
            Warshall(n);
            cout << "矩阵M的传递闭包为:" << endl;
            Print_Mat(n);
            cout << "输入矩阵阶数" << endl;
        }
        return 0;
    }

      

     

  • 相关阅读:
    基于吉日嘎底层架构的Web端权限管理操作演示-日志管理
    基于吉日嘎底层架构的Web端权限管理操作演示-组织机构管理
    基于吉日嘎底层架构的Web端权限管理操作演示-菜单模块管理
    基于吉日嘎底层架构的Web端权限管理操作演示-角色管理
    基于吉日嘎底层架构的Web端权限管理操作演示-用户管理
    扩展吉日嘎拉的用户角色管理,让用户角色编码和名称在一个组织里面唯一
    Redis 3.2.100 Windows 32位下载
    C#开发中Windows域认证登录2016(扩展吉日嘎拉GPM系统V4.2)
    JVM对象
    将博客搬至CSDN
  • 原文地址:https://www.cnblogs.com/graytido/p/10886732.html
Copyright © 2011-2022 走看看