Python高效编程(五)

实际编程和面试都会遇到的典型问题。

图片来源于网络

如何派生内置不可变类型并修改其实例化行为

#我们想自定义一种新类型元组,对于传入的可迭代对象,我们只想保留其中int类型>0的元素 新类型是内置tuple的子类

class IntTuple(tuple):
    def __new__(cls,iterable):
        # 使用生成器过滤
        g = (x for x in iterable if isinstance(x,int) and x > 0)
        return super(IntTuple,cls).__new__(cls,g)
        
    def __init__(self,iterable):
        super(IntTuple,self).__init__()
        
it = IntTuple([1,-2,3,'x'])    
it

如何为创建大量实例节省内存

class Player(object):
    def __init__(self,uid,name,status=0,level=1):
        self.uid = uid
        self.name = name
        self.status = status
        self.level = level
        
class Player2(object):
    __slots__ = ['uid','name','status','level']
    def __init__(self,uid,name,status=0,level=1):
        self.uid = uid
        self.name = name
        self.status = status
        self.level = level       

p1 = Player('001','uu')
p2 = Player2('001','uu')
set(dir(p1)) - set(dir(p2))
# p1比p2多了{'__dict__', '__weakref__'}
# '__dict__'可以动态绑定
p1.x = 123
del p1.__dict__['x']

import sys
# 占用了320内存
sys.getsizeof(p1.__dict__)

# p2事先定义__slots__ 声明了实例属性名字的列表
# p2就无法动态绑定 从而节省了内存
# p2.x = 123

如何创建可管理的对象属性

from math import pi

class Cricle(object):
    def __init__(self,radius):
        self._radius = radius
    
    
    @property
    def radius(self):
        return self._radius
    
    
    @radius.setter
    def radius(self,value):
        if not isinstance(value,(int,float)):
            raise ValueError('wrong type')
        self._radius = value
    
    
    def getArea(self):
        return self._radius ** 2 * pi

c = Cricle(5)
c.radius

如何让类支持比较操作

from math import pi
from functools import total_ordering

@total_ordering
class Cricle(object):
    def __init__(self,radius):
        self._radius = radius
    
    def area(self):
        return self._radius ** 2 * pi
    
    def __lt__(self,obj):
        return self.area() < obj.area()
    
    def __eq__(self,obj):
        return self.area() == obj.area()
    
c1 = Cricle(3)
c2 = Cricle(5.0)
c1 > c2

如何使用描述符对实例属性做类型检查

# 描述符 __get__ __set__ __del__

class Attr(object):
    def __init__(self,name,type_):
        self.name = name
        self.type_ = type_
    
    def __get__(self,instance,cls):
        return instance.__dict__[self.name]
    
    def __set__(self,instance,value):
        if not isinstance(value,self.type_):
            raise TypeError('expected an %s'%self.type_)
        instance.__dict__[self.name] = value
    
    def __delete__(self,instance):
        del instance.__dict__[self.name]
        

class Person(object):
    name = Attr('name',str)
    age = Attr('age',int)
    height = Attr('height',float)
    
p = Person()
p.name = '123'
p.age = 123
p.height = 1.0

如何通过实例方法名字的字符串调用方法

class Circle(object):
    def __init__(self,r):
        self.r = r
    
    def area(self):
        return self.r ** 2 * 3.14

class Rectangle(object):
    def __init__(self,w,h):
        self.w = w
        self.h = h
    
    def get_area(self):
        return self.w * self.h
    
class Triangle(object):
    def __init__(self,a,b,c):
        self.a = a
        self.b = b
        self.c = c
        
    def getArea(self):
        #海伦公式:√[p(p-a)(p-b)(p-c) ]其中p=1/2(a+b+c)
        p = (self.a + self.b + self.c) / 2
        return (p * (p - self.a) * (p - self.b) * (p - self.c)) ** 0.5
    

def getArea(shape):
    for name in ('area','get_area','getArea'):
        f = getattr(shape,name,None)
        if f:
            return f()
        
        
shape1 = Circle(3)
shape2 = Rectangle(5,6)
shape3 = Triangle(3,4,5)

shapes = [shape1,shape2,shape3]
print(list(map(getArea,shapes)))

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏大内老A

关于CLR内存管理一些深层次的讨论[下篇]

《上篇》中我们主要讨论的是程序集(Assembly)和应用程序域(AppDomain)的话题,着重介绍了两个不同的程序集加载方式——独占方式和共享方式(中立域方...

1999
来自专栏光变

3.1 ASM-方法-结构

ASM-方法-结构 本章将会介绍如果使用ASM core API生成或者转换Java编译后的method。 本将开始会展示编译后的method,然后使用很多说...

1222
来自专栏me的随笔

Python知识梳理

我们可以使用type()函数类获取对象的类型,Python3中内置数据类型包括:None,int,float,complex,str,list,dict,tup...

1212
来自专栏ImportSource

Java8真不用再搞循环了?

Java8以后真的不用循环了?真的不用了? 好吧,本文分享的内容是java8之前和java8之后一些代码的不同写法,我们会先介绍java8之前和java8之后不...

1.9K11
来自专栏阿杜的世界

Java虚拟机规范阅读笔记—第2章

在发生一个方法调用的时候会创建一个栈帧,在方法执行结束(无论是正常结束还是抛出异常)的时候栈帧会被销毁。栈帧在JVM的栈上分配,栈属于某个线程。每个栈帧都有自己...

922
来自专栏calvin

【nodejs】让nodejs像后端mvc框架(asp.net mvc)一样处理请求--参数自动映射篇(6/8)

路由、action的扫描、发现、注册搞定之后,后来我发现在我们的action里面获取参数往往都是通过request对象来一个一个获取。同样的一行代码我们不厌其烦...

1022
来自专栏维C果糖

Guava 指南 之「前置条件」

前置条件 Guava 提供了很多用于进行前置条件检查的工具,我们强烈建议静态导入这些方法。 每个方法都用三种形式: 没有额外的参数。抛出的任何异常都没有错误信息...

1987
来自专栏积累沉淀

Java设计模式(二十一)----访问者模式

访问者模式 一、概述 1、定义 2、分派的概念 3、分派的类型 4、双重分派 二、结...

20710
来自专栏Python爬虫与算法进阶

学点算法之栈的学习与应用

在学习栈前,脑海中对这个词只有一个印象:客栈 栈是什么 栈(有时称为“后进先出栈”)是一个项的有序集合,其中添加移除新项总发生在同一端。 这段话初学者是懵逼的...

3073
来自专栏软件开发 -- 分享 互助 成长

为什么无返回值的链表的插入操作头结点一定要用指向指针的指针

前言: 为什么链表的插入操作头结点一定要用指向指针的指针?之前自己对这个问题总是一知半解,今天终于花了点时间彻底搞懂了。 总的来说这样做的目的是为了应对“空链表...

2057

扫码关注云+社区