0

0

PyTorch张量中高效查找唯一行首次出现索引的优化方法

心靈之曲

心靈之曲

发布时间:2025-10-04 09:13:07

|

487人浏览过

|

来源于php中文网

原创

PyTorch张量中高效查找唯一行首次出现索引的优化方法

本文介绍了一种在PyTorch张量中高效查找各唯一行首次出现索引的方法。通过利用torch.unique的逆索引结果,并结合构建辅助二维张量及使用torch.argmin操作,可以避免低效的Python循环,显著提升处理大规模数据的性能。文章详细阐述了优化思路、实现代码及性能考量。

问题描述

pytorch中处理数据时,我们经常需要识别张量中的唯一行。更进一步地,有时我们需要获取这些唯一行在原始张量中首次出现的索引。例如,给定一个形状为 (n, d) 的二维张量,其中 n 是行数,d 是特征维度,目标是找到一个索引列表,其中每个索引对应于某个唯一行在原始张量中第一次出现的位置。

初始实现与效率瓶颈

一个直观的实现方式是首先使用torch.unique函数找出所有唯一行及其对应的逆索引,然后通过遍历这些唯一行,利用torch.where来查找每个唯一行在逆索引张量中首次出现的位置。以下是这种方法的示例代码:

import torch
import numpy as np

# 示例张量
data = torch.rand(100, 5)
# 随机复制一些行,制造重复数据
data[np.random.choice(100, 50, replace=False)] = torch.tensor([1.0, 2.0, 3.0, 4.0, 5.0])

# 查找唯一行及其相关信息
# u_data: 唯一行张量
# inverse_indices: 原始张量中每行对应的唯一行索引
# counts: 每个唯一行出现的次数
u_data, inverse_indices, counts = torch.unique(data, dim=0, return_inverse=True, return_counts=True)

# 查找每个唯一行首次出现的索引(低效循环方法)
unique_indices_loop = torch.zeros(len(u_data), dtype=torch.long)
for idx in range(len(u_data)):
    # 对于每个唯一行索引idx,在inverse_indices中找到其首次出现的位置
    unique_indices_loop[idx] = torch.where(inverse_indices == idx)[0][0]

print(f"通过循环找到的首次出现索引: {unique_indices_loop}")

尽管上述代码功能正确,但其核心问题在于使用了Python级别的for循环。在PyTorch这类高度优化的张量计算框架中,Python循环通常会导致显著的性能瓶颈,尤其是在处理大规模数据时。torch.where函数在循环内部的反复调用也增加了计算开销。为了提高效率,我们需要寻找一种完全基于张量操作的解决方案,以充分利用PyTorch的并行计算能力。

优化的PyTorch方法

为了避免低效的Python循环,我们可以将问题转化为一个二维张量操作。核心思想是构建一个辅助张量,巧妙地利用inverse_indices来标记原始行与唯一行之间的映射关系,然后通过argmin操作高效地找到首次出现的索引。

核心思想

  1. 利用 inverse_indices: torch.unique返回的inverse_indices张量记录了原始张量中每一行对应的是哪一个唯一行(其在u_data中的索引)。
  2. 构建辅助张量 A: 创建一个形状为 (原始行数, 唯一行数) 的二维张量 A。
  3. 填充 A: 将 A 初始化为一个足够大的占位符值(例如,大于原始行数的任意整数)。然后,对于原始张量中的每一行 i,将其对应的唯一行索引 inverse_indices[i] 赋值给 A[i, inverse_indices[i]]。这里的赋值操作实际上是将原始行索引 i 写入到辅助张量中对应唯一行的列。
  4. 使用 argmin: 对张量 A 沿着列(dim=0)执行 argmin 操作。对于每一列 j(代表第 j 个唯一行),argmin将返回该列中最小值所在的行索引。由于我们只在原始行 i 对应唯一行 j 的位置 A[i, j] 写入了 i,而其他位置是较大的占位符值,因此argmin将准确地找到第一个将自身映射到唯一行 j 的原始行索引。

步骤详解与示例代码

我们将使用一个足够大的值(例如,len(data) 或更大的一个数,如 1000)作为占位符,确保它大于任何可能的原始行索引。

import torch
import numpy as np

# 示例张量(与前文相同)
data = torch.rand(100, 5)
data[np.random.choice(100, 50, replace=False)] = torch.tensor([1.0, 2.0, 3.0, 4.0, 5.0])

# 查找唯一行及其相关信息
u_data, inverse_indices, counts = torch.unique(data, dim=0, return_inverse=True, return_counts=True)

# --- 优化的PyTorch方法 ---

# 1. 初始化辅助张量 A
# A 的维度为 (原始行数, 唯一行数)
# 使用一个大于原始行数的占位符值(如 len(data) 或 1000)进行初始化
# 这里使用 len(data) 作为占位符,因为原始行索引最大为 len(data) - 1
placeholder_val = len(data) 
A = placeholder_val * torch.ones((len(data), len(u_data)), dtype=torch.long)

# 2. 填充辅助张量 A
# 对于原始张量中的每一行 i (由 torch.arange(len(data)) 生成),
# 找到其对应的唯一行索引 inverse_indices[i]。
# 然后在 A[i, inverse_indices[i]] 的位置上写入原始行索引 i。
# 这里的赋值操作 A[torch.arange(len(data)), inverse_indices] = torch.arange(len(data))
# 实际上是将原始行索引本身作为值写入,而不是 inverse_indices。
# 修正:应写入原始行索引 i,而不是 inverse_indices[i]
A[torch.arange(len(data)), inverse_indices] = torch.arange(len(data))

# 3. 对 A 沿着列方向执行 argmin 操作
# 对于 A 的每一列 j(代表第 j 个唯一行),argmin(A[:, j]) 将返回该列中最小值的行索引。
# 由于我们只在 A[i, j] 处写入了 i,且其他地方是更大的占位符,
# 因此 argmin 会找到第一个将自身映射到唯一行 j 的原始行索引。
unique_indices_optimized = torch.argmin(A, dim=0)

print(f"通过优化方法找到的首次出现索引: {unique_indices_optimized}")

# 验证两种方法的结果是否一致
# 为了验证,我们还需要运行之前的循环方法
unique_indices_loop = torch.zeros(len(u_data), dtype=torch.long)
for idx in range(len(u_data)):
    unique_indices_loop[idx] = torch.where(inverse_indices == idx)[0][0]

print(f"两种方法结果是否一致: {torch.allclose(unique_indices_optimized.float(), unique_indices_loop.float())}")

通过这种方法,我们成功地将 Python 循环替换为高效的 PyTorch 张量操作,显著提升了查找唯一行首次出现索引的效率。

稿定AI绘图
稿定AI绘图

稿定推出的AI绘画工具

下载

效率与内存权衡

这种优化的方法虽然避免了 Python 循环,但引入了一个新的辅助张量 A。这个张量的尺寸是 (原始行数, 唯一行数),它可能比原始数据张量占用更多的内存。

  • 计算效率: torch.argmin 是一个高度优化的操作,通常能够充分利用 GPU 或多核 CPU 的并行计算能力,因此在计算速度上远超 Python 循环。
  • 内存消耗: 当原始行数 len(data) 和唯一行数 len(u_data) 都非常大时,辅助张量 A 可能会占用大量内存。在内存受限的环境下,这可能成为一个需要权衡的因素。

在实际应用中,如果数据量巨大且内存是瓶颈,可能需要考虑其他策略,例如分块处理或更复杂的算法。但对于大多数常见场景,这种无循环的张量化方法是首选,因为它在计算速度上提供了显著的优势。

总结

本文详细介绍了在 PyTorch 中高效查找唯一行首次出现索引的优化方法。通过利用 torch.unique 函数的 inverse_indices 输出,并结合构建一个辅助二维张量以及使用 torch.argmin 操作,我们能够将原本低效的 Python 循环转换为高性能的张量操作。这种方法在处理大规模数据时具有显著的计算效率优势,但需要注意其潜在的内存消耗。在选择具体实现方案时,应根据实际数据规模和硬件资源进行综合考量。

相关专题

更多
python开发工具
python开发工具

php中文网为大家提供各种python开发工具,好的开发工具,可帮助开发者攻克编程学习中的基础障碍,理解每一行源代码在程序执行时在计算机中的过程。php中文网还为大家带来python相关课程以及相关文章等内容,供大家免费下载使用。

715

2023.06.15

python打包成可执行文件
python打包成可执行文件

本专题为大家带来python打包成可执行文件相关的文章,大家可以免费的下载体验。

625

2023.07.20

python能做什么
python能做什么

python能做的有:可用于开发基于控制台的应用程序、多媒体部分开发、用于开发基于Web的应用程序、使用python处理数据、系统编程等等。本专题为大家提供python相关的各种文章、以及下载和课程。

739

2023.07.25

format在python中的用法
format在python中的用法

Python中的format是一种字符串格式化方法,用于将变量或值插入到字符串中的占位符位置。通过format方法,我们可以动态地构建字符串,使其包含不同值。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

617

2023.07.31

python教程
python教程

Python已成为一门网红语言,即使是在非编程开发者当中,也掀起了一股学习的热潮。本专题为大家带来python教程的相关文章,大家可以免费体验学习。

1235

2023.08.03

python环境变量的配置
python环境变量的配置

Python是一种流行的编程语言,被广泛用于软件开发、数据分析和科学计算等领域。在安装Python之后,我们需要配置环境变量,以便在任何位置都能够访问Python的可执行文件。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

547

2023.08.04

python eval
python eval

eval函数是Python中一个非常强大的函数,它可以将字符串作为Python代码进行执行,实现动态编程的效果。然而,由于其潜在的安全风险和性能问题,需要谨慎使用。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

575

2023.08.04

scratch和python区别
scratch和python区别

scratch和python的区别:1、scratch是一种专为初学者设计的图形化编程语言,python是一种文本编程语言;2、scratch使用的是基于积木的编程语法,python采用更加传统的文本编程语法等等。本专题为大家提供scratch和python相关的文章、下载、课程内容,供大家免费下载体验。

698

2023.08.11

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

3

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 0.6万人学习

Django 教程
Django 教程

共28课时 | 2.6万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.0万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号