文心一言 VS 讯飞星火 VS chatgpt (299)-- 算法导论22.1 3题

news2025/1/18 3:23:46

三、有向图 G = ( V , E ) G=(V,E) G=(V,E) 的转置是图 G T = ( V , E T ) G^{T} =(V,E^{T}) GT=(V,ET),这里 E T = { ( v , u ) ∈ V × V E^{T} =\{(v,u)∈ V \times V ET={(v,u)V×V:(u,v)∈ E}$因此,图 G T G^{T} GT 就是将有向图 G G G中所有边的方向反过来而形成的图。对于邻接链表和邻接矩阵两种表示,请给出从图 G G G计算出 G T G^{T} GT的有效算法,并分析算法的运行时间。如果要写代码,请用go语言。

文心一言:

为了从有向图 G G G 计算出其转置图 G T G^T GT,我们可以分别针对邻接链表和邻接矩阵两种表示形式来设计算法。下面我将分别给出这两种方法的描述以及相应的 Go 语言代码实现。

1. 邻接链表表示

在邻接链表中,每个节点维护一个链表,链表中的每个节点表示从当前节点出发可以到达的节点。为了得到转置图,我们需要遍历原图的所有边,并将每条边的方向反转,即对于边 ( u , v ) (u, v) (u,v),我们在 G T G^T GT 中添加边 ( v , u ) (v, u) (v,u)

算法步骤

  1. 初始化 G T G^T GT 的邻接链表表示。
  2. 遍历 G G G 的每一条边 ( u , v ) (u, v) (u,v)
  3. G T G^T GT 中,将 v v v 添加到 u u u 的链表中(如果 u u u 的链表不存在,则先创建)。

Go 代码实现

package main

import "fmt"

type Graph struct {
    adjList map[int][]int
}

func NewGraph() *Graph {
    return &Graph{adjList: make(map[int][]int)}
}

func (g *Graph) AddEdge(from, to int) {
    g.adjList[from] = append(g.adjList[from], to)
}

func (g *Graph) Transpose() *Graph {
    transposed := NewGraph()
    for from, tos := range g.adjList {
        for _, to := range tos {
            transposed.AddEdge(to, from)
        }
    }
    return transposed
}

func main() {
    g := NewGraph()
    g.AddEdge(0, 1)
    g.AddEdge(0, 2)
    g.AddEdge(1, 2)
    g.AddEdge(2, 0)

    gt := g.Transpose()
    fmt.Println("Transposed Graph:")
    for from, tos := range gt.adjList {
        fmt.Printf("%d -> %v\n", from, tos)
    }
}

时间复杂度 O ( V + E ) O(V+E) O(V+E),其中 V V V 是节点数, E E E 是边数。因为我们需要遍历每条边并可能更新每个节点的链表。

2. 邻接矩阵表示

在邻接矩阵中,矩阵的每个元素 A [ i ] [ j ] A[i][j] A[i][j] 表示从节点 i i i 到节点 j j j 是否存在边。为了得到转置图,我们只需将矩阵转置即可。

算法步骤

  1. 初始化 G T G^T GT 的邻接矩阵,大小与 G G G 相同。
  2. 遍历 G G G 的邻接矩阵的每一个元素 A [ i ] [ j ] A[i][j] A[i][j]
  3. 如果 A [ i ] [ j ] A[i][j] A[i][j] 非零(或根据图的类型,可能有其他表示方式),则将 G T G^T GT 的邻接矩阵中 A T [ j ] [ i ] A^T[j][i] AT[j][i] 设置为相同的值。

Go 代码实现

package main

import "fmt"

type MatrixGraph struct {
    matrix [][]int
    size   int
}

func NewMatrixGraph(size int) *MatrixGraph {
    return &MatrixGraph{matrix: make([][]int, size), size: size}
}

func (mg *MatrixGraph) AddEdge(from, to int) {
    mg.matrix[from][to] = 1
}

func (mg *MatrixGraph) Transpose() *MatrixGraph {
    transposed := NewMatrixGraph(mg.size)
    for i := 0; i < mg.size; i++ {
        for j := 0; j < mg.size; j++ {
            transposed.matrix[j][i] = mg.matrix[i][j]
        }
    }
    return transposed
}

func main() {
    mg := NewMatrixGraph(3)
    mg.AddEdge(0, 1)
    mg.AddEdge(0, 2)
    mg.AddEdge(1, 2)
    mg.AddEdge(2, 0)

    mgt := mg.Transpose()
    fmt.Println("Transposed Matrix Graph:")
    for _, row := range mgt.matrix {
        fmt.Println(row)
    }
}

**

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

讯飞星火:

首先,我们来分析邻接链表和邻接矩阵两种表示方法。

  1. 邻接链表表示法:
    在邻接链表中,每个顶点都有一个链表,链表中的每个元素表示与该顶点相邻的顶点。为了计算转置图 G T G^T GT,我们需要遍历原图 G G G的所有顶点,然后遍历每个顶点的链表,将链表中的顶点添加到新链表中,表示转置后的边。

  2. 邻接矩阵表示法:
    在邻接矩阵中,矩阵的每个元素表示两个顶点之间是否存在边。为了计算转置图 G T G^T GT,我们需要将原矩阵进行转置操作,即将矩阵的行和列互换。

接下来,我们给出两种表示法的算法实现:

  1. 邻接链表表示法:
func transposeGraph(graph []map[int]struct{}) map[int]map[int]struct{} {
    transposedGraph := make(map[int]map[int]struct{})
    for u, neighbors := range graph {
        transposedNeighbors := make(map[int]struct{})
        for v := range neighbors {
            transposedNeighbors[v] = struct{}{}
        }
        transposedGraph[u] = transposedNeighbors
    }
    return transposedGraph
}
  1. 邻接矩阵表示法:
func transposeMatrix(matrix [][]bool) [][]bool {
    rows := len(matrix)
    cols := len(matrix[0])
    transposedMatrix := make([][]bool, cols)
    for i := range transposedMatrix {
        transposedMatrix[i] = make([]bool, rows)
        for j := range transposedMatrix[i] {
            transposedMatrix[i][j] = matrix[j][i]
        }
    }
    return transposedMatrix
}

运行时间分析:

  1. 邻接链表表示法的时间复杂度为O(E),其中E为原图中的边数。因为我们需要遍历所有顶点的链表。
  2. 邻接矩阵表示法的时间复杂度为O(V^2),其中V为原图中的顶点数。因为我们需要遍历整个矩阵并进行转置操作。

360智脑:

邻接链表表示法

在邻接链表表示法中,我们可以直接遍历原图 G G G的每个节点的邻接链表,然后将每条边的方向反转后添加到新图 G T G^{T} GT的对应节点上。以下是用Go语言实现的代码:

type Graph struct {
    Vertices []*Node
}

type Node struct {
    Value int
    Adjacent []*Node
}

func (g *Graph) Transpose() *Graph {
    transposed := &Graph{make([]*Node, len(g.Vertices))}
    for i, node := range g.Vertices {
        transposed.Vertices[i] = &Node{node.Value, make([]*Node, 0)}
    }
    for _, node := range g.Vertices {
        for _, neighbor := range node.Adjacent {
            // Add edge from neighbor to current node in the transposed graph
            transposed.Vertices[neighbor.Value].Adjacent = append(transposed.Vertices[neighbor.Value].Adjacent, node)
        }
    }
    return transposed
}

该算法的运行时间是 O ( ∣ V ∣ + ∣ E ∣ ) O(|V| + |E|) O(V+E),其中 ∣ V ∣ |V| V是图中顶点的数量, ∣ E ∣ |E| E是图中边的数量。这是因为我们需要遍历每个节点和每条边一次。

邻接矩阵表示法

在邻接矩阵表示法中,我们可以通过交换矩阵的行和列来得到转置图。以下是用Go语言实现的代码:

type MatrixGraph struct {
    Matrix [][]bool
}

func (m *MatrixGraph) Transpose() *MatrixGraph {
    n := len(m.Matrix)
    transposed := make([][]bool, n)
    for i := range transposed {
        transposed[i] = make([]bool, n)
    }
    for i := 0; i < n; i++ {
        for j := 0; j < n; j++ {
            transposed[j][i] = m.Matrix[i][j]
        }
    }
    return &MatrixGraph{transposed}
}

该算法的运行时间是 O ( ∣ V ∣ 2 ) O(|V|^2) O(V2),因为在最坏的情况下,我们需要访问矩阵中的每个元素。如果图是稠密的(即 ∣ E ∣ ≈ ∣ V ∣ 2 |E| \approx |V|^2 EV2),那么这个算法的时间复杂度与邻接链表表示法的算法相当。但如果图是稀疏的(即 ∣ E ∣ < < ∣ V ∣ 2 |E| << |V|^2 E<<V2),那么邻接链表表示法的算法会更高效。

智谱清言:

在图的邻接链表表示中,每个顶点对应一个链表,链表中包含所有与该顶点相邻的顶点的信息。在邻接矩阵表示中,矩阵的行和列代表图的顶点,矩阵中的元素表示相应顶点之间的边。转置图意味着我们需要交换原图中边的方向。
下面分别给出使用邻接链表和邻接矩阵表示的有向图转置的算法,并使用Go语言实现。

邻接链表表示

算法

  1. 创建一个新的顶点列表,复制原图的顶点。
  2. 遍历原图的每个顶点。
  3. 对于每个顶点,遍历它的邻接表。
  4. 对于邻接表中的每条边 (u, v),在转置图中添加边 (v, u)
    运行时间:O(V + 2E),其中 V 是顶点数,E 是边数。遍历所有顶点一次,每个顶点处理它的邻接表,每条边被处理两次(一次在原图中,一次在转置图中)。
    Go 代码实现
package main
import (
	"fmt"
)
type Edge struct {
	To int
}
type Graph struct {
	Vertices []int
	Edges    []*Edge
}
func (g *Graph) AddEdge(from, to int) {
	g.Edges = append(g.Edges, &Edge{To: to})
}
func TransposeGraph(g *Graph) *Graph {
	transposed := &Graph{Vertices: make([]int, len(g.Vertices))}
	for _, v := range g.Vertices {
		transposed.AddEdge(v, g.Vertices...)
	}
	return transposed
}
func main() {
	g := &Graph{
		Vertices: []int{0, 1, 2, 3},
		Edges:    []*Edge{},
	}
	g.AddEdge(0, 1)
	g.AddEdge(0, 2)
	g.AddEdge(1, 2)
	g.AddEdge(2, 3)
	transposed := TransposeGraph(g)
	fmt.Println("Transposed Graph Edges:")
	for _, edge := range transposed.Edges {
		fmt.Printf("(%d, %d)\n", edge.To, edge.From)
	}
}

邻接矩阵表示

算法

  1. 创建一个新的顶点列表,复制原图的顶点。
  2. 遍历原图的邻接矩阵。
  3. 对于矩阵中的每个元素,如果 A[u][v] 为 1(表示存在从 u 到 v 的边),在转置图中设置 A[v][u] 为 1。
    运行时间:O(V^2),其中 V 是顶点数。遍历整个邻接矩阵。
    Go 代码实现
package main
import (
	"fmt"
)
func TransposeGraph(A [][]int) [][]int {
	V := len(A)
	transposed := make([][]int, V)
	for i := range transposed {
		transposed[i] = make([]int, V)
	}
	for i := 0; i < V; i++ {
		for j := 0; j < V; j++ {
			if A[i][j] == 1 {
				transposed[j][i] = 1
			}
		}
	}
	return transposed
}
func main() {
	A := [][]int{
		{0, 1, 1},
		{1, 0, 1},
		{1, 1, 0},
	}
	transposed := TransposeGraph(A)
	fmt.Println("Transposed Graph:")
	for _, row := range transposed {
		fmt.Println(row)
	}
}

上述代码展示了如何使用 Go 语言实现有向图的转置,

天工:

为了计算有向图 G G G 的转置 G T G^{T} GT,我们可以分别针对邻接链表和邻接矩阵两种表示方式设计算法。

邻接链表表示

算法描述
  1. 遍历图 G G G 的邻接链表。
  2. 对于每条边 ( u , v ) (u, v) (u,v),将其添加到 G T G^{T} GT 的邻接链表中,但方向相反,即作为顶点 v v v 的邻接边。
  3. 重复步骤2直到所有边都被处理。
运行时间分析

假设图 G G G n n n 个顶点和 m m m 条边,那么这个算法的时间复杂度是 O ( n + m ) O(n + m) O(n+m),因为我们需要遍历所有的边一次。空间复杂度也是 O ( n + m ) O(n + m) O(n+m),因为我们创建了一个新的邻接链表来存储转置图。

Go语言实现
type Graph struct {
	vertices []*Vertex
}

type Vertex struct {
	id       int
	adjacent []*Vertex
}

func transposeGraph(g *Graph) *Graph {
	t := &Graph{make([]*Vertex, len(g.vertices))}
	for i, v := range g.vertices {
		t.vertices[i] = &Vertex{id: v.id}
	}
	for _, v := range g.vertices {
		for _, neighbor := range v.adjacent {
			t.vertices[neighbor.id].adjacent = append(t.vertices[neighbor.id].adjacent, t.vertices[v.id])
		}
	}
	return t
}

邻接矩阵表示

算法描述
  1. 遍历图 G G G 的邻接矩阵。
  2. 对于每个元素 ( i , j ) (i, j) (i,j),如果 G [ i ] [ j ] G[i][j] G[i][j] 为真(表示有一条从 i i i j j j 的边),则在 G T G^{T} GT 的邻接矩阵中设置 G T [ j ] [ i ] G^{T}[j][i] GT[j][i] 为真。
  3. 重复步骤2直到所有元素都被处理。
运行时间分析

这个算法的时间复杂度是 O ( n 2 ) O(n^2) O(n2),其中 n n n 是顶点的数量,因为我们需要遍历整个邻接矩阵。空间复杂度也是 O ( n 2 ) O(n^2) O(n2),因为我们创建了一个新的邻接矩阵来存储转置图。

Go语言实现
type MatrixGraph struct {
	matrix [][]bool
}

func transposeMatrixGraph(g *MatrixGraph) *MatrixGraph {
	n := len(g.matrix)
	t := &MatrixGraph{make([][]bool, n)}
	for i := range t.matrix {
		t.matrix[i] = make([]bool, n)
	}
	for i := 0; i < n; i++ {
		for j := 0; j < n; j++ {
			if g.matrix[i][j] {
				t.matrix[j][i] = true
			}
		}
	}
	return t
}

在这两个实现中,我们都创建了一个新的图或矩阵来存储转置图的结果。如果我们不想使用额外的空间,我们可以考虑在原始图或矩阵上直接修改,但这可能会影响到原始数据的完整性。

灵小智:

对于邻接链表表示的有向图 G = ( V , E ) G=(V,E) G=(V,E),可以按照以下步骤计算出转置图 G T = ( V , E T ) G^T=(V,E^T) GT=(V,ET)的邻接链表表示:

  1. 初始化一个新的空邻接链表表示的转置图 G T G^T GT

  2. 对于有向图 G G G中的每个顶点 v ∈ V v\in V vV,遍历其邻接点列表中的每个邻接点 u u u

  3. 对于每个邻接点 u u u,将边 ( u , v ) (u,v) (u,v)添加到 G T G^T GT的邻接链表中,即将 u u u加入到 v v v的邻接点列表中。

下面是用Go语言实现上述算法的代码示例:

package main

import "fmt"

// 定义有向图的顶点结构
type Vertex struct {
    value int
    next *Vertex
}

// 定义有向图的邻接链表结构
type Graph struct {
    V int
    Adj []*Vertex
}

// 函数实现有向图的转置操作
func transposeGraph(g *Graph) *Graph {
    transposed := &Graph{V: g.V, Adj: make([]*Vertex, g.V)}

    for u := 0; u < g.V; u++ {
        for v := g.Adj[u]; v != nil; v = v.next {
            transposed.Adj[v.value] = &Vertex{value: u, next: transposed.Adj[v.value]}
        }
    }

    return transposed
}

func main() {
    // 举例一个有向图的邻接链表表示
    V := 4
    adj := make([]*Vertex, V)
    adj[0] = &Vertex{value: 1, next: &Vertex{value: 2, next: nil}}
    adj[1] = &Vertex{value: 2, next: nil}
    adj[2] = &Vertex{value: 0, next: &Vertex{value: 3, next: nil}}
    adj[3] = nil

    g := &Graph{V: V, Adj: adj}

    // 计算有向图的转置图
    transposed := transposeGraph(g)

    // 输出转置图的邻接链表表示
    for i := 0; i < transposed.V; i++ {
        fmt.Printf("顶点 %d 的邻接点: ", i)
        for v := transposed.Adj[i]; v != nil; v = v.next {
            fmt.Printf("%d ", v.value)
        }
        fmt.Println()
    }
}

该算法的运行时间为 O ( V + E ) O(V+E) O(V+E),其中 V V V为顶点数, E E E为边数。

chatgpt:

以下是使用邻接链表表示的图 G G G计算出 G T G^{T} GT的有效算法,并给出了相应的Go语言代码示例。

算法步骤:

  1. 创建一个空的图 G T G^{T} GT
  2. 遍历原图 G G G中的每个顶点 v v v,对于每个顶点 v v v,执行以下操作:
    • 如果 v v v在邻接链表中不存在,则将其加入邻接链表中
    • 遍历与顶点 v v v相关联的边 ( v , u ) (v, u) (v,u),其中 u u u为目标顶点。对于每条边 ( v , u ) (v, u) (v,u),将其反向添加到对应的邻接链表中:即将边 ( u , v ) (u, v) (u,v)添加到图 G T G^{T} GT的邻接链表中

代码示例(Go语言):

package main

import "fmt"

type Node struct {
    value int
    next  *Node
}

type Graph struct {
    vertices []*Node
}

func NewGraph() *Graph {
    return &Graph{}
}

func (g *Graph) addEdge(src, dest int) {
    newNode := &Node{value: dest, next: nil}
    if g.vertices[src] == nil {
        g.vertices[src] = newNode
    } else {
        curr := g.vertices[src]
        for curr.next != nil {
            curr = curr.next
        }
        curr.next = newNode
    }
}

func (g *Graph) transpose() *Graph {
    transposedGraph := NewGraph()
  
    for i := range g.vertices {
        transposedGraph.vertices = append(transposedGraph.vertices, nil)
    }

    for i := range g.vertices {
        curr := g.vertices[i]
        for curr != nil {
            transposedGraph.addEdge(curr.value, i)
            curr = curr.next
        }
    }
  
    return transposedGraph
}

func main() {
    graph := NewGraph()
    verticesCount := 5
  
    for i := 0; i < verticesCount; i++ {
        graph.vertices = append(graph.vertices, nil)
    }

    // 添加边
    graph.addEdge(0, 1)
    graph.addEdge(0, 2)
    graph.addEdge(1, 3)
    graph.addEdge(2, 3)
    graph.addEdge(3, 4)

    fmt.Println("原图 G:")
    
	  for i, vertex := range graph.vertices {
        curr := vertex
        for curr != nil {
            fmt.Printf("%d -> %d\n", i, curr.value)
            curr = curr.next
        }
	  }

	  transposedGraph := graph.transpose()

	  fmt.Println("\n转置图 G^T:")

	  for i, vertex := range transposedGraph.vertices {
		    curr := vertex
		    for curr != nil {
			      fmt.Printf("%d -> %d\n", i, curr.value)
			      curr = curr.next
		    }
	  }
}

该算法的运行时间取决于顶点数和边数。假设有 n n n个顶点和 m m m条边,则构建邻接链表的时间复杂度为 O ( m ) O(m) O(m),转置操作需要遍历整个邻接链表,时间复杂度为 O ( n + m ) O(n+m) O(n+m)。因此,总体运行时间复杂度为 O ( n + m ) O(n+m) O(n+m)

注意:这只是一种实现方式,具体的算法细节和实现可能因应用场景和数据结构的不同而有所调整。

kimi:

首先,我们来分析两种图的表示方法:邻接链表和邻接矩阵,以及如何从图 ( G ) 计算出其转置图 ( G^T )。

邻接链表表示

邻接链表通常使用一个数组来表示,数组的每个元素是一个链表,链表中的每个节点代表一个邻接点和边的权重(如果有的话)。对于有向图 ( G ),如果 ( u ) 到 ( v ) 有一条边,那么在 ( u ) 的链表中会有一个指向 ( v ) 的节点。

算法步骤:

  1. 初始化一个新的邻接链表,其大小与 ( G ) 相同。
  2. 遍历原始图 ( G ) 的每个顶点 ( u )。
  3. 对于 ( u ) 的每个邻接点 ( v ),将 ( v ) 添加到 ( G^T ) 中 ( u ) 的链表中。
  4. 完成所有顶点的遍历后,( G^T ) 的邻接链表就构建好了。

时间复杂度:

  • 假设 ( G ) 有 ( n ) 个顶点和 ( m ) 条边。
  • 每个顶点的邻接链表遍历时间复杂度为 ( O(\text{deg}(u)) ),其中 ( \text{deg}(u) ) 是顶点 ( u ) 的度。
  • 总的时间复杂度为 ( O(m) ),因为每个边只被访问一次。

邻接矩阵表示

邻接矩阵是一个 ( n \times n ) 的矩阵,其中 ( n ) 是顶点的数量。如果 ( G ) 中 ( u ) 到 ( v ) 有边,则矩阵中的 ( G[u][v] ) 为 1(或边的权重),否则为 0。

算法步骤:

  1. 初始化一个与 ( G ) 同样大小的矩阵 ( G^T ),并将所有元素设置为 0。
  2. 遍历 ( G ) 的邻接矩阵,对于每个非零元素 ( G[u][v] ):
    • 将 ( G^T[v][u] ) 设置为 ( G[u][v] )。
  3. 完成所有元素的遍历后,( G^T ) 的邻接矩阵就构建好了。

时间复杂度:

  • 由于矩阵是 ( n \times n ),总的时间复杂度是 ( O(n^2) )。

Go 语言实现

以下是使用 Go 语言实现的邻接链表和邻接矩阵转置的示例代码:

package main

import "fmt"

// 邻接链表的节点
type AdjListNode struct {
    vertex int
    next   *AdjListNode
}

// 邻接链表的图
type GraphAdjList struct {
    adjList []*AdjListNode
    V       int // 顶点的数量
}

// 添加边
func (g *GraphAdjList) addEdge(v, w int) {
    newEdge := &AdjListNode{vertex: w, next: g.adjList[v]}
    g.adjList[v] = newEdge
}

// 计算转置图的邻接链表
func (g *GraphAdjList) transpose() *GraphAdjList {
    gT := &GraphAdjList{V: g.V, adjList: make([]*AdjListNode, g.V)}
    for v := 0; v < g.V; v++ {
        for adj := g.adjList[v]; adj != nil; adj = adj.next {
            gT.addEdge(adj.vertex, v)
        }
    }
    return gT
}

// 邻接矩阵的图
type GraphAdjMatrix struct {
    matrix [][]int
    V      int // 顶点的数量
}

// 计算转置图的邻接矩阵
func (g *GraphAdjMatrix) transpose() *GraphAdjMatrix {
    gT := &GraphAdjMatrix{V: g.V, matrix: make([][]int, g.V)}
    for i := 0; i < g.V; i++ {
        gT.matrix[i] = make([]int, g.V)
        for j := 0; j < g.V; j++ {
            gT.matrix[i][j] = g.matrix[j][i]
        }
    }
    return gT
}

func main() {
    // 示例:使用邻接链表和邻接矩阵构建图并计算其转置
    // 这里省略了图的构建过程,直接展示转置操作
    // ...
}

请注意,上述代码只是一个简单的示例,实际应用中可能需要更多的功能,例如图的构建、边的权重处理等。

混元:

为了计算有向图 G = ( V , E ) G=(V,E) G=(V,E) 的转置 G T = ( V , E T ) G^T=(V,E^T) GT=(V,ET),我们可以分别针对邻接链表和邻接矩阵两种表示方法给出相应的算法,并分析它们的运行时间。

邻接链表表示法

算法描述
  1. 初始化一个空的邻接链表来存储转置图的边。
  2. 遍历原图 G G G 的每个节点 u u u
  3. 对于节点 u u u 的每个邻接节点 v v v,在转置图中添加一条从 v v v u u u 的边。
  4. 返回转置图的邻接链表表示。
Go 语言实现
type AdjacencyList struct {
    V int
    Adj [][]int
}

func TransposeAdjacencyList(graph AdjacencyList) AdjacencyList {
    V := graph.V
    transposed := make([][]int, V)
    
    for u := 0; u < V; u++ {
        for _, v := range graph.Adj[u] {
            transposed[v] = append(transposed[v], u)
        }
    }
    
    return AdjacencyList{V: V, Adj: transposed}
}
运行时间分析
  • 时间复杂度: O ( ∣ E ∣ ) O(|E|) O(E),其中 ∣ E ∣ |E| E 是原图的边数。因为我们需要遍历每条边来构建转置图。

邻接矩阵表示法

算法描述
  1. 初始化一个与原图 G G G 大小相同的邻接矩阵来存储转置图的边。
  2. 遍历原图 G G G 的每个节点对 ( u , v ) (u, v) (u,v)
  3. 如果 ( u , v ) (u, v) (u,v) 在原图中存在边,则在转置图的邻接矩阵中将 ( v , u ) (v, u) (v,u) 的对应位置设为边的权重(通常为1)。
  4. 返回转置图的邻接矩阵表示。
Go 语言实现
type AdjacencyMatrix struct {
    V int
    Mat [][]int
}

func TransposeAdjacencyMatrix(graph AdjacencyMatrix) AdjacencyMatrix {
    V := graph.V
    transposed := make([][]int, V)
    for i := range transposed {
        transposed[i] = make([]int, V)
    }
    
    for u := 0; u < V; u++ {
        for v := 0; v < V; v++ {
            if graph.Mat[u][v] != 0 {
                transposed[v][u] = graph.Mat[u][v]
            }
        }
    }
    
    return AdjacencyMatrix{V: V, Mat: transposed}
}
运行时间分析
  • 时间复杂度: O ( ∣ V ∣ 2 ) O(|V|^2) O(V2),其中 ∣ V ∣ |V| V 是原图的节点数。因为我们需要遍历矩阵中的每个元素来构建转置图。

在实际应用中,如果图的边数远小于节点数的平方,那么使用邻接链表表示法会更高效;反之,如果图的边数接近或超过节点数的平方,邻接矩阵表示法可能更合适。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1910778.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

PostgreSQL 如何解决数据迁移过程中的数据类型不匹配问题?

文章目录 一、了解常见的数据类型不匹配情况1. 整数类型差异2. 浮点数类型差异3. 字符类型差异4. 日期和时间类型差异 二、解决数据类型不匹配的一般策略1. 数据转换2. 调整数据库表结构3. 数据清洗和预处理 三、PostgreSQL 中的数据类型转换函数1. 数值类型转换2. 字符类型转换…

第17章 潜在语义分析:奇异值分解、非负矩阵分解及课后习题

1.单词向量空间模型通过单词的向量表示文本的语义内容。以单词-文本矩阵 X X X为输入&#xff0c;其中每一行对应一个单词&#xff0c;每一列对应一个文本&#xff0c;每一个元素表示单词在文本中的频数或权值&#xff08;如TF-IDF&#xff09; X [ x 11 x 12 ⋯ x 1 n x 21 …

nodejs 获取客服端ip,以及获取ip一直都是127.0.0.1的问题

一、问题描述 在做登录日志的时候想要获取客户端的ip, 网上查了一下 通过 req.headers[x-forwarded-for] || req.connection.remoteAddress; 获取&#xff0c; 结果获取了之后不管是开发环境&#xff0c;还是生产环境获取到的一直都是 127.0.0.1&#xff0c;这是因为在配置N…

Python基础知识——(002)

文章目录 P8——7. input函数的使用 基本的输入函数input P9——8. Python中的注释 P10——9. Python中的缩进与本章总结 本章总结 P11——10. 章节习题 P8——7. input函数的使用 基本的输入函数input 语法结构&#xff1a; x input(提示文字) 注意事项&#xff1a;无论输…

注册商标为什么要先查询

注册商标为什么要先查询 在知识产权日益受到重视的今天&#xff0c;商标的注册成为了许多企业和个人保护其品牌价值和市场地位的重要手段。然而&#xff0c;商标注册并非一蹴而就的过程&#xff0c;其中一个关键的步骤就是商标查询&#xff0c;也就是我们通常所说的“商标检索…

《梦醒蝶飞:释放Excel函数与公式的力量》10.1.1函数简介

10.1.1函数简介 BIN2DEC函数是Excel中用于将二进制数转换为十进制数的函数。它在处理二进制数时非常有用&#xff0c;尤其是在电子工程、计算机科学等领域。 10.1.2函数语法&#xff1a; BIN2DEC(number) number&#xff1a;这是要转换的二进制数&#xff0c;必须是以字符串…

多文件编程:c/c++分文件写法(入门)

前言 一个 C 项目通常会采取 声明与定义分离 的方式进行编写&#xff0c;其基本遵循&#xff1a;头文件中写声明&#xff0c;源文件中写定义。 此外&#xff0c;为了区分头文件与源文件&#xff0c;会采用不同的文件后缀&#xff1a; .h: 头文件 .cpp: 源文件 (当然还有其他的…

【SQL】做项目时用到的语句整理(去重/多表关联)

1. 对日期去重&#xff08;groupby&#xff09; 需要&#xff1a;新建一张表&#xff0c;对原来表中的某个列(href)进行去重&#xff0c;并按照最新的日期进行排版 适用&#xff1a;如果有一张表&#xff0c;我们重复往里面存入数据&#xff0c;有一些除了日期以外&#xff0…

【大模型LLM面试合集】大语言模型架构_MoE论文

1.MoE论文 参考文章&#xff1a; Mixture of Experts-IntroductionUnderstanding the Mixture-of-Experts Model in Deep Learning 论文相关&#xff1a; 论文名称&#xff1a;Outrageously Large Neural Networks: The Sparsely-Gated Mixture-of-Experts Layer论文地址&a…

上传图片,base64改为文件流,并转给后端

需求&#xff1a; html代码&#xff1a; <el-dialog v-model"dialogPicVisible" title"新增图片" width"500"><el-form :model"picForm"><el-form-item label"图片名称&#xff1a;" :label-width"10…

手把手带你本地部署大模型

这篇文章的唯一目的是实现在本地运行大模型&#xff0c;我们使用LMStudio这个工具协助达成这个目标。 文章目录 一&#xff0c;下载安装LM Studio二&#xff0c;本地部署大模型1&#xff0c;搜索模型2&#xff0c;下载大模型3&#xff0c;加载大模型4&#xff0c;测试大模型5&a…

STM32Cubemx配置生成 Keil AC6支持代码

文章目录 一、前言二、AC 6配置2.1 ARM ComPiler 选择AC62.2 AC6 UTF-8的编译命令会报错 三、STM32Cubemx 配置3.1 找到stm32cubemx的模板位置3.2 替换文件内核文件3.3 修改 cmsis_os.c文件3.4 修改本地 四、编译对比 一、前言 使用keil ARM compiler V5的时候&#xff0c;编译…

解读‘‘不要卷模型,要卷应用‘‘

前言 2024 年 7 月 4 日&#xff0c;世界人工智能大会暨人工智能全球治理高级别会议全体会议在上海世博中心举行。百度创始人李彦宏在产业发展主论坛上发言&#xff0c;呼吁不要卷模型&#xff0c;要卷应用。 目录 四个要点 积极的观点 不合理性 总结 四个要点 李彦宏的呼吁…

PMON的解读和开发

提示&#xff1a;龙芯2K1000PMON相关记录 文章目录 1 PMON的发展和编译环境PMONPMON2000 2 PMON2000的目录结构3 Targets目录的组成4 PMON编译环境的建立5 PMON2000的框架6 异常向量表7 Pmon的空间分配8 PMON的汇编部分(starto.S或sbdreset.S)的解读Start.SC代码部分dbginit 9 …

SpringCloud Alibaba Sentinel网关流量控制实践总结

官网地址&#xff1a;https://sentinelguard.io/zh-cn/docs/api-gateway-flow-control.html GitHub地址&#xff1a;GitHub Sentinel 网关限流 【1】概述 Sentinel 支持对 Spring Cloud Gateway、Zuul 等主流的 API Gateway 进行限流。 Sentinel 1.6.0 引入了 Sentinel API …

命令执行(RCE)面对各种过滤,骚姿势绕过总结

1、什么是RCE RCE又称远程代码执行漏洞&#xff0c;可以让攻击者直接向后台服务器远程注入操作系统命令或者代码&#xff0c;从而控制后台系统。 2、RCE产生原因 服务器没有对执行命令的函数做严格的过滤&#xff0c;最终导致命令被执行。 3、命令执行函数 PHP代码执行函数…

潜在空间可视化(Latent space visualization)

在“深度学习”系列中&#xff0c;我们不会看到如何使用深度学习来解决端到端的复杂问题&#xff0c;就像我们在《A.I. Odyssey》中所做的那样。我们更愿意看看不同的技术&#xff0c;以及一些示例和应用程序。 1、引言 上次&#xff08;Autoencoders - Deep Learning bits #…

PLC物联网关在工业自动化领域的应用的意义-天拓四方

随着信息技术的飞速发展&#xff0c;物联网技术正逐步渗透到各个行业领域&#xff0c;其中&#xff0c;工业自动化领域的PLC与物联网的结合&#xff0c;为工业自动化的发展开辟了新的道路。PLC物联网关作为连接PLC与物联网的重要桥梁&#xff0c;其重要性日益凸显。 PLC物联网…

STM32 低功耗模式 睡眠、停止和待机 详解

目录 1.睡眠模式&#xff08;Sleep Mode&#xff09; 2.停止模式&#xff08;stop mode&#xff09; 3.待机模式&#xff08;Standby Mode&#xff09; STM32提供了三种低功耗模式&#xff0c;分别是睡眠模式&#xff08;Sleep Mode&#xff09;、停止模式&#xff08;Stop …

Python基于you-get下载网页上的视频

​ 1.python 下载地址 下载 : https://www.python.org/downloads/ 2. 配置环境变量 配置 python_home 地址 配置 python_scripts 地址 在path 中加入对应配置 3. 验证 ​ C:\Users>python --version Python 3.12.4C:\Users>wheel version wheel 0.43.04. 下载 c…