当前位置:   article > 正文

模型转换 PyTorch转ONNX 入门

pytorch转onnx

前言

本文主要介绍如何将PyTorch模型转换为ONNX模型,为后面的模型部署做准备。转换后的xxx.onnx模型,进行加载和测试。最后介绍使用Netron,可视化ONNX模型,看一下网络结构;查看使用了那些算子,以便开发部署。

目录

前言

一、PyTorch模型转ONNX模型

1.1 转换为ONNX模型且加载权重

1.2 转换为ONNX模型但不加载权重

1.3 torch.onnx.export() 函数

二、加载ONNX模型

三、可视化ONNX模型


一、PyTorch模型转ONNX模型

将PyTorch模型转换为ONNX模型,通常是使用torch.onnx.export( )函数来转换的,基本的思路是:

  1. 加载PyTorch模型,可以选择只加载模型结构;也可以选择加载模型结构和权重。
  2. 然后定义PyTorch模型的输入维度,比如(1, 3, 224, 224),这是一个三通道的彩色图,分辨率为224x224。
  3. 最后使用torch.onnx.export( )函数来转换,生产xxx.onnx模型。

下面有一个简单的例子:

  1. import torch
  2. import torch.onnx
  3. # 加载 PyTorch 模型
  4. model = ...
  5. # 设置模型输入,包括:通道数,分辨率等
  6. dummy_input = torch.randn(1, 3, 224, 224, device='cpu')
  7. # 转换为ONNX模型
  8. torch.onnx.export(model, dummy_input, "model.onnx", export_params=True)

1.1 转换为ONNX模型且加载权重

这里举一个resnet18的例子,基本思路是:

  1. 首先加载了一个预训练的 ResNet18 模型;
  2. 然后将其设置为评估模式。接下来定义一个与模型输入张量形状相同的输入张量,并使用 torch.randn() 函数生成了一个随机张量。
  3. 最后,使用 onnx.export() 函数将 PyTorch 模型转换为 ONNX 格式,并将其保存到指定的输出文件中。

程序如下:

  1. import torch
  2. import torchvision.models as models
  3. # 加载预训练的 ResNet18 模型
  4. model = models.resnet18(pretrained=True)
  5. # 将模型设置为评估模式
  6. model.eval()
  7. # 定义输入张量,需要与模型的输入张量形状相同
  8. input_shape = (1, 3, 224, 224)
  9. x = torch.randn(input_shape)
  10. # 需要指定输入张量,输出文件路径和运行设备
  11. # 默认情况下,输出张量的名称将基于模型中的名称自动分配
  12. device = torch.device("cuda" if torch.cuda.is_available() else "cpu")
  13. # 将 PyTorch 模型转换为 ONNX 格式
  14. output_file = "resnet18.onnx"
  15. torch.onnx.export(model, x.to(device), output_file, export_params=True)

1.2 转换为ONNX模型但不加载权重

举一个resnet18的例子:基本思路是:

  1. 首先加载了一个预训练的 ResNet18 模型;
  2. 然后使用 onnx.export() 函数将 PyTorch 模型转换为 ONNX 格式;指定参数do_constant_folding=False,不加载模型的权重。
  1. import torch
  2. import torchvision.models as models
  3. # 加载 PyTorch 模型
  4. model = models.resnet18()
  5. # 将模型转换为 ONNX 格式但不加载权重
  6. dummy_input = torch.randn(1, 3, 224, 224)
  7. torch.onnx.export(model, dummy_input, "resnet18.onnx", do_constant_folding=False)

下面构建一个简单网络结构,并转换为ONNX

  1. import torch
  2. import torchvision
  3. import numpy as np
  4. # 定义一个简单的PyTorch 模型
  5. class MyModel(torch.nn.Module):
  6. def __init__(self):
  7. super(MyModel, self).__init__()
  8. self.conv1 = torch.nn.Conv2d(3, 32, kernel_size=3, stride=1, padding=1)
  9. self.relu = torch.nn.ReLU()
  10. self.maxpool = torch.nn.MaxPool2d(kernel_size=2, stride=2)
  11. self.conv2 = torch.nn.Conv2d(32, 64, kernel_size=3, stride=1, padding=1)
  12. self.flatten = torch.nn.Flatten()
  13. self.fc1 = torch.nn.Linear(64 * 8 * 8, 10)
  14. def forward(self, x):
  15. x = self.conv1(x)
  16. x = self.relu(x)
  17. x = self.maxpool(x)
  18. x = self.conv2(x)
  19. x = self.relu(x)
  20. x = self.maxpool(x)
  21. x = self.flatten(x)
  22. x = self.fc1(x)
  23. return x
  24. # 创建模型实例
  25. model = MyModel()
  26. # 指定模型输入尺寸
  27. dummy_input = torch.randn(1, 3, 32, 32)
  28. # 将PyTorch模型转为ONNX模型
  29. torch.onnx.export(model, dummy_input, 'mymodel.onnx', do_constant_folding=False)

1.3 torch.onnx.export() 函数

看一下这个函数的参数

  1. torch.onnx.export(
  2. model,
  3. args,
  4. f,
  5. export_params=True,
  6. opset_version=10,
  7. do_constant_folding=True,
  8. input_names=['input'],
  9. output_names=['output'],
  10. dynamic_axes=None,
  11. verbose=False,
  12. example_outputs=None,
  13. keep_initializers_as_inputs=None)
  • model:需要导出的 PyTorch 模型
  • args:PyTorch模型输入数据的尺寸,指定通道数、长和宽。可以是单个 Tensor 或元组,也可以是元组列表。
  • f:导出的 ONNX 文件路径和名称,mymodel.onnx。
  • export_params:是否导出模型参数。如果设置为 False,则不导出模型参数。
  • opset_version:导出的 ONNX 版本。默认值为 10。
  • do_constant_folding:是否对模型进行常量折叠。如果设置为 True,不加载模型的权重。
  • input_names:模型输入数据的名称。默认为 'input'。
  • output_names:模型输出数据的名称。默认为 'output'。
  • dynamic_axes:动态轴的列表,允许在导出的 ONNX 模型中创建变化的维度。
  • verbose:是否输出详细的导出信息。
  • example_outputs:用于确定导出 ONNX 模型输出形状的样本输出。
  • keep_initializers_as_inputs:是否将模型的初始化器作为输入导出。如果设置为 True,则模型初始化器将被作为输入的一部分导出。

下面是只是一个常用的模板

  1. import torch.onnx
  2. # 转为ONNX
  3. def Convert_ONNX(model):
  4. # 设置模型为推理模式
  5. model.eval()
  6. # 设置模型输入的尺寸
  7. dummy_input = torch.randn(1, input_size, requires_grad=True)
  8. # 导出ONNX模型
  9. torch.onnx.export(model, # model being run
  10. dummy_input, # model input (or a tuple for multiple inputs)
  11. "xxx.onnx", # where to save the model
  12. export_params=True, # store the trained parameter weights inside the model file
  13. opset_version=10, # the ONNX version to export the model to
  14. do_constant_folding=True, # whether to execute constant folding for optimization
  15. input_names = ['modelInput'], # the model's input names
  16. output_names = ['modelOutput'], # the model's output names
  17. dynamic_axes={'modelInput' : {0 : 'batch_size'}, # variable length axes
  18. 'modelOutput' : {0 : 'batch_size'}})
  19. print(" ")
  20. print('Model has been converted to ONNX')
  21. if __name__ == "__main__":
  22. # 构建模型并训练
  23. # xxxxxxxxxxxx
  24. # 测试模型精度
  25. #testAccuracy()
  26. # 加载模型结构与权重
  27. model = Network()
  28. path = "myFirstModel.pth"
  29. model.load_state_dict(torch.load(path))
  30. # 转换为ONNX
  31. Convert_ONNX(model)

二、加载ONNX模型

加载ONNX模型,通常需要用到ONNX、ONNX Runtime,所以需要先安装。

pip install onnx
pip install onnxruntime

加载ONNX模型可以使用ONNX Runtime库,以下是一个加载ONNX模型的示例代码:

  1. import onnxruntime as ort
  2. # 加载 ONNX 模型
  3. ort_session = ort.InferenceSession("model.onnx")
  4. # 准备输入信息
  5. input_info = ort_session.get_inputs()[0]
  6. input_name = input_info.name
  7. input_shape = input_info.shape
  8. input_type = input_info.type
  9. # 运行ONNX模型
  10. outputs = ort_session.run(input_name, input_data)
  11. # 获取输出信息
  12. output_info = ort_session.get_outputs()[0]
  13. output_name = output_info.name
  14. output_shape = output_info.shape
  15. output_data = outputs[0]
  16. print("outputs:", outputs)
  17. print("output_info :", output_info )
  18. print("output_name :", output_name )
  19. print("output_shape :", output_shape )
  20. print("output_data :", output_data )

以下是一个示例程序,将 resnet18 模型从 PyTorch 转换为 ONNX 格式,然后加载和测试 ONNX 模型的过程:

  1. import torch
  2. import torchvision.models as models
  3. import onnx
  4. import onnxruntime
  5. # 加载 PyTorch 模型
  6. model = models.resnet18(pretrained=True)
  7. model.eval()
  8. # 定义输入和输出张量的名称和形状
  9. input_names = ["input"]
  10. output_names = ["output"]
  11. batch_size = 1
  12. input_shape = (batch_size, 3, 224, 224)
  13. output_shape = (batch_size, 1000)
  14. # 将 PyTorch 模型转换为 ONNX 格式
  15. torch.onnx.export(
  16. model, # 要转换的 PyTorch 模型
  17. torch.randn(input_shape), # 模型输入的随机张量
  18. "resnet18.onnx", # 保存的 ONNX 模型的文件名
  19. input_names=input_names, # 输入张量的名称
  20. output_names=output_names, # 输出张量的名称
  21. dynamic_axes={input_names[0]: {0: "batch_size"}, output_names[0]: {0: "batch_size"}} # 动态轴,即输入和输出张量可以具有不同的批次大小
  22. )
  23. # 加载 ONNX 模型
  24. onnx_model = onnx.load("resnet18.onnx")
  25. onnx_model_graph = onnx_model.graph
  26. onnx_session = onnxruntime.InferenceSession(onnx_model.SerializeToString())
  27. # 使用随机张量测试 ONNX 模型
  28. x = torch.randn(input_shape).numpy()
  29. onnx_output = onnx_session.run(output_names, {input_names[0]: x})[0]
  30. print(f"PyTorch output: {model(torch.from_numpy(x)).detach().numpy()[0, :5]}")
  31. print(f"ONNX output: {onnx_output[0, :5]}")

上述代码中,首先加载预训练的 resnet18 模型,并定义了输入和输出张量的名称和形状。

然后,使用 torch.onnx.export() 函数将模型转换为 ONNX 格式,并保存为 resnet18.onnx 文件。

接着,使用 onnxruntime.InferenceSession() 函数加载 ONNX 模型,并使用随机张量进行测试。

最后,将 PyTorch 模型和 ONNX 模型的输出进行比较,以确保它们具有相似的输出。

三、可视化ONNX模型

使用Netron,可视化ONNX模型,看一下网络结构;查看使用了那些算子,以便开发部署。

这里简单介绍一下

Netron是一个轻量级、跨平台的模型可视化工具,支持多种深度学习框架的模型可视化,包括TensorFlow、PyTorch、ONNX、Keras、Caffe等等。它提供了可视化网络结构、层次关系、输出尺寸、权重等信息,并且可以通过鼠标移动和缩放来浏览模型。Netron还支持模型的导出和导入,方便模型的分享和交流。

 Netron的网页在线版本,直接在网页中打开和查看ONNX模型Netron

开源地址:GitHub - lutzroeder/netron: Visualizer for neural network, deep learning, and machine learning models

支持多种操作系统:

macOSDownload 

LinuxDownload 

WindowsDownload 

BrowserStart 

Python Server: Run pip install netron and netron [FILE] or netron.start('[FILE]').

下面是可视化模型截图:

还能查看某个节点(运算操作)的信息,比如下面MaxPool,点击一下,能看到使用的3x3的池化核,是否有填充pads,步长strides等参数。

分享完毕~

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/繁依Fanyi0/article/detail/367230
推荐阅读
相关标签
  

闽ICP备14008679号