container.py 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616
  1. # Copyright (c) 2021 PaddlePaddle Authors. All Rights Reserved.
  2. #
  3. # Licensed under the Apache License, Version 2.0 (the "License");
  4. # you may not use this file except in compliance with the License.
  5. # You may obtain a copy of the License at
  6. #
  7. # http://www.apache.org/licenses/LICENSE-2.0
  8. #
  9. # Unless required by applicable law or agreed to in writing, software
  10. # distributed under the License is distributed on an "AS IS" BASIS,
  11. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. # See the License for the specific language governing permissions and
  13. # limitations under the License.
  14. from collections import OrderedDict
  15. from collections.abc import Iterable, Mapping
  16. from ...base.dygraph.base import param_guard
  17. from ...base.framework import Parameter
  18. from .layers import Layer
  19. __all__ = []
  20. class LayerDict(Layer):
  21. """
  22. LayerDict holds sublayers in the ordered dictionary, and sublayers it contains are properly registered.
  23. Held sublayers can be accessed like a regular ordered python dictionary.
  24. Parameters:
  25. sublayers (LayerDict|OrderedDict|list[(key,Layer)...], optional): iterable of key/value pairs, the type of value is 'paddle.nn.Layer' .
  26. Examples:
  27. .. code-block:: python
  28. >>> import paddle
  29. >>> import numpy as np
  30. >>> from collections import OrderedDict
  31. >>> sublayers = OrderedDict([
  32. ... ('conv1d', paddle.nn.Conv1D(3, 2, 3)),
  33. ... ('conv2d', paddle.nn.Conv2D(3, 2, 3)),
  34. ... ('conv3d', paddle.nn.Conv3D(4, 6, (3, 3, 3))),
  35. >>> ])
  36. >>> layers_dict = paddle.nn.LayerDict(sublayers=sublayers)
  37. >>> l = layers_dict['conv1d']
  38. >>> for k in layers_dict:
  39. ... l = layers_dict[k]
  40. ...
  41. >>> print(len(layers_dict))
  42. 3
  43. >>> del layers_dict['conv2d']
  44. >>> print(len(layers_dict))
  45. 2
  46. >>> conv1d = layers_dict.pop('conv1d')
  47. >>> print(len(layers_dict))
  48. 1
  49. >>> layers_dict.clear()
  50. >>> print(len(layers_dict))
  51. 0
  52. """
  53. def __init__(self, sublayers=None):
  54. super().__init__()
  55. if sublayers is not None:
  56. self.update(sublayers)
  57. def __getitem__(self, key):
  58. return self._sub_layers[key]
  59. def __setitem__(self, key, sublayer):
  60. return self.add_sublayer(key, sublayer)
  61. def __delitem__(self, key):
  62. del self._sub_layers[key]
  63. def __len__(self):
  64. return len(self._sub_layers)
  65. def __iter__(self):
  66. return iter(self._sub_layers)
  67. def __contains__(self, key):
  68. return key in self._sub_layers
  69. def clear(self):
  70. """
  71. Clear all the sublayers in the LayerDict.
  72. Parameters:
  73. None.
  74. Examples:
  75. .. code-block:: python
  76. >>> import paddle
  77. >>> from collections import OrderedDict
  78. >>> sublayers = OrderedDict([
  79. ... ('conv1d', paddle.nn.Conv1D(3, 2, 3)),
  80. ... ('conv2d', paddle.nn.Conv2D(3, 2, 3)),
  81. ... ('conv3d', paddle.nn.Conv3D(4, 6, (3, 3, 3))),
  82. >>> ])
  83. >>> layer_dict = paddle.nn.LayerDict(sublayers=sublayers)
  84. >>> len(layer_dict)
  85. 3
  86. >>> layer_dict.clear()
  87. >>> len(layer_dict)
  88. 0
  89. """
  90. self._sub_layers.clear()
  91. def pop(self, key):
  92. """
  93. Remove the key from the LayerDict and return the layer of the key.
  94. Parameters:
  95. key (str): the key to be removed.
  96. Examples:
  97. .. code-block:: python
  98. >>> import paddle
  99. >>> from collections import OrderedDict
  100. >>> sublayers = OrderedDict([
  101. ... ('conv1d', paddle.nn.Conv1D(3, 2, 3)),
  102. ... ('conv2d', paddle.nn.Conv2D(3, 2, 3)),
  103. ... ('conv3d', paddle.nn.Conv3D(4, 6, (3, 3, 3))),
  104. >>> ])
  105. >>> layer_dict = paddle.nn.LayerDict(sublayers=sublayers)
  106. >>> len(layer_dict)
  107. 3
  108. >>> layer_dict.pop('conv2d')
  109. >>> len(layer_dict)
  110. 2
  111. """
  112. v = self[key]
  113. del self[key]
  114. return v
  115. def keys(self):
  116. """
  117. Return the iterable of the keys in LayerDict.
  118. Parameters:
  119. None.
  120. Examples:
  121. .. code-block:: python
  122. >>> import paddle
  123. >>> from collections import OrderedDict
  124. >>> sublayers = OrderedDict([
  125. ... ('conv1d', paddle.nn.Conv1D(3, 2, 3)),
  126. ... ('conv2d', paddle.nn.Conv2D(3, 2, 3)),
  127. ... ('conv3d', paddle.nn.Conv3D(4, 6, (3, 3, 3))),
  128. >>> ])
  129. >>> layer_dict = paddle.nn.LayerDict(sublayers=sublayers)
  130. >>> for k in layer_dict.keys():
  131. ... print(k)
  132. conv1d
  133. conv2d
  134. conv3d
  135. """
  136. return self._sub_layers.keys()
  137. def items(self):
  138. """
  139. Return the iterable of the key/value pairs in LayerDict.
  140. Parameters:
  141. None.
  142. Examples:
  143. .. code-block:: python
  144. >>> import paddle
  145. >>> from collections import OrderedDict
  146. >>> sublayers = OrderedDict([
  147. ... ('conv1d', paddle.nn.Conv1D(3, 2, 3)),
  148. ... ('conv2d', paddle.nn.Conv2D(3, 2, 3)),
  149. ... ('conv3d', paddle.nn.Conv3D(4, 6, (3, 3, 3))),
  150. >>> ])
  151. >>> layer_dict = paddle.nn.LayerDict(sublayers=sublayers)
  152. >>> for k, v in layer_dict.items():
  153. ... print(f"{k}:", v)
  154. conv1d : Conv1D(3, 2, kernel_size=[3], data_format=NCL)
  155. conv2d : Conv2D(3, 2, kernel_size=[3, 3], data_format=NCHW)
  156. conv3d : Conv3D(4, 6, kernel_size=[3, 3, 3], data_format=NCDHW)
  157. """
  158. return self._sub_layers.items()
  159. def values(self):
  160. """
  161. Return the iterable of the values in LayerDict.
  162. Parameters:
  163. None.
  164. Examples:
  165. .. code-block:: python
  166. >>> import paddle
  167. >>> from collections import OrderedDict
  168. >>> sublayers = OrderedDict([
  169. ... ('conv1d', paddle.nn.Conv1D(3, 2, 3)),
  170. ... ('conv2d', paddle.nn.Conv2D(3, 2, 3)),
  171. ... ('conv3d', paddle.nn.Conv3D(4, 6, (3, 3, 3))),
  172. >>> ])
  173. >>> layer_dict = paddle.nn.LayerDict(sublayers=sublayers)
  174. >>> for v in layer_dict.values():
  175. ... print(v)
  176. Conv1D(3, 2, kernel_size=[3], data_format=NCL)
  177. Conv2D(3, 2, kernel_size=[3, 3], data_format=NCHW)
  178. Conv3D(4, 6, kernel_size=[3, 3, 3], data_format=NCDHW)
  179. """
  180. return self._sub_layers.values()
  181. def update(self, sublayers):
  182. """
  183. Update the key/values pairs in sublayers to the LayerDict, overwriting the existing keys.
  184. Parameters:
  185. sublayers (LayerDict|OrderedDict|list[(key,Layer)...]): iterable of key/value pairs, the type of value is 'paddle.nn.Layer' .
  186. Examples:
  187. .. code-block:: python
  188. >>> import paddle
  189. >>> from collections import OrderedDict
  190. >>> sublayers = OrderedDict([
  191. ... ('conv1d', paddle.nn.Conv1D(3, 2, 3)),
  192. ... ('conv2d', paddle.nn.Conv2D(3, 2, 3)),
  193. ... ('conv3d', paddle.nn.Conv3D(4, 6, (3, 3, 3))),
  194. >>> ])
  195. >>> new_sublayers = OrderedDict([
  196. ... ('relu', paddle.nn.ReLU()),
  197. ... ('conv2d', paddle.nn.Conv2D(4, 2, 4)),
  198. >>> ])
  199. >>> layer_dict = paddle.nn.LayerDict(sublayers=sublayers)
  200. >>> layer_dict.update(new_sublayers)
  201. >>> for k, v in layer_dict.items():
  202. ... print(f"{k}:", v)
  203. conv1d : Conv1D(3, 2, kernel_size=[3], data_format=NCL)
  204. conv2d : Conv2D(4, 2, kernel_size=[4, 4], data_format=NCHW)
  205. conv3d : Conv3D(4, 6, kernel_size=[3, 3, 3], data_format=NCDHW)
  206. relu : ReLU()
  207. """
  208. assert isinstance(sublayers, Iterable), (
  209. "The type of sublayers is not iterable of key/value pairs, the type of sublayers is "
  210. + type(sublayers).__name__
  211. )
  212. if isinstance(sublayers, (OrderedDict, LayerDict, Mapping)):
  213. for key, layer in sublayers.items():
  214. self.add_sublayer(key, layer)
  215. else:
  216. # handle this format [(key1, layer1), (key2, layer2)...]
  217. for i, kv in enumerate(sublayers):
  218. if len(kv) != 2:
  219. raise ValueError(
  220. "The length of the "
  221. + str(i)
  222. + "'s element in sublayers is "
  223. + str(len(kv))
  224. + ", which must be 2."
  225. )
  226. self.add_sublayer(kv[0], kv[1])
  227. class ParameterList(Layer):
  228. """ParameterList Container.
  229. This container acts like a Python list, but parameters it contains will be properly added.
  230. Parameters:
  231. parameters (iterable, optional): Iterable Parameters to be added.
  232. Examples:
  233. .. code-block:: python
  234. >>> import paddle
  235. >>> class MyLayer(paddle.nn.Layer):
  236. ... def __init__(self, num_stacked_param):
  237. ... super().__init__()
  238. ... # create ParameterList with iterable Parameters
  239. ... self.params = paddle.nn.ParameterList(
  240. ... [paddle.create_parameter(
  241. ... shape=[2, 2], dtype='float32')] * num_stacked_param)
  242. ...
  243. ... def forward(self, x):
  244. ... for i, p in enumerate(self.params):
  245. ... tmp = self._helper.create_variable_for_type_inference('float32')
  246. ... self._helper.append_op(
  247. ... type="mul",
  248. ... inputs={"X": x,
  249. ... "Y": p},
  250. ... outputs={"Out": tmp},
  251. ... attrs={"x_num_col_dims": 1,
  252. ... "y_num_col_dims": 1})
  253. ... x = tmp
  254. ... return x
  255. ...
  256. >>> x = paddle.uniform(shape=[5, 2], dtype='float32')
  257. >>> num_stacked_param = 4
  258. >>> model = MyLayer(num_stacked_param)
  259. >>> print(len(model.params))
  260. 4
  261. >>> res = model(x)
  262. >>> print(res.shape)
  263. [5, 2]
  264. >>> replaced_param = paddle.create_parameter(shape=[2, 3], dtype='float32')
  265. >>> model.params[num_stacked_param - 1] = replaced_param # replace last param
  266. >>> res = model(x)
  267. >>> print(res.shape)
  268. [5, 3]
  269. >>> model.params.append(paddle.create_parameter(shape=[3, 4], dtype='float32')) # append param
  270. >>> print(len(model.params))
  271. 5
  272. >>> res = model(x)
  273. >>> print(res.shape)
  274. [5, 4]
  275. """
  276. def __init__(self, parameters=None):
  277. super().__init__()
  278. if parameters is not None:
  279. for idx, param in enumerate(parameters):
  280. assert isinstance(param, Parameter)
  281. self.add_parameter(str(idx), param)
  282. def __getitem__(self, idx):
  283. with param_guard(self._parameters):
  284. return self._parameters[str(idx)]
  285. def __setitem__(self, idx, param):
  286. assert isinstance(param, Parameter)
  287. setattr(self, str(idx), param)
  288. def __len__(self):
  289. return len(self._parameters)
  290. def __iter__(self):
  291. with param_guard(self._parameters):
  292. return iter(self._parameters.values())
  293. def append(self, parameter):
  294. """Appends a given parameter at the end of the list.
  295. Parameters:
  296. parameter (Parameter): parameter to append
  297. """
  298. idx = len(self._parameters)
  299. self.add_parameter(str(idx), parameter)
  300. return self
  301. class LayerList(Layer):
  302. """
  303. LayerList holds sublayers, and sublayers it contains are properly registered.
  304. Holded sublayers can be indexed like a regular python list.
  305. Parameters:
  306. sublayers (iterable of Layer, optional): sublayers to hold
  307. Examples:
  308. .. code-block:: python
  309. >>> import paddle
  310. >>> class MyLayer(paddle.nn.Layer):
  311. ... def __init__(self):
  312. ... super().__init__()
  313. ... self.linears = paddle.nn.LayerList(
  314. ... [paddle.nn.Linear(10, 10) for i in range(10)])
  315. ...
  316. ... def forward(self, x):
  317. ... # LayerList can act as an iterable, or be indexed using ints
  318. ... for i, l in enumerate(self.linears):
  319. ... x = self.linears[i // 2](x) + l(x)
  320. ... return x
  321. """
  322. def __init__(self, sublayers=None):
  323. super().__init__()
  324. if sublayers is not None:
  325. for idx, layer in enumerate(sublayers):
  326. self.add_sublayer(str(idx), layer)
  327. def _get_abs_idx(self, idx):
  328. if isinstance(idx, int):
  329. if not (-len(self) <= idx < len(self)):
  330. raise IndexError(
  331. f'index {idx} is out of range, should be an integer in range [{-len(self)}, {len(self)})'
  332. )
  333. if idx < 0:
  334. idx += len(self)
  335. return idx
  336. def __getitem__(self, idx):
  337. if isinstance(idx, slice):
  338. return self.__class__(list(self._sub_layers.values())[idx])
  339. else:
  340. idx = self._get_abs_idx(idx)
  341. return self._sub_layers[str(idx)]
  342. def __setitem__(self, idx, sublayer):
  343. idx = self._get_abs_idx(idx)
  344. return setattr(self, str(idx), sublayer)
  345. def __delitem__(self, idx):
  346. if isinstance(idx, slice):
  347. for k in range(len(self._sub_layers))[idx]:
  348. delattr(self, str(k))
  349. else:
  350. idx = self._get_abs_idx(idx)
  351. delattr(self, str(idx))
  352. str_indices = [str(i) for i in range(len(self._sub_layers))]
  353. self._sub_layers = OrderedDict(
  354. list(zip(str_indices, self._sub_layers.values()))
  355. )
  356. def __len__(self):
  357. return len(self._sub_layers)
  358. def __iter__(self):
  359. return iter(self._sub_layers.values())
  360. def append(self, sublayer):
  361. """
  362. Appends a sublayer to the end of the list.
  363. Parameters:
  364. sublayer (Layer): sublayer to append
  365. Examples:
  366. .. code-block:: python
  367. >>> import paddle
  368. >>> linears = paddle.nn.LayerList([paddle.nn.Linear(10, 10) for i in range(10)])
  369. >>> another = paddle.nn.Linear(10, 10)
  370. >>> linears.append(another)
  371. >>> print(len(linears))
  372. 11
  373. """
  374. self.add_sublayer(str(len(self)), sublayer)
  375. return self
  376. def insert(self, index, sublayer):
  377. """
  378. Insert a sublayer before a given index in the list.
  379. Parameters:
  380. index (int): index to insert.
  381. sublayer (Layer): sublayer to insert
  382. Examples:
  383. .. code-block:: python
  384. >>> import paddle
  385. >>> linears = paddle.nn.LayerList([paddle.nn.Linear(10, 10) for i in range(10)])
  386. >>> another = paddle.nn.Linear(10, 10)
  387. >>> linears.insert(3, another)
  388. >>> print(linears[3] is another)
  389. True
  390. >>> another = paddle.nn.Linear(10, 10)
  391. >>> linears.insert(-1, another)
  392. >>> print(linears[-2] is another)
  393. True
  394. """
  395. assert isinstance(index, int) and -len(self._sub_layers) <= index < len(
  396. self._sub_layers
  397. ), f"index should be an integer in range [{-len(self)}, {len(self)})"
  398. index = self._get_abs_idx(index)
  399. for i in range(len(self._sub_layers), index, -1):
  400. self._sub_layers[str(i)] = self._sub_layers[str(i - 1)]
  401. self._sub_layers[str(index)] = sublayer
  402. def extend(self, sublayers):
  403. """
  404. Appends sublayers to the end of the list.
  405. Parameters:
  406. sublayers (iterable of Layer): iterable of sublayers to append
  407. Returns:
  408. None
  409. Examples:
  410. .. code-block:: python
  411. >>> import paddle
  412. >>> linears = paddle.nn.LayerList([paddle.nn.Linear(10, 10) for i in range(10)])
  413. >>> another_list = paddle.nn.LayerList([paddle.nn.Linear(10, 10) for i in range(5)])
  414. >>> linears.extend(another_list)
  415. >>> print(len(linears))
  416. 15
  417. >>> print(another_list[0] is linears[10])
  418. True
  419. """
  420. offset = len(self)
  421. for i, sublayer in enumerate(sublayers):
  422. idx = str(offset + i)
  423. self.add_sublayer(idx, sublayer)
  424. return self
  425. class Sequential(Layer):
  426. """Sequential container.
  427. Sub layers will be added to this container in the order of argument in the constructor.
  428. The argument passed to the constructor can be iterable Layers or iterable name Layer pairs.
  429. Parameters:
  430. layers(Layer|list|tuple): Layer or list/tuple of iterable name Layer pair.
  431. Returns:
  432. None.
  433. Examples:
  434. .. code-block:: python
  435. >>> import paddle
  436. >>> data = paddle.uniform(shape=[30, 10], dtype='float32')
  437. >>> # create Sequential with iterable Layers
  438. >>> model1 = paddle.nn.Sequential(
  439. ... paddle.nn.Linear(10, 1), paddle.nn.Linear(1, 2)
  440. >>> )
  441. >>> model1[0] # access the first layer
  442. >>> res1 = model1(data) # sequential execution
  443. >>> # create Sequential with name Layer pairs
  444. >>> model2 = paddle.nn.Sequential(
  445. ... ('l1', paddle.nn.Linear(10, 2)),
  446. ... ('l2', paddle.nn.Linear(2, 3))
  447. >>> )
  448. >>> model2['l1'] # access l1 layer
  449. >>> model2.add_sublayer('l3', paddle.nn.Linear(3, 3)) # add sublayer
  450. >>> res2 = model2(data) # sequential execution
  451. """
  452. def __init__(self, *layers):
  453. super().__init__()
  454. if len(layers) > 0 and isinstance(layers[0], (list, tuple)):
  455. for name, layer in layers:
  456. self.add_sublayer(name, layer)
  457. else:
  458. for idx, layer in enumerate(layers):
  459. self.add_sublayer(str(idx), layer)
  460. def __getitem__(self, name):
  461. if isinstance(name, slice):
  462. return self.__class__(*(list(self._sub_layers.values())[name]))
  463. elif isinstance(name, str):
  464. return self._sub_layers[name]
  465. else:
  466. if name >= len(self._sub_layers):
  467. raise IndexError(f'index {name} is out of range')
  468. elif name < 0 and name >= -len(self._sub_layers):
  469. name += len(self._sub_layers)
  470. elif name < -len(self._sub_layers):
  471. raise IndexError(f'index {name} is out of range')
  472. return list(self._sub_layers.values())[name]
  473. def __setitem__(self, name, layer):
  474. assert isinstance(layer, Layer)
  475. setattr(self, str(name), layer)
  476. def __delitem__(self, name):
  477. name = str(name)
  478. assert name in self._sub_layers
  479. del self._sub_layers[name]
  480. def __len__(self):
  481. return len(self._sub_layers)
  482. def forward(self, input):
  483. for layer in self._sub_layers.values():
  484. input = layer(input)
  485. return input