keras.layers.Layer中无法定义name的问题及解决

 更新时间:2023年02月21日 10:19:38   作者:唐僧爱吃唐僧肉  
这篇文章主要介绍了keras.layers.Layer中无法定义name的问题及解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

keras.layers.Layer中无法定义name问题

在使用keras之中定义层的时候,如果按照以下的方法直接在keras.layers.Layer中定义相应的self.name

会发生相应的报错

import params as pp
import params_flow as pf
import tensorflow as tf
import tensorflow.keras as keras
class MyParams(tf.keras.layers.Layer):   
    def __init__(self):
        super(MyParams,self).__init__()
        self.name = 'hello'
    def build(self,input_shape):
        self.dense0 = keras.layers.Dense(units = 25,
                                #kernel_initializer = self.create_initializer(),
                                name = "dense0")
    def call(self,inputs):
        results = self.dense0(inputs)
data = MyParams()
print(data.name)

此时会相应的报错

AttributeError: Can't set the attribute "name", likely because it conflicts with an existing read-only @property of the object. Please choose a different name.

也就是说,在keras之中的"self.name"为只读属性,不能够被定义,此时需要更换另外一个名字。

但是只有name这个属性能够对应到self.weights的名称之中,那么这里我们该如何定义呢

此时我发现,直接定义在__init__函数之前,可以对keras的名称完成相应的定义

import params as pp
import params_flow as pf
import tensorflow as tf
import tensorflow.keras as keras
class MyParams(tf.keras.layers.Layer):   
    name = 'hello'
    def __init__(self):
        super(MyParams,self).__init__()
        
    def build(self,input_shape):
        self.dense0 = keras.layers.Dense(units = 25,
                                #kernel_initializer = self.create_initializer(),
                                name = "dense0")
    def call(self,inputs):
        results = self.dense0(inputs)
data = MyParams()
print(data.name)

同时我们传入一个tensor的input_ids类型进行相应的输入,发现已经能够对weights的权重名称进行改变了

input_ids = keras.layers.Input(shape=(50,), dtype='int32', name="input_ids")
outputs = data(input_ids)
data.weights

对应的输出内容如下

[<tf.Variable 'hello/dense0/kernel:0' shape=(50, 25) dtype=float32, numpy=
 array([[-0.10505645,  0.09756875,  0.14427656, ..., -0.17254017,
         -0.18592533, -0.13920134],
        [-0.10033116, -0.17831415, -0.03435555, ..., -0.02460951,
          0.13194972,  0.21918347],
        [ 0.15699485, -0.24836   ,  0.01044622, ...,  0.04577217,
          0.23334488,  0.09155059],
        ...,
        [-0.22210473,  0.14221036,  0.07721925, ...,  0.03358698,
          0.08100349,  0.15415356],
        [-0.1433322 , -0.00878078, -0.0760702 , ..., -0.06091703,
          0.18796855, -0.19009456],
        [-0.0446853 ,  0.14639893,  0.1729418 , ..., -0.04699725,
          0.12940568, -0.24003454]], dtype=float32)>,
 <tf.Variable 'hello/dense0/bias:0' shape=(25,) dtype=float32, numpy=
 array([0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0., 0.,
        0., 0., 0., 0., 0., 0., 0., 0.], dtype=float32)>]

可以发现这里的权重内容已经变成hello打头的内容了,然而此时又一个对应的问题出现了,这得益于__init__函数前面内容的特殊性:

如果我们将对应的内容改为如下

import params as pp
import params_flow as pf
import tensorflow as tf
import tensorflow.keras as keras
class MyParams(tf.keras.layers.Layer):   
    print('begining')
    name = 'hello'
    def __init__(self):
        super(MyParams,self).__init__()
        
    def build(self,input_shape):
        self.dense0 = keras.layers.Dense(units = 25,
                                #kernel_initializer = self.create_initializer(),
                                name = "dense0")
    def call(self,inputs):
        results = self.dense0(inputs)
data = MyParams()
print(data.name)

此时运行的时候会运行一次print(‘begining’),输出对应的begining的内容,

也就是说__init__函数之前的内容会在定义MyParams这个类的时候就调用,而MyParams这个类只会被定义一次,也就是说__init__函数之前的内容只会被调用一次。

这样就带来了一个问题,也就是name = 'hello’这里的name没有办法修改,也就是说我们需要想一种办法将__init__函数前面的name='hello’修改一次,这里究竟该如何修改呢?

此时我们只需要在定义之后每次使用的时候重新对name的值进行定义即可

具体的例子如下

import tensorflow as tf
import tensorflow.keras as keras
class MyParams(tf.keras.layers.Layer):   
    #print('begining')
    name = 'hello'
    def __init__(self):
        super(MyParams,self).__init__()
    def build(self,input_shape):
        self.dense0 = keras.layers.Dense(units = 25,
                                #kernel_initializer = self.create_initializer(),
                                name = "dense0")
    def call(self,inputs):
        results = self.dense0(inputs)
data = MyParams()
print(data.name)
data.name = 'hello10000'
print(data.name)

输出的内容为

输出的内容1

进一步查看一下对应的weight的属性

查看weights对应的属性名称

可以看出这里weights之中的属性名称已经成功地被我们定义了

总结

由于name的属性的特殊性,如果在__init__函数之中直接定义可读变量会造成报错,此时我们需要调整思路,通过__init__函数之前定义name变量实现对于name的修改

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • 如何使用Python提取Chrome浏览器保存的密码

    如何使用Python提取Chrome浏览器保存的密码

    今天小编教大家怎么用Python提取Chrome浏览器保存的密码,在这需要导入一些必要模块定义一些有用的函数来帮助我们在主函数中调用,具体实例代码跟随小编一起学习下吧
    2021-06-06
  • 解析Python中的异常处理

    解析Python中的异常处理

    这篇文章主要介绍了解析Python中的异常处理,异常处理是学习每门编程语言都必须掌握的重要知识,本文代码基于Python2.x版本,需要的朋友可以参考下
    2015-04-04
  • 关于Python解包知识点总结

    关于Python解包知识点总结

    在本篇文章里小编给各位分享的是关于Python解包知识点总结,有兴趣的朋友们可以学习参考下。
    2020-05-05
  • python导出mysql指定binlog文件实现demo

    python导出mysql指定binlog文件实现demo

    这篇文章主要介绍了python导出mysql指定binlog文件实现demo,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-12-12
  • 用python3读取python2的pickle数据方式

    用python3读取python2的pickle数据方式

    今天小编就为大家分享一篇用python3读取python2的pickle数据方式,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2019-12-12
  • Python中基础数据类型 set集合知识点总结

    Python中基础数据类型 set集合知识点总结

    在本篇文章里小编给大家总结了一篇关于Python中基础数据类型 set集合知识点总结内容,有需要的朋友们可以学习下。
    2021-08-08
  • Python3+selenium配置常见报错解决方案

    Python3+selenium配置常见报错解决方案

    这篇文章主要介绍了Python3+selenium配置常见报错解决方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-08-08
  • Python使用Asyncio进行web编程方法详解

    Python使用Asyncio进行web编程方法详解

    这篇文章主要为大家介绍了Python使用Asyncio进行web编程的方法示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-08-08
  • 详解Python常用的魔法方法

    详解Python常用的魔法方法

    在Python中,所有以“__”双下划线包起来的方法,都统称为“Magic Method”,中文称『魔术方法』,例如类的初始化方法 __init__ ,Python中所有的魔术方法均在官方文档中有相应描述,今天给大家整理了本篇文章,需要的朋友可以参考下
    2021-06-06
  • Python+KgCaptcha实现验证码的开发详解

    Python+KgCaptcha实现验证码的开发详解

    验证码通常是为了区分用户是人还是计算机,也可以防止解开密码等恶意行为,而客户端上多数会用在关键操作上。现在验证码的种类样式也特别多,本文主要介绍了如何用Python和KgCaptcha做出验证码功能,需要的可以参考一下
    2023-04-04

最新评论