社区所有版块导航
Python
python开源   Django   Python   DjangoApp   pycharm  
DATA
docker   Elasticsearch  
aigc
aigc   chatgpt  
WEB开发
linux   MongoDB   Redis   DATABASE   NGINX   其他Web框架   web工具   zookeeper   tornado   NoSql   Bootstrap   js   peewee   Git   bottle   IE   MQ   Jquery  
机器学习
机器学习算法  
Python88.com
反馈   公告   社区推广  
产品
短视频  
印度
印度  
Py学习  »  Python

python类属性与全局变量

rush dee • 5 年前 • 1388 次点击  

我在一个类中定义了一个属性(yyear),并使用一个全局变量(year)来初始化它。 当我在类中调用该方法时,它正在更新全局变量,即使我没有给它赋值。 为什么会这样?是的。 错误是对类变量定义的错误位置做了一些处理。

JAN = MAR =MAY =JUL =AUG= OCT =DEC=[0]*31
APR=JUN=SEP=NOV=[0]*30
FEB=[0]*28

YEAR = [JAN, FEB, MAR, APR, MAY, JUN, JUL, AUG, SEP, OCT, NOV, DEC]

class load:
    yYEAR=YEAR
    def __init__(self, DURATION, LTYPE):
        self.DURATION=DURATION
        self.LTYPE= LTYPE


    def daily247(self):
        if self.LTYPE==1 :
            for month in range(len(self.yYEAR)):
                for day in range(len(self.yYEAR[month])):
                    self.yYEAR[month][day]= 1
            else:
                self.yYEAR= YEAR
        self.dispLoad()     

    def dispLoad(self):
        print(self.yYEAR[0])
        print(self.yYEAR[1])

event1= load(10,1)
event1.daily247() # Expected result ones in all nested elements in YEAR

event2= load(10,0)
event2.daily247()  # Expected result zeros in all nested elements in YEAR
print(YEAR[1])  # expected result is to have zeros for all elements
Python社区是高质量的Python/Django开发社区
本文地址:http://www.python88.com/topic/47706
 
1388 次点击  
文章 [ 2 ]  |  最新文章 5 年前
rush dee
Reply   •   1 楼
rush dee    6 年前

这个错误是由于提到的python可变列表“小故障”造成的。然而,使用 [:] .copy() 在这种情况下不起作用,因为原始列表是二维的。

制作二维列表的深度副本解决了这个问题。

import copy

并使用 deepcopy()

class load:
    def __init__(self, DURATION, LTYPE):
        self.DURATION=DURATION
        self.LTYPE= LTYPE
        self.yYEAR = copy.deepcopy(YEAR)
gilch
Reply   •   2 楼
gilch    6 年前

你的 YEAR 是列表类型。python的列表是可变的。在所有大写字母中拼写它意味着一旦设置就不应该更改它(一个“常量”),但这只是一个python根本没有强制执行的约定。

如果您想要一个可以更改的单独副本,而不是对全局的引用,请执行
yYEAR = YEAR[:] 相反。那么这个副本就属于这个班了。

这个 [:] 是一种切片表示法,它只复制整件事,因为您没有指定边界。
yYEAR = YEAR.copy() 也可以,但前者在列表方面更为传统。

如果需要每个实例的副本,请在 __init__ 相反,就像
self.yYEAR = YEAR[:] 是的。真的,你可以拼写成 self.year .那就更像是蟒蛇了。

通过使用不可变元组而不是列表,您可以对常量进行更具防御性的编码,以防止意外变异。然后你就可以把你的本地名单 yYear = [*YEAR] 或者如果你还在使用旧版本的python, yYear = list(YEAR) 也行。但是要注意,不可变元组仍然可以包含可变元素,因此这也不能完全强制执行常量正确性。


还要注意,语法如下 APR=JUN=SEP=NOV=[0]*30 意味着这四个月都将引用同一个列表对象。