我在一个类上工作,以创建各种对称AE。我现在将该类移植到TF 2.0,它比我想象的要复杂。但是,我使用图层和模型的子类来实现此目的。因此,我想将几个keras层分组到一个keras层。但是如果我想写这样的话:
def __init__(self, name, keras_layer, **kwargs):
self.keras_layer = tf.keras.layer.Conv2D
super(CoderLayer, self).__init__(name=name, **kwargs)
我收到以下错误,因为tf要使用此未初始化的层:
TypeError: _method_wrapper() missing 1 required positional argument: 'self'
我也尝试将其包装在列表中,但也没有用。
编辑
这是一个可行的最小示例和完整的回溯:
import tensorflow as tf
print(tf.__version__) # 2.0.0-alpha0
class CoderLayer(tf.keras.layers.Layer):
def __init__(self, name, keras_layer):
self.keras_layer = keras_layer
self.out = keras_layer(12, [3, 3])
super(CoderLayer, self).__init__(name=name)
def call(self, inputs):
return self.out(inputs)
inputs = tf.keras.Input(shape=(200, 200, 3), batch_size=12)
layer = CoderLayer("minimal_example", tf.keras.layers.Conv2D)
layer(inputs)
追溯:
Traceback (most recent call last):
File "..\baseline_cae.py", line 24, in <module>
layer(inputs)
File "..\AppData\Local\Continuum\anaconda3\lib\site-packages\tensorflow\python\keras\engine\base_layer.py", line 581, in __call__
self._clear_losses()
File "..\AppData\Local\Continuum\anaconda3\lib\site-packages\tensorflow\python\training\tracking\base.py", line 456, in _method_wrapper
result = method(self, *args, **kwargs)
File "..\AppData\Local\Continuum\anaconda3\lib\site-packages\tensorflow\python\keras\engine\base_layer.py", line 818, in _clear_losses
layer._clear_losses()
TypeError: _method_wrapper() missing 1 required positional argument: 'self'
最佳答案
问题在于将未实例化的类设置为tf.keras.layers.Layer
子类中的属性。如果删除以下行self.keras_layer = keras_layer
该代码将工作:
import tensorflow as tf
class CoderLayer(tf.keras.layers.Layer):
def __init__(self, name, keras_layer):
super(CoderLayer, self).__init__(name=name)
self.out = keras_layer(12, [3, 3])
def call(self, inputs):
return self.out(inputs)
inputs = tf.keras.Input(shape=(200, 200, 3), batch_size=12)
layer = CoderLayer("minimal_example", tf.keras.layers.Conv2D)
print(layer(inputs))
# Tensor("minimal_example_3/conv2d_12/BiasAdd:0", shape=(12, 198, 198, 12), dtype=float32)
这可能是一个错误。 This是已经提出的类似问题(如果将未实例化的类放入列表中并尝试
__setattr__()
,则将得到相同的异常)。如果要使用多层,则可以采用以下解决方法:
class CoderLayer(tf.keras.layers.Layer):
def __init__(self, name, layername):
super(CoderLayer, self).__init__(name=name)
self.layer = layername
self.out = tf.keras.layers.__dict__[layername](1, 2)
def call(self, inputs):
return self.out(inputs)
inputs = tf.random.normal([1, 3, 3, 1])
layer = CoderLayer("mylayer", 'Conv2D')
layer(inputs).numpy()