面向对象 -- 基本语法

创建类
class 类名:
    def 方法名( self [, 参数列表] ):
        pass
创建对象
对象变量 = 类名( [ 参数列表 ] )
  • 定义一个猫类 Cat
  • 定义两个方法 eat 和 drink
  • 定义一个猫类的对象 tom
class Cat:
    def eat(self):
        print("小猫爱吃鱼")
    def drink(self):
        print("小猫在喝水")
tom = Cat()

print(tom)
<__main__.Cat object at 0x00000000029BF0B8>
  • Cat()在内存中创建了Cat类对象
  • 变量tom中保存着该对象的内存地址,使用print输出对象变量时,会输出对象在内存中的地址(以16进制表示)
  • 通过对象变量.方法名()的形式调用方法
tom.drink()
tom.eat()

小猫在喝水
小猫爱吃鱼
  • 为猫类定义一个名字属性
class Cat:
    def __init__(self, name):        
        self.name = name  # 添加name属性
    def eat(self):
        print("小猫%s爱吃鱼" % self.name)
    def drink(self):
        print("小猫%s在喝水" % self.name)

使用类名()创建对象时,自动执行以下操作:

  • 对象在内存中分配空间
  • 执行__init__ 方法,为对象添加属性并设置初始值,属性又称数据成员
  • __init__ 方法内部,使用self.属性 = 形参接收外部传递的参数
  • 在方法中使用self.属性来引用属性值
tom = Cat("Tom")
tom.eat()
tom.drink()

小猫Tom爱吃鱼
小猫Tom在喝水
内置方法
方法名 类型 作用
__del__ 方法 对象被从内存中销毁前,会被 自动 调用
__str__ 方法 返回对象的描述信息print 函数输出使用
  • __del__方法
  • 当使用 类名() 创建对象时,自动调用 __init__ 方法
  • 对象被从内存中销毁前,自动调用 __del__ 方法
  • 对象的生命周期在调用 类名() 时开始,在调用 __del__ 方法后结束
  • 在对象生命周期内,可以访问对象属性,调用对象方法
  • __str__方法
  • 使用print输出对象变量时,默认输出对象内存地址
  • 利用__str__方法,可以自定义输出内容和格式
  • __str__方法必须返回一个字符串
class Cat:
    def __init__(self, name):
        self.name = name
    def __str__(self):
        return "我是小猫:%s" % self.name

tom = Cat("Tom")
print(tom)

我是小猫:Tom
实例演示:绘图板

需求:

  • 在400X300的绘图纸上,可以绘制点、直线和圆
  • 使用类来抽象上述概念
  • 设计类
class Pointer:
    def __init__(self, x, y):
        self.x = x
        self.y = y
    def __str__(self):
        return 'point(%d,%d)' % (self.x, self.y)

class Line:
    def __init__(self, p1, p2):
        self.begin = p1
        self.end = p2
    def __str__(self):
        return 'line((%d,%d), (%d, %d))' % (self.begin.x, self.begin.y
                                            , self.end.x, self.end.y)

class Circle:
    def __init__(self, r, point):
        self.r = r
        self.o = point
    def __str__(self):
        return 'r:%d, o:(%d, %d))' % (self.r, self.o.x, self.o.y)
实例演示:我爱我家

需求:

  • 房子(House):有 户型总面积家具名称列表
    新房子没有任何的家具
  • 家具(HouseItem):有 名字占地面积,其中,
    席梦思(bed) 占地 4 平米
    衣柜(chest) 占地 2 平米
    餐桌(table) 占地 1.5 平米
  • 将以上三件家具 添加 到房子中
  • 打印房子时,要求输出:户型、总面积、剩余面积、家具名称列表
  • 设计类图
  • 设计属性和方法
  • 在房子类中定义一个剩余面积属性,其初始值和总面积相等
  • 定义add_item方法用来向房子中添加家具,添加家具后让
    剩余面积 = 剩余面积 - 家具面积
  • 创建家具类
class HouseItem:
    def __init__(self, name, area):
        self.name = name
        self.area = area
    def __str__(self):
        return "[%s] 占地面积 %.2f" % (self.name, self.area)

bed = HouseItem("席梦思", 4)
chest = HouseItem("衣柜", 2)
table = HouseItem("餐桌", 1.5)

print(bed)
print(chest)
print(table)

[席梦思] 占地面积 4.00
[衣柜] 占地面积 2.00
[餐桌] 占地面积 1.50
  • 创建房子类
class House:
    def __init__(self, house_type, area):
        self.house_type = house_type
        self.area = area
        self.free_area = area
        self.item_list = []
    def __str__(self):
        return ("户型:%s\n总面积:%.2f[剩余:%.2f]\n家具:%s"
                % (self.house_type, self.area,
                   self.free_area, self.item_list))
    def add_item(self, item):
        print("要添加 %s" % item)

my_home = House("两室一厅", 60)
my_home.add_item(bed)
my_home.add_item(chest)
my_home.add_item(table)
print(my_home)

要添加 [席梦思] 占地面积 4.00
要添加 [衣柜] 占地面积 2.00
要添加 [餐桌] 占地面积 1.50
户型:两室一厅
总面积:60.00[剩余:60.00]
家具:[]
  • 实现add_item方法
  • 判断家具的面积是否超过剩余面积,如果超过提示不能添加这件家具
  • 将家具名称追加到家具名称列表中
  • 用房子剩余面积 - 家具面积
  def add_item(self, item):
    print("要添加 %s" % item)
    if item.area > self.free_area:
        print("%s 的面积太大,不能添加到房子中" % item.name)
        return
    self.item_list.append(item.name)
    self.free_area -= item.area
实例演示:打砖块

需求

  • 800X400的屏幕上,有2个大小不等的圆
  • 判断2个圆是否发生了碰撞
import math
class Pointer:
    def __init__(self, x, y):
        self.x = x
        self.y = y
    def __str__(self):
        return 'point(%d,%d)' % (self.x, self.y)

class Circle:
    def __init__(self, r, point):
        self.r = r
        self.o = point
    def __str__(self):
        return 'r:%d, o:(%d, %d))' % (self.r, self.o.x, self.o.y)
    def func(self, a):
        a.func()

if __name__ == '__main__':
    c1 = Circle(40, Pointer(100, 100))
    c2 = Circle(50, Pointer(160, 160))
    x = c1.o.x - c2.o.x
    y = c1.o.y - c2.o.y
    z = math.sqrt(x ** 2 + y ** 2)  # z = (x**2 + y**2)**0.5
    if z <= c1.r + c2.r:
        print('发生碰撞')
    else:
        print('没有碰撞')
实例演示:士兵突击

需求

  • 士兵许三多有一把AK47
  • 士兵可以开火
  • 能够发射子弹
  • 装填子弹 — 增加子弹数量
  • 设计枪类

属性

  • model:型号
  • bullet_count:子弹数量

shoot方法

  • 判断是否有子弹,没有子弹无法射击
  • 使用 print 提示射击,并且输出子弹数量
class Gun:
    def __init__(self, model):
        self.model = model  # 枪的型号        
        self.bullet_count = 0  # 子弹数量
    def add_bullet(self, count):
        self.bullet_count += count
    def shoot(self):
        if self.bullet_count <= 0:
            print("没有子弹了...")
            return
        self.bullet_count -= 1        
        print("%s 发射子弹[%d]..." % (self.model, self.bullet_count))

ak47 = Gun("ak47")
ak47.add_bullet(50)
ak47.shoot()
  • 设计士兵类

属性

  • name:姓名
  • gun:枪,每个新兵都没有枪,初始值设置为None

fire 方法

  • 判断是否有枪,没有枪没法冲锋
  • 喊一声口号
  • 装填子弹
  • 射击
class Soldier:
    def __init__(self, name):
        self.name = name
        self.gun = None
    def fire(self):
        if self.gun is None:  # 判断是否有枪
            print("[%s] 还没有枪..." % self.name)
            return
        print("冲啊...[%s]" % self.name)  # 喊一声口号
        self.gun.add_bullet(50)  # 装填子弹
        self.gun.shoot()  # 射击
身份运算符
  • 身份运算符用于比较两个对象的内存地址是否一致,即是否指向同一对象
  • 在 Python 中针对None比较时,建议使用is判断
运算符 描述 实例
is 判断两个变量是否指向同一对象 x is y,类似id(x) == id(y)
is not 判断两个变量是否指向不同对象 x is not y,类似id(a) != id(b)

is 与 == 区别:

  • is用于判断两个变量引用的对象是否为同一个
  • ==用于判断两个变量引用的是否相等
>>> a = [1, 2, 3]
>>> b = [1, 2, 3]
>>> b is a 
False
>>> b == a
True




- end -

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,928评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,192评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,468评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,186评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,295评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,374评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,403评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,186评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,610评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,906评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,075评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,755评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,393评论 3 320
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,079评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,313评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,934评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,963评论 2 351

推荐阅读更多精彩内容

  • 国家电网公司企业标准(Q/GDW)- 面向对象的用电信息数据交换协议 - 报批稿:20170802 前言: 排版 ...
    庭说阅读 10,937评论 6 13
  • 面向对象和面向过程的基本概念 面向过程:侧重于怎么做。1、把完成一个需求的所有步骤 从头到尾 逐步实现2、根据开...
    ivan_cq阅读 1,170评论 0 0
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,094评论 1 32
  • 秋风凉爽,正是出行的好时机。 他坐在公交车的站牌下面。 站牌上写着这一站通过2路车、4路车、6路车和13路车。他要...
    唐老头子阅读 187评论 0 0
  • 一、印象深刻的三件事 ①、又回到四楼以前的教室上课 ②、我们小组两位同学的精彩演讲 ③、能力分好多种 二、为...
    离雨_阅读 210评论 0 0