# 作业一:总结 # 1.什么是绑定到对象的方法,如何定义,如何调用,给谁用?有什么特性? ######################################################################################################################## # 绑定到对象的方法:对象/实例本身只有数据属性,但是python的class机制会将类的函数绑定到对象上,称为对象的方法,或者叫绑定方法。 # 如何定义:在类中定义相应的函数,而且函数的参数中必须要有 self 参数,代表绑定在 object 对象上。 # 如何调用,给谁用:对象.函数名(self#代表将self对象自己传进去),然后就会执行。自然绑定到对象的方法是给对象来用。 # 有什么特性:对象的绑定方法的特别之处在于:obj.func()会把obj传给func的第一个参数。 #举例: # class A: # def a(self): # print("I love egon forever!") # p=A() # q=A() # print(p.__dict__,q.__dict__) # print(A.__dict__) # print(p.a) #<bound method A.a of <__main__.A object at 0x0000000001EAD0B8>> # print(q.a) #<bound method A.a of <__main__.A object at 0x0000000001EAD5C0>> # #绑定方法唯一绑定一个对象,同一个类的方法绑定到不同的对象上,属于不同的方法,内存地址都不会一样。 # print(A.a) #<function A.a at 0x00000000021DF730> # p.a() #I love egon forever! ######################################################################################################################## # 2.什么是绑定到类的方法,如何定义,如何调用,给谁用?有什么特性 # 绑定到类的方法:类的方法是给类用的,类在使用时会将类本身当作参数传给类方法的第一个参数。 # 如何定义:Python内置了函数classmethod来把类中的函数定义成类的方法。 # 如何调用,给谁用: 类名.方法名(cls#将调用者自动传入,即子类=cls),因为是类调用的绑定方法,即调用的那个类用该方法实例化。给类用。 # 有什么特性:类在使用时会将类本身当作参数传给类方法的第一个参数。 # 举例: # class A: # @classmethod # def a(cls): # print("I love egon forever!") # return cls() # class B(A): # pass # b=B.a() # print(b) #<__main__.B object at 0x00000000021CD5C0> 子类B调用的是父类A的绑定方法,但是实例b却是由子类B生成的。 ######################################################################################################################## # 3.什么是解除绑定的函数,如何定义,如何调用,给谁用?有什么特性 # 什么是解除绑定的函数:但凡是定义在类的内部,并且被staticmethod装饰器修饰过的方法,都是解除绑定的方法。 # 如何定义:位于类定义的命名空间中,不会对任何实例类型进行操作,python为我们内置了函数staticmethod来把类中的函数定义成静态方法。 # 如何调用:类/对象.函数名(),来直接生成实例或调用函数。 # 特性:没有任何自动传值的功能。 #举例: # class Forever: # def __init__(self,name,age): # self.name=name # self.age=age # @staticmethod # def love(): # return Forever("egon","17 year old rainy season ") # lover=Forever.love() # print(lover) #<__main__.Forever object at 0x000000000222D668> # print(lover.__dict__) #{'name': 'egon', 'age': '17 year old rainy season '} # 4.什么是property,如何定义,如何使用,给谁用,什么情况下应该将一个属性定义成property,有什么好处? ######################################################################################################################## # 什么是property,如何定义,如何使用,给谁用:property是一种特殊的属性,访问它时会执行一段功能(函数)然后返回值,使用的时候直接 # obj.func,但是这个func必须是在property装饰下才能这么直接调用的。这是给对象调用的,只不过伪装成了一个对象的数据属性,其实是执行 # 了一段函数。 # 什么情况下应该将一个属性定义成property,有什么好处:将一个类的函数定义成特性以后,对象再去使用的时候obj.func,根本无法察觉自己的 # func是执行了一个函数然后计算出来的,这种特性的使用方式遵循了统一访问的原则,与obj.func()完全不同。 #举例: # 1、 # import math # class Gen: # def __init__(self,length,diameter): # self.length=length # self.diameter=diameter # self.volume=length*math.pi*((diameter/2)**2) # egon=Gen(180,100) # print(egon.volume) # 2、 # class Gen: # def __init__(self,length,diameter): # self.length=length # self.diameter=diameter # @property # def volume(self): # return self.length*math.pi*((self.diameter/2)**2) # egon=Gen(180,100) # print(egon.volume) # print(Gen.volume) #<property object at 0x0000000001DB8318> 可以发现类无法调用property了,已经成为对象的专属数据属性了。 # print(Gen.volume(egon)) # 3、 # class Gen: # def __init__(self,length,diameter): # self.length=length # self.diameter=diameter # def volume(self): # return self.length*math.pi*((self.diameter/2)**2) # egon=Gen(180,100) # print(egon.volume()) # 可以通过以上三个例子说明:加了property装饰器的在类下面定义的函数,可以直接作为对象的数据属性类调用。 ######################################################################################################################## # 作业二: # 要求一:自定义用户信息数据结构,写入文件,然后读出内容,利用eval重新获取数据结构 # with open('user.db','w') as write_file: #以覆盖写的方式创建”user.db",没有该文件的话,自己创建。 # write_file.write(str #写入文件的必须是字符串。 # ({ # "egon":{"password":"123",'status':False,'timeout':0}, # "alex":{"password":"456",'status':False,'timeout':0}, # }) # ) # with open('user.db','r') as read_file: # data=read_file.read() # d=eval(data) #读取出来的也是字符串,所以需要转化为字典才能提取key。 # print(d['egon']['password']) # print(d['egon']['status']) # print(d['egon']['timeout']) ######################################################################################################################## # # 要求二:定义用户类,定义属性db,执行obj.db可以拿到用户数据结构 #以下文件中内容为自我修改的。 # class User: #{"yuan":{"age":16,"sex":"lang","password":519},"egon":{"age":17,"sex":"mid", # db_path = 'user.db' # "password":520},"alex":{"age":18,"sex":"none","password":521}}#"user.db"文件中的内容。 # def __init__(self,username): #User类的内置函数,传入username参数,后面引用。 # self.username=username #定义self的数据属性,obj.username # @property #加装饰器。 # def db(self): #定义db函数,将其加装饰器后实际就成了obj的数据属性了,只不过默默执行了函数。 # data=open(self.db_path,'r').read() #打开self.db_path,因为self自己里面没有,所以到类里找到了“user.db",限定这一类统一 # print(type(data)) #使用”user.db"。 # return eval(data) #由于读取到的是str格式,所以用eval转化为字典格式。 # u=User('egon') #实例化u,名字为egon。 # print(u.db['egon']) #实际就是eval(data)["egon"]--->{'age': 17, 'sex': 'mid', 'password': 520} # print(u.db['egon']['password']) #再找key=“password"-->520。结束。 ######################################################################################################################## # 要求三:分析下述代码的执行流程 # import time #导入时间模块,以备后用。 # class User: #定义用户类。 # db_path='user.db' #定义类的数据属性,这样所有User类实例化的对象,都可以用这个类的属性,统一归一设计。 # def __init__(self,name): #定义类的内置函数,实例化中使用。 # self.name=name #对象的数据属性只有一个,self.name。 # @property #增加property装饰器,为对象增加一个属性,其实这个属性是执行函数。 # def db(self): #实际上是为对象定义db函数。 # with open(self.db_path,'r') as read_file: #只读方式打开self.db_path,对象本身没有,只能到类中找,找到了"user.db"文件。 # info=read_file.read() #info变量拿到文件所有内容。 # return eval(info) #返回给self.db文件的所有内容,这是爱根的一个bug,只要print(self.db),就能拿到所有用户信息。 # @db.setter #这是被property装饰的属性,在赋值或修改的时候使用。 # def db(self,value): #其实是定义self对象的修改函数或新增函数,把value传进去。 # with open(self.db_path,'w') as write_file: #以覆盖写的方式打开"user.db"文件。 # write_file.write(str(value)) #将新的值value写进去。以字符串的形式。 # write_file.flush() #然后立即刷新,即使刷新,避免时间差,但时间差还是有的,只不过很小罢了。 # def login(self): #定义类的函数属性,即对象的绑定方法。 # data=self.db #数据来源为self.db,其实就是拿到了property下db(self)函数的返回值evla(info),即字典格式的文件内容。 # if data[self.name]['status']: #先找出该对象的key=登录名字,继而找出key="status"的登录状态,看是True还是False。 # print('已经登录') #上面表示如果是True,则打印该内容。 # return True #然后返回True,终结该函数。 # if data[self.name]['timeout'] < time.time(): #如果不是True的话判断该用户名下对应的时间是否小于当前时间,只有小于才有资格 # count=0 #继续往suite里面走,然后定义计数器。 # while count < 3: #当count<3的时候,才会进行这个循环。 # passwd=input('password>>: ') #与用户交互,输入密码。 # if not passwd: #密码只有对与错,定义这个的意思是如果输入的是回车键"enter",那么将不会执行count+=1,会再让你输入。 # continue #直接跳出循环,从偷开始循环。 # if passwd == data[self.name]['password']: #如果密码正确的话,进入下个suite。 # data[self.name]['status']=True #修改登录状态,将False改为True。 # data[self.name]['timeout']=0 #若以前被锁定过,因为登录成功了,所以将时间修改为0。 # self.db=data #调用@db.setter下的db函数,将修改过的data内容写进去。 # break #跳出该循环,因为登录成功了。 # count+=1 #这个count+=1是输错密码的时候用的,因为正确的时候有break,所以可以放在这里。 # else: #当count=3的时候不执行以上while循环了,所以会执行这个else。我将他修改为这个形式。 # data[self.name]['timeout']=time.time()+10 #else隐掉,下面两行往前缩进一个suite。一样的效果, # self.db=data #都是在count=3的时候执行的,和上面的count+=1一个道理。就是将timeout增加10s。 # else: #当既没有登录,而且时间被滞后的情况下,执行该项。 # print('账号已经锁定10秒') #告诉用户账户被锁定。 # # u1=User('egon') # u1.login() # u2=User('alex') # u2.login() ######################################################################################################################## # 要求四:根据上述原理,编写退出登录方法(退出前要判断是否是登录状态),自定义property,供用户查看自己账号的锁定时间 import time class User: db_path="user.db" def __init__(self,name): self.name=name @property def db(self): with open(self.db_path,"r") as read_file: info=read_file.read() return eval(info) @property def lock_time(self): lock_time=self.db[self.name]["timeout"]-time.time() return lock_time @db.setter def db(self,value): with open(self.db_path,"w") as write_file: write_file.write(str(value)) write_file.flush() def login(self): data=self.db if data[self.name]['status']: print('已经登录') return True if self.lock_time<=0: count=0 while count < 3: passwd=input('password>>: ') if not passwd: continue if passwd == data[self.name]['password']: data[self.name]['status']=True data[self.name]['timeout']=0 self.db=data print("你已经登录!") break count+=1 else: data[self.name]['timeout']=time.time()+10 self.db=data else: print('账号已经锁定10秒,还剩%s秒'%self.lock_time) def logout(self): data=self.db if data[self.name]["status"]: while True: choice = input("Do you want to logout right now?Y/N") if choice=="Y": data[self.name]["status"]=False self.db=data break if choice=="N": exit() if not choice: continue else: print("Fuck you ,input Error!Sb!") elif not data[self.name]["status"]: if self.lock_time>0: print("you have been locked and haven't login,you must wait %ss to login!"%self.lock_time) if self.lock_time<=0: while True: choice=input("you haven't login,do you want to login?Y/N") if choice=="Y": self.login() break if choice=="N": exit() if not choice: continue else: print("Fuck you ,input Error!SB!") # u1=User('egon') # u1.login() # u1.logout() # u2=User('alex') # u2.login() # u2.logout()