pytorch中的detach和detach_


pytorch 的 Variable 對象中有兩個方法,detach和 detach_ :

detach

官方文檔中,對這個方法是這么介紹的。

  • 返回一個新的從當前圖中分離的 Variable。
  • 返回的 Variable 永遠不會需要梯度
  • 如果 被 detach 的Variable volatile=True, 那么 detach 出來的 volatile 也為 True
  • 還有一個注意事項,即:返回的 Variable 和 被 detach 的Variable 指向同一個 tensor
 1 import torch  2 from torch.nn import init  3 from torch.autograd import Variable  4 t1 = torch.FloatTensor([1., 2.])  5 v1 = Variable(t1)  6 t2 = torch.FloatTensor([2., 3.])  7 v2 = Variable(t2)  8 v3 = v1 + v2  9 v3_detached = v3.detach() 10 v3_detached.data.add_(t1) # 修改了 v3_detached Variable中 tensor 的值
11 print(v3, v3_detached)    # v3 中tensor 的值也會改變

detach的源碼:

1 # detach 的源碼
2 def detach(self): 3     result = NoGrad()(self)  # this is needed, because it merges version counters
4     result._grad_fn = None 5     return result

detach_

官網給的解釋是:將 Variable 從創建它的 graph 中分離,把它作為葉子節點。

從源碼中也可以看出這一點

  • 將 Variable 的grad_fn 設置為 None,這樣,BP 的時候,到這個 Variable 就找不到 它的 grad_fn,所以就不會再往后BP了。
  • 將 requires_grad 設置為 False。這個感覺大可不必,但是既然源碼中這么寫了,如果有需要梯度的話可以再手動 將 requires_grad 設置為 true
1 # detach_ 的源碼
2 def detach_(self): 3     """Detaches the Variable from the graph that created it, making it a 4  leaf. 5     """
6     self._grad_fn = None 7     self.requires_grad = False

能用來干啥

可以對部分網絡求梯度。

如果我們有兩個網絡 , 兩個關系是這樣的  現在我們想用 來為B網絡的參數來求梯度,但是又不想求A網絡參數的梯度。我們可以這樣:

# y=A(x), z=B(y) 求B中參數的梯度,不求A中參數的梯度
# 第一種方法
y = A(x)
z = B(y.detach())
z.backward()
 
# 第二種方法
y = A(x)
y.detach_()
z = B(y)
z.backward()

在這種情況下,detach 和 detach_ 都可以用。但是如果 你也想用 y來對 A進行 BP 呢?那就只能用第一種方法了。因為第二種方法已經將 A 模型的輸出給 detach(分離)了。

 

轉自: https://blog.csdn.net/u012436149/article/details/76714349


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM