【问题标题】:Get bad result for random walk随机游走得到不好的结果
【发布时间】:2018-09-22 17:06:03
【问题描述】:

我想实现随机游走并计算稳态。

假设我的图表如下图所示:

上图在一个文件中定义如下:

1   2   0.9
1   3   0.1
2   1   0.8
2   2   0.1
2   4   0.1
etc

为了阅读和构建这个图表,我使用了以下方法:

def _build_og(self, original_ppi):
""" Build the original graph, without any nodes removed. """

try:
    graph_fp = open(original_ppi, 'r')
except IOError:
    sys.exit("Could not open file: {}".format(original_ppi))

G = nx.DiGraph()
edge_list = []

# parse network input
for line in graph_fp.readlines():
    split_line = line.rstrip().split('\t')
    # assume input graph is a simple edgelist with weights
    edge_list.append((split_line[0], split_line[1],  float(split_line[2])))

G.add_weighted_edges_from(edge_list)
graph_fp.close()
print edge_list

return G

在上面的函数中,我需要将图定义为有向图还是简单图?

我们构建转换矩阵如下:

def _build_matrices(self, original_ppi, low_list, remove_nodes):
        """ Build column-normalized adjacency matrix for each graph.

        NOTE: these are column-normalized adjacency matrices (not nx
              graphs), used to compute each p-vector
        """
        original_graph = self._build_og(original_ppi)
        self.OG = original_graph
        og_not_normalized = nx.to_numpy_matrix(original_graph)
        self.og_matrix = self._normalize_cols(og_not_normalized)

我使用以下方法对矩阵进行归一化:

def _normalize_cols(self, matrix):
        """ Normalize the columns of the adjacency matrix """
        return normalize(matrix, norm='l1', axis=0)

现在来模拟我们定义的随机游走:

def run_exp(self, source):

        CONV_THRESHOLD = 0.000001
        # set up the starting probability vector
        p_0 = self._set_up_p0(source)
        diff_norm = 1
        # this needs to be a deep copy, since we're reusing p_0 later
        p_t = np.copy(p_0)

        while (diff_norm > CONV_THRESHOLD):
            # first, calculate p^(t + 1) from p^(t)
            p_t_1 = self._calculate_next_p(p_t, p_0)

            # calculate L1 norm of difference between p^(t + 1) and p^(t),
            # for checking the convergence condition
            diff_norm = np.linalg.norm(np.subtract(p_t_1, p_t), 1)

            # then, set p^(t) = p^(t + 1), and loop again if necessary
            # no deep copy necessary here, we're just renaming p
            p_t = p_t_1

我们使用以下方法定义初始状态(p_0):

def _set_up_p0(self, source):
    """ Set up and return the 0th probability vector. """
    p_0 = [0] * self.OG.number_of_nodes()
    # convert self.OG.number_of_nodes() to list
    l =  list(self.OG.nodes())
    #nx.draw(self.OG, with_labels=True)
    #plt.show()
    for source_id in source:
        try:
            # matrix columns are in the same order as nodes in original nx
            # graph, so we can get the index of the source node from the OG
            source_index = l.index(source_id)
            p_0[source_index] = 1 / float(len(source))
        except ValueError:
            sys.exit("Source node {} is not in original graph. Source: {}. Exiting.".format(source_id, source))

    return np.array(p_0)  

要生成下一个状态,我们使用以下函数

以及幂次迭代策略:

def _calculate_next_p(self, p_t, p_0):
        """ Calculate the next probability vector. """
        print 'p_0\t{}'.format(p_0)
        print 'p_t\t{}'.format(p_t)
        epsilon = np.squeeze(np.asarray(np.dot(self.og_matrix, p_t)))
        print 'epsilon\t{}'.format(epsilon)
        print 10*"*"
        return np.array(epsilon)

假设随机游走可以从任何节点(1、2、3 或 4)开始。

运行代码时,我得到以下结果:

2       0.32
3       0.31
1       0.25
4       0.11

结果必须是:

(0.28, 0.30, 0.04, 0.38).

那么有人可以帮我找出我的错误在哪里吗?

我不知道问题是否出在我的转换矩阵中。

【问题讨论】:

  • 1.您的问题中有大量代码。你能简化一些事情,以便我们更好地了解发生了什么吗?从您的代码中复制/粘贴更容易,但它使我们更难以理解 2. 不清楚什么是“不同”。您能否绘制或解释结果有何不同以及预期的输出是什么?
  • @cd98 我想计算给定有向图的稳定分布向量。该向量必须等于 [0.35, 0.56, 0.07]。但在我的结果中,我得到 [0.41, 0.35, 0.24]。

标签: python networkx random-walk


【解决方案1】:

这是矩阵应该是什么(假设你的转换矩阵从左边乘以状态向量,它是一个左随机矩阵,其中列加起来为 1,@987654321 @entry 是从ji 的概率。

import numpy as np
transition = np.array([[0, 0.8, 0, 0.1], [0.9, 0.1, 0.5, 0], [0.1, 0, 0.3, 0], [0, 0.1, 0.2, 0.9]])
state = np.array([1, 0, 0, 0])    # could be any other initial position
diff = tol = 0.001
while diff >= tol:
    next_state = transition.dot(state)
    diff = np.linalg.norm(next_state - state, ord=np.inf)
    state = next_state
print(np.around(state, 3))

这打印出[0.279 0.302 0.04 0.378]

我不知道您是否错误地加载了数据,或者其他什么。 “列标准化”的步骤是一个警告信号:如果给定的转换概率加起来不等于 1,您应该报告错误数据,而不是对列进行标准化。当数据已经以矩阵形式呈现时,我完全不知道为什么要使用 NetworkX:给定的表格可以读作

column   row   entry 

这个矩阵就是计算所需要的。

【讨论】:

  • 感谢您的回答。我不知道您如何以这种形式获得转换矩阵。我有一个 .txt 文件,其中包含节点和它们之间的弧。 1 2 0.9 表示节点 1 和节点 2 之间的权重为 0.9 的弧。
  • 对,所以transition[1, 0] = 0.9(基于 0 的索引的偏移量)。矩阵的第一行是从节点 1、2、3、4 到 节点 1 的概率。第二行:从节点 1、2 到 节点 2 的概率, 3, 4. 等等。
猜你喜欢
  • 1970-01-01
  • 2014-03-26
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2021-07-29
  • 2015-06-16
  • 1970-01-01
相关资源
最近更新 更多