如何高效地从二维张量的每一行中提取不同起始位置的固定长度子序列

发布时间 - 2026-01-06 00:00:00    点击率:

本文介绍一种无需显式循环即可从 pytorch 二维张量各行中按指定起始索引和统一长度批量切片的方法,核心是利用 `torch.arange` 构造索引张量,并通过 `gather` 实现向量化索引选取。

在深度学习与科学计算中,常需对批量数据(如序列、特征图)进行行级动态切片——即每行按各自起始位置截取固定长度的子序列。若使用 Python 循环或列表推导式逐行处理,不仅代码冗长,还会严重损害 GPU 加速优势。幸运的是,PyTorch 提供了完全向量化的解决方案:torch.gather + 动态索引张量构造

其关键思想是:将每行的切片需求(起始索引 + 长度)转化为一个与输出形状一致的列索引张量,再通过 gather(dim=1, index=...) 沿列维度(dim=1)收集对应元素。

以下是完整实现:

import torch

def batch_row_slice(data: torch.Tensor, start_indices: torch.Tensor, length: int, dim: int = 1) -> torch.Tensor:
    """
    从2D张量每行中提取固定长度子序列(向量化实现)

    Args:
        data: 输入2D张量,shape = (N, D)
        start_indices: 每行起始列索引,shape = (N,),dtype=torch.long
        length: 每个切片的固定长度(所有行相同)
        dim: 沿哪个维度切片(默认为列维度,即 dim=1)

    Returns:
        切片结果,shape = (N, length)
    """
    if dim != 1:
        raise NotImplementedError("仅支持沿列维度(dim=1)切片")

    # 构造每行的索引序列:[[s0, s0+1], [s1, s1+1], ..., [s_{N-1}, s_{N-1}+1]]
    # 使用 broadcasting 和 arange 实现高效生成
    arange = torch.arange(length, device=data.device, dtype=start_indices.dtype)
    # start_indices: (N,) → (N, 1); arange: (length,) → (1, length)
    # 广播后得到 (N, length) 的起始偏移矩阵
    index_tensor = start_indices.unsqueeze(1) + arange.unsqueeze(0)

    # 使用 gather 沿 dim=1 收集对应列元素
    return torch.gather(data, dim=1, index=index_tensor)

# 示例用法
data = torch.tensor([[ 1.,  2.,  3.,  4.,  5.],
                     [ 6.,  7.,  8.,  9., 10.],
                     [11., 12., 13., 14., 15.]])

start_idx = torch.tensor([0, 3, 1], dtype=torch.long)  # 注意:必须为 long 类型!
result = batch_row_slice(data, start_idx, length=2)

print(result)
# 输出:
# tensor([[ 1.,  2.],
#         [ 9., 10.],
#         [12., 13.]])

关键要点说明:

  • start_indices 必须是 torch.long 类型(浮点型索引不被 gather 接受);
  • length 必须对所有行保持一致,这是向量化前提;
  • 索引越界行为由 gather 自动处理(超出范围时返回 0 填充),生产环境建议提前校验 start_idx + length
  • 方法天然支持 GPU 张量,全程无 CPU-GPU 数据拷贝,性能远超循环;
  • 可轻松扩展至更高维(如 (B, T, D) 中按 B 维切时间步),只需调整 dim 和索引广播逻辑。

通过此方法,你可以在保持代码简洁性的同时,充分发挥 PyTorch 的张量并行能力,适用于 Transformer attention mask 构建、时序窗口采样、动态 padding 等典型场景。


# python  # ai  # 深度学习  # pytorch 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 如何快速打造个性化非模板自助建站?  如何在自有机房高效搭建专业网站?  Laravel Admin后台管理框架推荐_Laravel快速开发后台工具  微信小程序 五星评分(包括半颗星评分)实例代码  Laravel如何自定义错误页面(404, 500)?(代码示例)  在线教育网站制作平台,山西立德教育官网?  DeepSeek是免费使用的吗 DeepSeek收费模式与Pro版本功能详解  Laravel怎么实现软删除SoftDeletes_Laravel模型回收站功能与数据恢复【步骤】  Android自定义控件实现温度旋转按钮效果  手机怎么制作网站教程步骤,手机怎么做自己的网页链接?  七夕网站制作视频,七夕大促活动怎么报名?  弹幕视频网站制作教程下载,弹幕视频网站是什么意思?  Windows10如何更改计算机工作组_Win10系统属性修改Workgroup  北京网站制作的公司有哪些,北京白云观官方网站?  如何快速搭建高效简练网站?  清除minerd进程的简单方法  Firefox Developer Edition开发者版本入口  如何在云主机上快速搭建多站点网站?  uc浏览器二维码扫描入口_uc浏览器扫码功能使用地址  Laravel如何实现多级无限分类_Laravel递归模型关联与树状数据输出【方法】  个人网站制作流程图片大全,个人网站如何注销?  教学论文网站制作软件有哪些,写论文用什么软件 ?  Laravel Artisan命令怎么自定义_创建自己的Laravel命令行工具完全指南  Android利用动画实现背景逐渐变暗  Laravel的HTTP客户端怎么用_Laravel HTTP Client发起API请求教程  Laravel中间件如何使用_Laravel自定义中间件实现权限控制  html5audio标签播放结束怎么触发事件_onended回调方法【教程】  Win11怎么恢复误删照片_Win11数据恢复工具使用【推荐】  北京网站制作公司哪家好一点,北京租房网站有哪些?  详解Huffman编码算法之Java实现  如何快速生成橙子建站落地页链接?  javascript中数组(Array)对象和字符串(String)对象的常用方法总结  Linux网络带宽限制_tc配置实践解析【教程】  如何快速生成高效建站系统源代码?  Laravel如何使用Gate和Policy进行授权?(权限控制)  php打包exe后无法访问网络共享_共享权限设置方法【教程】  网站视频制作书签怎么做,ie浏览器怎么将网站固定在书签工具栏?  Laravel storage目录权限问题_Laravel文件写入权限设置  Laravel Asset编译怎么配置_Laravel Vite前端构建工具使用  微信小程序 canvas开发实例及注意事项  百度输入法全感官ai怎么关 百度输入法全感官皮肤关闭  如何在万网ECS上快速搭建专属网站?  PHP 实现电台节目表的智能时间匹配与今日/明日轮播逻辑  网站制作软件有哪些,制图软件有哪些?  Laravel如何清理系统缓存命令_Laravel清除路由配置及视图缓存的方法【总结】  php读取心率传感器数据怎么弄_php获取max30100的心率值【指南】  千问怎样用提示词获取健康建议_千问健康类提示词注意事项【指南】  小米17系列还有一款新机?主打6.9英寸大直屏和旗舰级影像  Laravel项目结构怎么组织_大型Laravel应用的最佳目录结构实践  教你用AI将一段旋律扩展成一首完整的曲子