1. __init__ # 构造函数
2. Self.name = name # 实例变量、普通属性 或者叫 普通字段
3. public_object = "public" # 类变量、公有属性 或则叫 静态字段
4. self.__heart= "Normal" # 私有属性 在外面无法访问
5. def shot(self) # 类方法
class Role(object): #1、在定义类时继承object就是新式类,没有就是就是旧类式
public_object = "public" #2、在类例定义一个公有属性:所有实例都能访问的属性叫“公有属性”
def __init__(self,name,role,weapon,life_value=100,money=15000): #构造函数==初始化方法:模板
self.name = name #3、普通属性
self.__heart= "Normal" #4、私有属性在外面无法访问
def shot(self): #5、类的方法
print("%s is shooting..."%self.name)
更详细代码在这里:
class Role(object): #在定义类时继承object就是新式类,没有就是就是旧类式
public_object = "public" #在类例定义一个公有属性:所有实例都能访问的属性叫“公有属性”
def __init__(self,name,role,weapon,life_value=100,money=15000): #构造函数==初始化方法:模板
self.name = name #普通属性
self.role = role
self.weapon = weapon
self.life_value = life_value
self.money = money
self.__heart= "Normal" #私有属性在外面无法访问
def shot(self): #类的方法
print("%s is shooting..."%self.name)
def got_shot(self):
print("ah...,I got shot...")
def buy_gun(self,gun_name):
print("%s just bought %s" %(self.name,gun_name))
self.weapon = gun_name #在购买后让实例值改变
#在下面实例化其实就是传入: Role('r1','Alex','police','AK47') 把r1传给了self
#r1就是实例化后产生的当前Role类的实例,所以self就是实例本身
#我理解r1其实就是__init__函数的内存地址
#所以在下面函数中调用self.name就相当于调用r1.name所以可以调用
r1 = Role('Alex','police','AK47') #生成一个角色 只要一实例化就会自动调用__init__
r2 = Role('Jack','terrorist','B22') #生成一个角色
r1.shot()
print(r2.weapon) #在调用r2.buy_gun('AK47')前是:B22
r2.buy_gun('AK47')
print(r2.weapon) #在调用r2.buy_gun('AK47')后是:AK47
#私有属性
# print(r2.__heart) #这里的.__heart是私有属性,所以在外部无法访问
print(r2._Role__heart) #强制访问私有属性的方法
#公有属性
print(r2.public_object) #打印出类的公有属性
Role.public_object = 'change_public' #从全局改变类的公有属性 r1,r2的公有属性都会变
print(r2.public_object) #这里打印可以看出公有属性变成类“change_public"
r2.public_object = "public_r2" #改变r2对象的公有属性,r1不会变
print(r2.public_object) #打印出改变后的r2公有属性
类中函数私有化:
class Secretive:
def __accessible(self):
print("you can't see me,unless you're calling internally")
def accessible(self):
print("The secret message is:")
self.__accessible()
s = Secretive()
s.accessible()
# 运行结果:
# The secret message is:
# you can't see me,unless you're calling internally
对类中属性和方法进行一种封装,隐藏了实现细节
子类继承父类后,就具有了父类的所有属性和方法,先继承,后重写
新式类深度优先、经典类广度优先
黑人,白人都继承父类Person就可以都有父类的属性和方法了:
class Person(object):
def __init__(self,name,age): #执行Person.__init__(self,name,age)时就会将传入的参数执行一遍
self.name = name #所以在BlackPerson中不仅有name,age而且还有sex
self.age = age
self.sex = "normal"
def talk(self):
print("person is talking....")
class WhitePerson(Person):
pass
class BlackPerson(Person):
def __init__(self,name,age,strength): #先覆盖,再继承,再重构
#先覆盖父类的__init__方法,再继承父类__init__,再加自己的参数
Person.__init__(self,name,age) #先继承父类Person,这里self就是BlackPerson本身
#先将name,age传给子类BlackPerson,然后调用Person.__init__构造方法将参数出入父类()
self.strength = strength #然后再重构自己的方法,即写自己的参数
print(self.name,self.age,self.sex)
print(self.strength)
def talk(self):
print("black balabla")
def walk(self):
print("is walking....")
b = BlackPerson("wei er smith",22,"Strong")
b.talk()
b.walk()
# 运行结果:
# wei er smith 22 normal
# Strong
# black balabla
# is walking....
# person is talking....
一种接口,多种表现形式
中国人、和美国人都能讲话,调用中国人的类讲中文,调用美国人将英文
多态举例:
class Animal:
def __init__(self, name): # Constructor of the class
self.name = name
def talk(self): # Abstract method, defined by convention only
raise NotImplementedError("Subclass must implement abstract method")
class Cat(Animal):
def talk(self):
return 'Meow!'
class Dog(Animal):
def talk(self):
return 'Woof! Woof!'
animals = [Cat('Missy'),
Dog('Lassie')]
for animal in animals:
print(animal.name + ': ' + animal.talk())
# 运行结果:
# Missy: Meow!
# Lassie: Woof! Woof!
- pythn3无论新式类还是经典类都是用 广度优先
- python2中,新式类:广度优先,经典类:深度优先
1)首先,写法不一样:
class A: #经典类写法
pass
class B(object): #新式类写法
pass
SchoolMember.init(self,name,age,sex) #经典类写法
super(Teacher,self).init(name,age,sex) #新式类写法
注:如果一个类的构造方法被重写,那么就需要调用超类的构造方法,否则对象可能不能给被正确的初始化
其实调用超类构造方法很容易,SongBird类中只添加一行代码
如果将下面代码没有2中的那句调用父类Bird中的self.hungry方法会报错
if self.hungry == True:
AttributeError: SongBird instance has no attribute 'hungry'
例1: 新式类,经典类,无超类构造方法报错:
class Bird:
def __init__(self):
self.hungry = True
def eat(self):
if self.hungry == True:
print("Aaaah...")
self.hungry = False
else:
print("no, thanks")
a = Bird()
a.eat()
a.eat()
class SongBird(Bird):
def __init__(self):
# Bird.__init__(self) # 经典类写法
# super(SongBird,self).__init__() # 新式类写法
self.sound = 'Squawk'
def sing(self):
print(self.sound)
b = SongBird()
b.sing()
b.eat()
例2: 模拟学校角色,讲师,学生:
class SchoolMember(object):
'''学校成员基类'''
member = 0
def __init__(self,name,age,sex):
self.name = name
self.age = age
self.sex = sex
self.enroll() #只要一实例化就自动注册
def enroll(self):
'''注册'''
print("just enrolled a new school member [%s]"%self.name)
SchoolMember.member += 1
def tell(self):
print("------info:%s-------"%self.name)
for k,v in self.__dict__.items():
print("\t",k,':',v)
def __del__(self): #'''析构方法'''
print("开除了[%s]...."%self.name)
SchoolMember.member -= 1
class Teacher(SchoolMember):
def __init__(self,name,age,sex,salary,course):
SchoolMember.__init__(self,name,age,sex) #经典类写法
#super(Teacher,self).__init__(name,age,sex) #新式类写法
self.salary = salary
self.cource = course
# self.enroll()
def teaching(self):
print("Teacher [%s] is teachin g [%s]"%(self.name,self.cource))
class Student(SchoolMember):
def __init__(self,name,age,sex,course,tuition):
SchoolMember.__init__(self,name,age,sex)
self.cource = course
self.tuition = tuition
self.amount = 0
# self.enroll()
def pay_tuition(self,amount):
print("student [%s] has just paied [%s]"%(self.name,amount))
self.amount += amount
t1 = Teacher("Wusir",28,"F*M",3000,"python")
s1 = Student("HaiTao",38,"M","PYS15",300000)
s2 = Student("Lichuang",12,"M","PYS15",11000)
print(SchoolMember.member)
del s2
print(SchoolMember.member)
t1.tell()
@staticmethod静态方法:
class Dog(object):
def __init__(self,name):
self.name = name
@staticmethod
def eat():
print("I am a static method")
d = Dog("ChenRonghua")
d.eat() #方法1:使用实例调用
Dog.eat() #方法2:使用类直接调用
作用:无需实例化直接被类调用
特性: 类方法只能访问类变量,不能访问实例变量
类方法使用场景: 当我们还未创建实例,但是需要调用类中的方法
调用方式: 既可以被类直接调用,也可以通过实例调用
@classmethod类方法:
class Dog(object):
name = '类变量' #在这里如果不定义类变量仅定义实例变量依然报错
def __init__(self,name):
self.name = '实例变量'
self.name = name
@classmethod
def eat(self,food):
print("%s is eating %s"%(self.name,food))
Dog.eat('baozi') #方法1:使用类直接调用
d = Dog("ChenRonghua")
d.eat("包子") #方法2:使用实例d调用
作用:属性方法把一个方法变成一个属性,隐藏了实现细节,调用时不必加括号直接d.eat即可调用self.eat()方法
例1:@property属性方法:
class Dog(object):
def __init__(self, name):
self.name = name
@property
def eat(self):
print(" %s is eating" % self.name)
d = Dog("ChenRonghua")
d.eat()
# 调用会出以下错误, 说NoneType is not callable, 因为eat此时已经变成一个静态属性了,
# 不是方法了, 想调用已经不需要加()号了,直接d.eat就可以了
例2:航空公司具体实例:
class Flight(object):
def __init__(self,name):
self.flight_name = name
def checking_status(self):
print("checking flight %s status " % self.flight_name)
return 1
@property
def flight_status(self):
status = self.checking_status()
if status == 0 :
print("flight got canceled...")
elif status == 1 :
print("flight is arrived...")
elif status == 2:
print("flight has departured already...")
else:
print("cannot confirm the flight status...,please check later")
f = Flight("CA980")
f.flight_status
new:
1. __new__ 是在一个对象实例化的时候所调用的第一个方法
2. 它的第一个参数是这个类,其他的参数是用来直接传递给 __init__ 方法
3. __new__ 决定是否要使用该 __init__ 方法,因为 __new__ 可以调用其他类的构造方法或者直接返回别的实例对象来作为本类的实例,如果 __new__ 没有返回实例对象,则 __init__ 不会被调用
4. __new__ 主要是用于继承一个不可变的类型比如一个 tuple 或者 string
5. __new__ return的是一个构建的实例
init:
构造器,当一个实例对象被定义时调用
del:
析构器,当删除一个实例对象时调用
hasattr判断对象是否有某方法,返回True或False:
class Dog(object):
def eat(self,food):
print("eat method!!!")
d = Dog()
#hasattr判断对象d是否有eat方法,有返回True,没有返回False
print(hasattr(d,'eat')) #True
print(hasattr(d,'cat')) #False
getattr获取对象方法内存地址:
class Dog(object):
def eat(self):
print("eat method!!!")
d = Dog()
if hasattr(d,'eat'): # hasattr判断实例是否有eat方法
func = getattr(d, 'eat') # getattr获取实例d的eat方法内存地址
func() # 执行实例d的eat方法
#运行结果: eat method!!!
使用stattr给类实例对象动态添加一个新的方法:
class Dog(object):
def eat(self,food):
print("eat method!!!")
d = Dog()
def bulk(self): #给Dog类添加一个方法必须先写这个方法
print('bulk method add to Dog obj')
d = Dog()
setattr(d,"bulk",bulk) #将bulk方法添加到实例d中,命名为talk方法
d.bulk(d) #实例d调用刚刚添加的talk方法时必须将实例d自身当变量传入,因为他不会自己传入self
#1. 注:setattr(x,’y’,z)用法: x就是实例对象 y就是在实例中调用时用的名字,z就是改变属性的值/或则要添加的函数的名字(正真的函数)
#2. setattr( 具体实例名称 , ’在类中调用时使用的名称’ , 要添加的真实函数的名称)
#3. 作用: setattr(d,"bulk",bulk) 将bulk方法添加到实例d中并且在实例中以bulk名称调用
使用stattr给类实例对象动态添加一个新的属性:
class Dog(object):
def __init__(self,name):
self.name = name
def eat(self,food):
print("eat method!!!")
d = Dog('Fly')
#1 实例d中没有sex这个属性,就会动态添加一个属性 sex = Male
setattr(d,"sex",'Male') #给实例d添加一个属性:sex=Male
print("将实例d添加一个新属性sex=Male:\t",d.sex)
#2 如果实例d中本身存在这个属性那么 新的值就会覆盖这个属性
setattr(d,'name','name change to jack')
print("原本名字是Fly改变后的名字是:\t",d.name)
# 运行结果:
# 将实例d添加一个新属性sex=Male: Male
# 原本名字是Fly改变后的名字是: name change to jack
delattr:
class Dog(object):
def __init__(self,name):
self.name = name
def eat(self,food):
print("%s is eating....."%self.name)
d = Dog("NiuHanYang")
choice = input(">>:").strip()
if hasattr(d,choice):
delattr(d,choice) #使用delattr(d,choice)删除实例的属性那么所以下面打印就会报错
print(d.name)
# 运行结果:
# >>:name #输入的值是name
# 下面是报错信息
# Traceback (most recent call last):
# File "C:/Users/admin/PycharmProjects/s14/Day7/test1.py", line 10, in
创建单例模式举例:
class Foo(object):
instance = None
def __init__(self):
self.name = 'alex'
def __new__(cls, *args, **kwargs):
if Foo.instance:
return Foo.instance
else:
Foo.instance = object.__new__(cls,*args,**kwargs)
return Foo.instance
obj1 = Foo() # obj1和obj2获取的就是__new__方法返回的内容
obj2 = Foo()
print(obj1,obj2) # 运行结果: <__main__.Foo object at 0x00D3B450> <__main__.Foo object at 0x00D3B450>
# 运行结果说明:
# 这可以看到我们新建的两个Foo()对象内存地址相同,说明使用的•同一个类,没有重复建立类