文章目录
最近刚刚开始从 Keras 换成 PyTorch,在使用过程中可能会遇到一些常见的问题,做一些整理。
可以在 python 文件头部使用如下函数打开 nan 检查:
Python
1 |
torch.autograd.set_detect_anomaly(True) |
如果遇到了 nan 的 Tensor,它会抛出异常。幸运的话它会告诉你 nan 产生的位置。比如说我遇到过:
PowerShell
1 |
RuntimeError: Function 'SmoothL1LossBackward' returned nan values in its 0th output. |
有些时候,往往会遇到比如 Adam 就没有 nan 而 SGD 就会出现 nan,这种通常都是 Loss 设得太大,可以调低学习率试试。
其他可能产生 nan 的地方可以尝试定位下:
1、脏数据,输入有 NaN
2、设置 clip gradient
3、更换初始化参数方法
如果是为了测试模型的前向运算运行时间,需要设置 model 为评估模式:
Python
1 |
model.eval() |
同时在 GPU 上测速时需要使用 torch.cuda.synchronize() 同步 CUDA 操作:
Python
1 2 3 4 5 |
torch.cuda.synchronize() start = time.time() result = model(input) torch.cuda.synchronize() end = time.time() |
在一些任务中,如果不是使用已有训练参数而是从 0 开始训练一个空白的网络,进行参数的初始化(例如 Conv2D)会有利于加快模型的收敛,例如下面参数初始化方式是(通常可以放在 model 的 init 函数结尾):
Python
1 2 3 4 5 6 7 8 9 10 11 12 |
# weight initialization for m in self.modules(): if isinstance(m, nn.Conv2d): nn.init.kaiming_normal_(m.weight, mode='fan_out') if m.bias is not None: nn.init.zeros_(m.bias) elif isinstance(m, (nn.BatchNorm2d, nn.GroupNorm)): nn.init.ones_(m.weight) nn.init.zeros_(m.bias) elif isinstance(m, nn.Linear): nn.init.normal_(m.weight, 0, 0.01) nn.init.zeros_(m.bias) |
工程实践中经常用 torchvision 预训练参数然后提取其中部分层进行修改。这里面可以有两种方式:
第一种,直接 copy 全部的代码,然后根据自身需要输出中间层:
例如对于 shufflenetv2 代码可以这样修改返回你需要的层(_forward_impl 是原始的,_forward_impl_with_layers 是修改的):
Python
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 |
def _forward_impl(self, x): # See note [TorchScript super()] x = self.conv1(x) x = self.maxpool(x) x = self.stage2(x) x = self.stage3(x) x = self.stage4(x) x = self.conv5(x) x = x.mean([2, 3]) # globalpool x = self.fc(x) return x
def _forward_impl_with_layers(self, x): # See note [TorchScript super()] layer1 = self.conv1(x) layer2 = self.maxpool(layer1) layer3 = self.stage2(layer2) layer4 = self.stage3(layer3) layer5 = self.stage4(layer4) x = self.conv5(layer5) x = x.mean([2, 3]) # globalpool x = self.fc(x) return layer1, layer2, layer3, layer4, layer5, x
def forward(self, x): return self._forward_impl_with_layers(x) |
另外一种方法不下载代码直接调用 torchvision 中的层,这个可能需要分析每个代码的实现才能知道想要的层,比如这样打印:
Python
1 2 3 |
import torchvision model = models.shufflenet_v2_x0_5(pretrained=True) print('model = {}'.format(model)) |
打印结果类似:
PowerShell
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 |
model = ShuffleNetV2( (conv1): Sequential( (0): Conv2d(3, 24, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), bias=False) (1): BatchNorm2d(24, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (2): ReLU(inplace=True) ) (maxpool): MaxPool2d(kernel_size=3, stride=2, padding=1, dilation=1, ceil_mode=False) (stage2): Sequential( (0): InvertedResidual( (branch1): Sequential( (0): Conv2d(24, 24, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), groups=24, bias=False) (1): BatchNorm2d(24, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (2): Conv2d(24, 24, kernel_size=(1, 1), stride=(1, 1), bias=False) (3): BatchNorm2d(24, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (4): ReLU(inplace=True) ) (branch2): Sequential( (0): Conv2d(24, 24, kernel_size=(1, 1), stride=(1, 1), bias=False) (1): BatchNorm2d(24, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (2): ReLU(inplace=True) (3): Conv2d(24, 24, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), groups=24, bias=False) (4): BatchNorm2d(24, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (5): Conv2d(24, 24, kernel_size=(1, 1), stride=(1, 1), bias=False) (6): BatchNorm2d(24, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (7): ReLU(inplace=True) ) ) (1): InvertedResidual( (branch1): Sequential() (branch2): Sequential( (0): Conv2d(24, 24, kernel_size=(1, 1), stride=(1, 1), bias=False) (1): BatchNorm2d(24, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (2): ReLU(inplace=True) (3): Conv2d(24, 24, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), groups=24, bias=False) (4): BatchNorm2d(24, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (5): Conv2d(24, 24, kernel_size=(1, 1), stride=(1, 1), bias=False) (6): BatchNorm2d(24, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (7): ReLU(inplace=True) ) ) (2): InvertedResidual( (branch1): Sequential() (branch2): Sequential( (0): Conv2d(24, 24, kernel_size=(1, 1), stride=(1, 1), bias=False) (1): BatchNorm2d(24, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (2): ReLU(inplace=True) (3): Conv2d(24, 24, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), groups=24, bias=False) (4): BatchNorm2d(24, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (5): Conv2d(24, 24, kernel_size=(1, 1), stride=(1, 1), bias=False) (6): BatchNorm2d(24, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (7): ReLU(inplace=True) ) ) (3): InvertedResidual( (branch1): Sequential() (branch2): Sequential( (0): Conv2d(24, 24, kernel_size=(1, 1), stride=(1, 1), bias=False) (1): BatchNorm2d(24, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (2): ReLU(inplace=True) (3): Conv2d(24, 24, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), groups=24, bias=False) (4): BatchNorm2d(24, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (5): Conv2d(24, 24, kernel_size=(1, 1), stride=(1, 1), bias=False) (6): BatchNorm2d(24, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (7): ReLU(inplace=True) ) ) ) (stage3): Sequential( (0): InvertedResidual( (branch1): Sequential( (0): Conv2d(48, 48, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), groups=48, bias=False) (1): BatchNorm2d(48, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (2): Conv2d(48, 48, kernel_size=(1, 1), stride=(1, 1), bias=False) (3): BatchNorm2d(48, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (4): ReLU(inplace=True) ) (branch2): Sequential( (0): Conv2d(48, 48, kernel_size=(1, 1), stride=(1, 1), bias=False) (1): BatchNorm2d(48, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (2): ReLU(inplace=True) (3): Conv2d(48, 48, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), groups=48, bias=False) (4): BatchNorm2d(48, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (5): Conv2d(48, 48, kernel_size=(1, 1), stride=(1, 1), bias=False) (6): BatchNorm2d(48, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (7): ReLU(inplace=True) ) ) (1): InvertedResidual( (branch1): Sequential() (branch2): Sequential( (0): Conv2d(48, 48, kernel_size=(1, 1), stride=(1, 1), bias=False) (1): BatchNorm2d(48, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (2): ReLU(inplace=True) (3): Conv2d(48, 48, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), groups=48, bias=False) (4): BatchNorm2d(48, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (5): Conv2d(48, 48, kernel_size=(1, 1), stride=(1, 1), bias=False) (6): BatchNorm2d(48, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (7): ReLU(inplace=True) ) ) (2): InvertedResidual( (branch1): Sequential() (branch2): Sequential( (0): Conv2d(48, 48, kernel_size=(1, 1), stride=(1, 1), bias=False) (1): BatchNorm2d(48, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (2): ReLU(inplace=True) (3): Conv2d(48, 48, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), groups=48, bias=False) (4): BatchNorm2d(48, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (5): Conv2d(48, 48, kernel_size=(1, 1), stride=(1, 1), bias=False) (6): BatchNorm2d(48, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (7): ReLU(inplace=True) ) ) (3): InvertedResidual( (branch1): Sequential() (branch2): Sequential( (0): Conv2d(48, 48, kernel_size=(1, 1), stride=(1, 1), bias=False) (1): BatchNorm2d(48, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (2): ReLU(inplace=True) (3): Conv2d(48, 48, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), groups=48, bias=False) (4): BatchNorm2d(48, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (5): Conv2d(48, 48, kernel_size=(1, 1), stride=(1, 1), bias=False) (6): BatchNorm2d(48, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (7): ReLU(inplace=True) ) ) (4): InvertedResidual( (branch1): Sequential() (branch2): Sequential( (0): Conv2d(48, 48, kernel_size=(1, 1), stride=(1, 1), bias=False) (1): BatchNorm2d(48, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (2): ReLU(inplace=True) (3): Conv2d(48, 48, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), groups=48, bias=False) (4): BatchNorm2d(48, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (5): Conv2d(48, 48, kernel_size=(1, 1), stride=(1, 1), bias=False) (6): BatchNorm2d(48, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (7): ReLU(inplace=True) ) ) (5): InvertedResidual( (branch1): Sequential() (branch2): Sequential( (0): Conv2d(48, 48, kernel_size=(1, 1), stride=(1, 1), bias=False) (1): BatchNorm2d(48, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (2): ReLU(inplace=True) (3): Conv2d(48, 48, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), groups=48, bias=False) (4): BatchNorm2d(48, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (5): Conv2d(48, 48, kernel_size=(1, 1), stride=(1, 1), bias=False) (6): BatchNorm2d(48, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (7): ReLU(inplace=True) ) ) (6): InvertedResidual( (branch1): Sequential() (branch2): Sequential( (0): Conv2d(48, 48, kernel_size=(1, 1), stride=(1, 1), bias=False) (1): BatchNorm2d(48, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (2): ReLU(inplace=True) (3): Conv2d(48, 48, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), groups=48, bias=False) (4): BatchNorm2d(48, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (5): Conv2d(48, 48, kernel_size=(1, 1), stride=(1, 1), bias=False) (6): BatchNorm2d(48, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (7): ReLU(inplace=True) ) ) (7): InvertedResidual( (branch1): Sequential() (branch2): Sequential( (0): Conv2d(48, 48, kernel_size=(1, 1), stride=(1, 1), bias=False) (1): BatchNorm2d(48, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (2): ReLU(inplace=True) (3): Conv2d(48, 48, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), groups=48, bias=False) (4): BatchNorm2d(48, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (5): Conv2d(48, 48, kernel_size=(1, 1), stride=(1, 1), bias=False) (6): BatchNorm2d(48, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (7): ReLU(inplace=True) ) ) ) (stage4): Sequential( (0): InvertedResidual( (branch1): Sequential( (0): Conv2d(96, 96, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), groups=96, bias=False) (1): BatchNorm2d(96, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (2): Conv2d(96, 96, kernel_size=(1, 1), stride=(1, 1), bias=False) (3): BatchNorm2d(96, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (4): ReLU(inplace=True) ) (branch2): Sequential( (0): Conv2d(96, 96, kernel_size=(1, 1), stride=(1, 1), bias=False) (1): BatchNorm2d(96, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (2): ReLU(inplace=True) (3): Conv2d(96, 96, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), groups=96, bias=False) (4): BatchNorm2d(96, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (5): Conv2d(96, 96, kernel_size=(1, 1), stride=(1, 1), bias=False) (6): BatchNorm2d(96, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (7): ReLU(inplace=True) ) ) (1): InvertedResidual( (branch1): Sequential() (branch2): Sequential( (0): Conv2d(96, 96, kernel_size=(1, 1), stride=(1, 1), bias=False) (1): BatchNorm2d(96, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (2): ReLU(inplace=True) (3): Conv2d(96, 96, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), groups=96, bias=False) (4): BatchNorm2d(96, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (5): Conv2d(96, 96, kernel_size=(1, 1), stride=(1, 1), bias=False) (6): BatchNorm2d(96, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (7): ReLU(inplace=True) ) ) (2): InvertedResidual( (branch1): Sequential() (branch2): Sequential( (0): Conv2d(96, 96, kernel_size=(1, 1), stride=(1, 1), bias=False) (1): BatchNorm2d(96, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (2): ReLU(inplace=True) (3): Conv2d(96, 96, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), groups=96, bias=False) (4): BatchNorm2d(96, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (5): Conv2d(96, 96, kernel_size=(1, 1), stride=(1, 1), bias=False) (6): BatchNorm2d(96, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (7): ReLU(inplace=True) ) ) (3): InvertedResidual( (branch1): Sequential() (branch2): Sequential( (0): Conv2d(96, 96, kernel_size=(1, 1), stride=(1, 1), bias=False) (1): BatchNorm2d(96, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (2): ReLU(inplace=True) (3): Conv2d(96, 96, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), groups=96, bias=False) (4): BatchNorm2d(96, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (5): Conv2d(96, 96, kernel_size=(1, 1), stride=(1, 1), bias=False) (6): BatchNorm2d(96, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (7): ReLU(inplace=True) ) ) ) (conv5): Sequential( (0): Conv2d(192, 1024, kernel_size=(1, 1), stride=(1, 1), bias=False) (1): BatchNorm2d(1024, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True) (2): ReLU(inplace=True) ) (fc): Linear(in_features=1024, out_features=1000, bias=True) ) |
比如获得 conv1 层输出就是
Python
1 |
model.conv1 |
有时在你安装某一个版本的 PyTorch (比如 1.5.0) 时会出现如下错误提示:
Shell
1 2 3 4 5 6 |
The NVIDIA driver on your system is too old (found version 10000). Please update your GPU driver by downloading and installing a new version from the URL: http://www.nvidia.com/Download/index.aspx Alternatively, go to: https://pytorch.org to install a PyTorch version that has been compiled with your version of the CUDA driver. |
在安装 PyTorch 的时候往往会指定相应的 CUDA 版本,这个错误的意思可能是你没有安装特定版本的 CUDA 或者你的 CUDA 版本与你的 GPU Driver 版本不匹配。
在 Nvidia 官网中给了我们如下的版本匹配:https://docs.nvidia.com/cuda/cuda-toolkit-release-notes/index.html
如果需要升级,你可以使用如下方式升级:
1)增加软件源:
1 |
sudo add-apt-repository ppa:graphics-drivers/ppa && sudo apt update |
2)查看可以使用的版本:
1 |
ubuntu-drivers devices |
例如我这里查询结果是:
3)升级指定版本(根据上面表格找到合适的版本升级):
1 |
sudo apt install nvidia-VERSION_NUMBER_HERE |
如果出现某些冲突问题可以尝试先卸载再安装:
1 |
sudo apt --purge autoremove nvidia* |
PS:另外一种方式你也可以先不升级指定版本,先使用如下命令查看本地 CUDA 版本:
1 |
nvcc --version |
比如我这里显示的就是:
那么我就应该安装支持 CUDA 10.0 的版本。可能 PyTorch 1.5 就不可用了,但是 PyTorch 1.4 还是可以的,可以使用如下命令安装:
1 |
pip install torch==1.4.0+cu100 torchvision==0.5.0+cu100 -f https://download.pytorch.org/whl/torch_stable.html |
具体什么版本支持可以参考:
https://download.pytorch.org/whl/torch_stable.html
这个页面。
PS:其他常用命令:
查看 GPU 型号:
1 |
lspci | grep -i nvidia |
查看驱动版本:
1 |
cat /proc/driver/nvidia/version |
查看 PyTorch 所用 CUDA 版本,在 PyTorch 环境中运行如下脚本:
Python
1 2 3 4 |
import torch print('torch.__version__ = {}'.format(torch.__version__)) print('torch.version.cuda = {}'.format(torch.version.cuda)) print('torch.cuda.is_available() = {}'.format(torch.cuda.is_available())) |
如果在训练时遇到如下错误:
Shell
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
File "/home/liuxiao/.local/lib/python3.7/site-packages/torch/nn/modules/contai ner.py", line 100, in forward input = module(input) File "/home/liuxiao/.local/lib/python3.7/site-packages/torch/nn/modules/module .py", line 532, in __call__ result = self.forward(*input, **kwargs) File "/home/liuxiao/.local/lib/python3.7/site-packages/torch/nn/modules/batchn orm.py", line 107, in forward exponential_average_factor, self.eps) File "/home/liuxiao/.local/lib/python3.7/site-packages/torch/nn/functional.py" , line 1666, in batch_norm raise ValueError('Expected more than 1 value per channel when training, got input size {}'.format(size)) ValueError: Expected more than 1 value per channel when training, got input size torch.Size([1, 32, 1]) |
一个可能的原因是出现了输入 batch_size = 1 的情况,这时可以考虑在 DataLoader 属性加上 drop_last=True 解决,它会抛弃掉不够一个 batch size 的情况。例如:
Python
1 2 |
train_loader = torch.utils.data.DataLoader(dataset=train_set, shuffle=False, batch_size=opt.batch_size, drop_last=True) |
如果实在无法避免或者就需要 batch_size = 1 的训练方式,还可以考虑把网络中的 BatchNorm 换成 InstanceNorm。
如果获取变量值时,遇到下面错误:
Shell
1 |
RuntimeError: Can't call numpy() on Variable that requires grad. Use var.detach().numpy() instead. |
这里面通常有两种情况:
一种是这个变量是含有训练参数的,需要反向传播,则使用 var.detach().numpy() 获取。
另一种如果这个变量是不进行训练的不需要反向传播,则将相关的代码用如下方式(with torch.no_grad())修饰即可:
Python
1 2 |
with torch.no_grad(): your code here |
如果在运行多线程训练时出现类似如下错误:
1 2 3 |
RuntimeError: error executing torch_shm_manager at "/hdd/kps_pipeline/venv/lib/python3.6/site-packages/torch/bin/torch_shm_manager" at /pytorch/torch/lib/libshm/core.cpp:99 torch_shm_manager: error while loading shared libraries: libcudart.so.10.0: cannot open shared object file: No such file or directory torch_shm_manager: error while loading shared libraries: libcudart.so.10.0: cannot open shared object file: No such file or directory |
可能的解决方法是注释掉如下设置(如果有的话):
Python
1 |
# torch.multiprocessing.set_sharing_strategy('file_system') |
如果在运行多线程训练时出现类似如下错误:
1 2 3 4 |
Cowardly refusing to serialize non-leaf tensor which requires_grad, since autograd does not support crossing process boundaries. If you just want to transfer the data, call detach() on the tensor before serializing (e.g., putting it on the queue). |
我们看下相关报错的函数是这样的:
Python
1 2 3 4 5 6 7 8 9 10 11 |
def reduce_tensor(tensor): storage = tensor.storage()
if tensor.requires_grad and not tensor.is_leaf: raise RuntimeError("Cowardly refusing to serialize non-leaf tensor which requires_grad, " "since autograd does not support crossing process boundaries. " "If you just want to transfer the data, call detach() on the tensor " "before serializing (e.g., putting it on the queue).")
check_serializing_named_tensor(tensor) torch.utils.hooks.warn_if_has_hooks(tensor) |
经过分析我这里的发生的原因是在多线程 DataLoader 中使用了一个模型生成数据,然而这个模型的参数有一部分却是 requires_grad = True 属性的。
可以采用如下方式处理模型让生成的 Tensor 都为 no_grad:
Python
1 2 3 4 5 |
# No need to backward use eval() # Use to fix RuntimeError: Cowardly refusing to serialize non-leaf tensor which requires_grad for param in self.superpoint.parameters(): param.requires_grad = False self.superpoint.eval() |
由于 numpy 中的 random 不是 thread safe 的,因此在多线程中,其不同线程的 random 无法生成不同的随机数,需要每个线程重新设置 random.seed 才可以。因此对于 DataLoader 在 num_workers > 0 时就可能产生问题(比如需要每次生成不同的随机数据)。对于此问题有几种修改方式:
第一种
利用 worker_init_fn 每个线程重新设置种子,示例代码如下:
Python
1 |
ds = DataLoader(ds, 10, shuffle=False, num_workers=4, worker_init_fn=lambda _: np.random.seed()) |
第二种
在文件开头加上下面两行设置:
Python
1 2 |
import torch.multiprocessing as mp mp.set_start_method('spawn') |
如果只是 Debug 而不是 Run 的时候出现,此类问题是由于在 PyCharm 中开启了调试子线程的功能,在 File->Settings->Building, Execution, Deployment->Python Debugger 中,将 Attach to subprocess automatically while debugging关闭即可。如图所示:
如果在运行 PyTorch 时出现这一次错误,一个可能的原因是你的显卡已经不被高版本的 PyTorch 所支持。
比如在最近的更新中 PyTorch 1.3.1 及以后版本的显卡支持已经升级为 Compute Capability >= 3.7,完整的各种设备支持的 Compute Capability 列表如下:
https://developer.nvidia.com/cuda-gpus
GPU | Compute Capability |
---|---|
NVIDIA TITAN RTX | 7.5 |
Geforce RTX 2080 Ti | 7.5 |
Geforce RTX 2080 | 7.5 |
Geforce RTX 2070 | 7.5 |
Geforce RTX 2060 | 7.5 |
NVIDIA TITAN V | 7 |
NVIDIA TITAN Xp | 6.1 |
NVIDIA TITAN X | 6.1 |
GeForce GTX 1080 Ti | 6.1 |
GeForce GTX 1080 | 6.1 |
GeForce GTX 1070 | 6.1 |
GeForce GTX 1060 | 6.1 |
GeForce GTX 1050 | 6.1 |
GeForce GTX TITAN X | 5.2 |
GeForce GTX TITAN Z | 3.5 |
GeForce GTX TITAN Black | 3.5 |
GeForce GTX TITAN | 3.5 |
GeForce GTX 980 Ti | 5.2 |
GeForce GTX 980 | 5.2 |
GeForce GTX 970 | 5.2 |
GeForce GTX 960 | 5.2 |
GeForce GTX 950 | 5.2 |
GeForce GTX 780 Ti | 3.5 |
GeForce GTX 780 | 3.5 |
GeForce GTX 770 | 3 |
GeForce GTX 760 | 3 |
GeForce GTX 750 Ti | 5 |
GeForce GTX 750 | 5 |
GeForce GTX 690 | 3 |
GeForce GTX 680 | 3 |
GeForce GTX 670 | 3 |
GeForce GTX 660 Ti | 3 |
GeForce GTX 660 | 3 |
GeForce GTX 650 Ti BOOST | 3 |
GeForce GTX 650 Ti | 3 |
GeForce GTX 650 | 3 |
GeForce GTX 560 Ti | 2.1 |
GeForce GTX 550 Ti | 2.1 |
GeForce GTX 460 | 2.1 |
GeForce GTS 450 | 2.1 |
GeForce GTS 450* | 2.1 |
GeForce GTX 590 | 2 |
GeForce GTX 580 | 2 |
GeForce GTX 570 | 2 |
GeForce GTX 480 | 2 |
GeForce GTX 470 | 2 |
GeForce GTX 465 | 2 |
GeForce GT 740 | 3 |
GeForce GT 730 | 3.5 |
GeForce GT 730 DDR3,128bit | 2.1 |
GeForce GT 720 | 3.5 |
GeForce GT 705* | 3.5 |
GeForce GT 640 (GDDR5) | 3.5 |
GeForce GT 640 (GDDR3) | 2.1 |
GeForce GT 630 | 2.1 |
GeForce GT 620 | 2.1 |
GeForce GT 610 | 2.1 |
GeForce GT 520 | 2.1 |
GeForce GT 440 | 2.1 |
GeForce GT 440* | 2.1 |
GeForce GT 430 | 2.1 |
GeForce GT 430* | 2.1 |
解决方法有两种:
1)最简单的解决方法是降级成早期版本,比如 Pytorch 1.2:
Shell
1 |
conda install pytorch==1.2.0 torchvision==0.4.0 cudatoolkit=10.0 -c pytorch |
参见:
https://pytorch.org/get-started/previous-versions/
2)如果一定要使用新版,则需要使用从 Source Build 的方式安装:
https://github.com/pytorch/pytorch#from-source
[1] https://blog.csdn.net/weixin_41278720/article/details/80778640
[2] https://discuss.pytorch.org/t/getting-nan-after-first-iteration-with-custom-loss/25929/14
[3] https://zllrunning.github.io/2018/03/24/20180324/
[4] https://github.com/MVIG-SJTU/AlphaPose/issues/402
[5] https://github.com/pytorch/pytorch/issues/5059
[6] https://blog.csdn.net/Nin7a/article/details/104138036
[7] https://blog.csdn.net/sinat_33425327/article/details/84823272