22

我通常使用

matrix[:, i:]

它似乎没有我预期的那么快。

4

2 回答 2

20

如果您想获得一个稀疏矩阵作为输出,进行行切片的最快方法是拥有一个csr类型,并且对于列切片csc如此处详述。在这两种情况下,你只需要做你目前正在做的事情:

matrix[l1:l2,c1:c2]

如果您想要另一种类型作为输出,则可能有更快的方法。在这个其他答案中,解释了许多对矩阵进行切片的方法,并比较了它们的不同时间。例如,如果您想要一个ndarray作为输出,最快的切片是:

matrix.A[l1:l2,c1:c2] 

或者:

matrix.toarray()[l1:l2,c1:c2]

比:

matrix[l1:l2,c1:c2].A #or .toarray()
于 2013-06-02T17:15:39.363 回答
10

我发现scipy.sparse.csr_matrix通过滚动您自己的行索引器可以更快地进行宣传的快速行索引。这是想法:

class SparseRowIndexer:
    def __init__(self, csr_matrix):
        data = []
        indices = []
        indptr = []

        # Iterating over the rows this way is significantly more efficient
        # than csr_matrix[row_index,:] and csr_matrix.getrow(row_index)
        for row_start, row_end in zip(csr_matrix.indptr[:-1], csr_matrix.indptr[1:]):
             data.append(csr_matrix.data[row_start:row_end])
             indices.append(csr_matrix.indices[row_start:row_end])
             indptr.append(row_end-row_start) # nnz of the row

        self.data = np.array(data)
        self.indices = np.array(indices)
        self.indptr = np.array(indptr)
        self.n_columns = csr_matrix.shape[1]

    def __getitem__(self, row_selector):
        data = np.concatenate(self.data[row_selector])
        indices = np.concatenate(self.indices[row_selector])
        indptr = np.append(0, np.cumsum(self.indptr[row_selector]))

        shape = [indptr.shape[0]-1, self.n_columns]

        return sparse.csr_matrix((data, indices, indptr), shape=shape)

也就是说,可以通过将每行的非零值存储在单独的数组中(每行具有不同的长度)并将所有这些行数组放入对象类型数组中来利用 numpy 数组的快速索引(允许每一行具有不同的大小),可以有效地建立索引。列索引的存储方式相同。该方法与标准 CSR 数据结构略有不同,标准 CSR 数据结构将所有非零值存储在单个数组中,需要查找以查看每一行的开始和结束位置。这些查找可以减慢随机访问速度,但对于检索连续行应该是有效的。

分析结果

我的矩阵mat是 1,900,000x1,250,000 csr_matrix,包含 400,000,000 个非零元素。 ilocs是一个包含 200,000 个随机行索引的数组。

>>> %timeit mat[ilocs]
2.66 s ± 233 ms per loop (mean ± std. dev. of 7 runs, 1 loop each)

相比:

>>> row_indexer = SparseRowIndexer(mat)
>>> %timeit row_indexer[ilocs]
59.9 ms ± 4.51 ms per loop (mean ± std. dev. of 7 runs, 10 loops each)

与布尔掩码相比,使用花式索引时,SparseRowIndexer 似乎更快。

于 2018-06-14T14:52:58.500 回答