Python 定制类 特殊方法

1、特殊方法

定义在class中

不需要直接调用,python的某些函数或操作符会自动的调用对应的特殊方法。

如定义了person类,使用print p 语句打印person类的实例时,就调用了特殊方法__str__()

此时就需要在person类中实现这个方法。

使用特殊方法时注意:

只需要编写用到的特殊方法

有关联性的特殊方法都必须实现(如__getattr__,__setattr__,delattr__)

2、python中 __str__和__repr__

__str__()用于显示给用户,而__repr__()用于显示给开发人员。

class Person(object):
    def __init__(self, name, gender):
        self.name = name
        self.gender = gender
    def __str__(self):
        return ‘(Person: %s, %s)‘ % (self.name, self.gender)
    __repr__ = __str__ #直接让repr和str相同

定义了repr函数后,在调试时直接向命令行输入p即可输出person的信息,否则会输出

<__main__.Person object at 0x0000000002E66C88>

__str__和__repr__函数会被子类继承。

3、python中 __cmp__

为了对对象进行排序,可以使用函数sorted函数,前提是设置了类的__cmp__方法。

class Student(object):
    def __init__(self, name, score):
        self.name = name
        self.score = score
    def __str__(self):
        return ‘(%s: %s)‘ % (self.name, self.score)
    __repr__ = __str__

    def __cmp__(self, s):
        if self.name < s.name:
            return -1
        elif self.name > s.name:
            return 1
        else:
            return 0
 
>>> L = [Student(‘Tim‘, 99), Student(‘Bob‘, 88), Student(‘Alice‘, 77)]
>>> print sorted(L)
[(Alice: 77), (Bob: 88), (Tim: 99)]
 

4、python中 __len__

调用len()返回实例的长度.

class Students(object):
    def __init__(self, *args):
        self.names = args
    def __len__(self):
        return len(self.names)
 
>>> ss = Students(‘Bob‘, ‘Alice‘, ‘Tim‘)
>>> print len(ss)
3
任务:

斐波那契数列是由 0, 1, 1, 2, 3, 5, 8...构成。

请编写一个Fib类,Fib(10)表示数列的前10个元素,print Fib(10) 可以打印出数列的前 10 个元素,len(Fib(10))可以正确返回数列的个数10。

代码1:(参考代码)

可以使用str()将整型列表转化为字符串。

class Fib(object):

def __init__(self, num):

a, b, L = 0, 1, []

for n in range(num):

L.append(a)

a, b = b, a + b

self.numbers = L

def __str__(self):

return str(self.numbers)

__repr__ = __str__

def __len__(self):

return len(self.numbers)

f = Fib(10)

print f

print len(f)

代码2:(自己写的)

class Fib(object):

    def __init__(self, num):
        self.num = num

    def __str__(self):
        if(self.num==1):
            return "[0]"
        elif(self.num==2):
            return "[0,1]"
        else:
            fib_str = "[0,"
            fib=[0,1]
            for k in range(2,self.num+1):
                fib_str = fib_str+ str(fib[k-1]) + ","
                fib.append(fib[k-1]+fib[k-2])
            return fib_str[:-1]+‘]‘
    def __len__(self):
        return self.num   

f = Fib(10)
print f
print len(f)

5、python中数学运算(运算符重载)

例:有理数可以用p/q表示,其中p和q都是整数。

调用求最大公约数的gcd函数(递归):

欧几里得定理:若a=b*r+q,则 GCD(a,b)=GCD(b,q)

def gcd(a,b):
    if(b==0):
        return a
    return gcd(b,a%b)
class Rational(object):
    def __init__(self, p, q):
        self.p = p
        self.q = q

    def __add__(self, r):
        return Rational(self.p * r.q + self.q * r.p, self.q * r.q)

    def __sub__(self, r):
        return Rational(self.p*r.q-r.p*self.q,self.q*r.q)

    def __mul__(self, r):
        return Rational(self.p*r.p,self.q*r.q)

    def __div__(self, r):
        return Rational(self.p*r.q,self.q*r.p)

    def __str__(self):
        g = gcd(self.p,self.q)
        return ‘%s/%s‘ % (self.p/g, self.q/g)

    __repr__ = __str__

r1 = Rational(1, 2)
r2 = Rational(1, 4)
print r1 + r2
print r1 - r2
print r1 * r2
print r1 / r2

6、python中类型转换

__int__ 和 __float__方法实现类型转换,
__str__ 方法也可以看做是一种类型转换。
class Rational(object):
    def __init__(self, p, q):
        self.p = p
        self.q = q

    def __int__(self):
        return self.p // self.q

    def __float__(self):
        return float(self.p)/float(self.q)

print float(Rational(7, 2))
print float(Rational(1, 3))

7、python中 @property(装饰器函改写set和get方法)

@property 修饰方法后即是get方法。

“@+方法名字+点+setter”是个固定格式与@property搭配使用,set方法。

如果没有定义set方法,就不能对“属性”赋值,这时,就可以创建一个只读“属性”。

 
class Student(object):
    def __init__(self, name, score):
        self.name = name
        self.__score = score
    @property
    def score(self):
        return self.__score
    @score.setter
    def score(self, score):
        if score < 0 or score > 100:
            raise ValueError(‘invalid score‘)
        self.__score = score

8、python中 __slots__ 属性限制

__slots__规定了一个类允许的属性列表。不允许添加列表外的属性。

class Student(object):
    __slots__ = (‘name‘, ‘gender‘, ‘score‘)
    def __init__(self, name, gender, score):
        self.name = name
        self.gender = gender
        self.score = score
强行添加grade属性会报错。
>>> s = Student(‘Bob‘, ‘male‘, 59)
>>> s.name = ‘Tim‘ # OK
>>> s.score = 99 # OK
>>> s.grade = ‘A‘
Traceback (most recent call last):
  ...
AttributeError: ‘Student‘ object has no attribute ‘grade‘

9、python中__call__  把类实例变成可调用对象

class Person(object):
    def __init__(self, name, gender):
        self.name = name
        self.gender = gender

    def __call__(self, friend):
        print ‘My name is %s...‘ % self.name
        print ‘My friend is %s...‘ % friend
 
>>> p = Person(‘Bob‘, ‘male‘)
>>> p(‘Tim‘)
My name is Bob...
My friend is Tim...
 
第四节的斐波那契数列问题,可以改用call实现。
class Fib(object):

    def __call__(self,num):
        a=0
        b=1
        L=[]

        for n in range(1,num+1):
            L.append(a)

            temp = b

            b = a + b
            a = temp

        return L

f = Fib()
print f(10)
 
参考代码:
class Fib(object):
    def __call__(self, num):
        a, b, L = 0, 1, []
        for n in range(num):
            L.append(a)
            a, b = b, a + b
        return L

f = Fib()
print f(10)

注意代码中 高亮语句,使用参考代码的写法,就省去了一个中间变量。

 
时间: 2024-10-27 02:32:25

Python 定制类 特殊方法的相关文章

python 定制类

看到类似__slots__这种形如__xxx__的变量或者函数名就要注意,这些在Python中是有特殊用途的. __slots__我们已经知道怎么用了,__len__()方法我们也知道是为了能让class作用于len()函数. 除此之外,Python的class中还有许多这样有特殊用途的函数,可以帮助我们定制类. __str__ 我们先定义一个Student类,打印一个实例: >>> class Student(object): ... def __init__(self, name):

python元编程之使用动态属性实现定制类--特殊方法__setattr__,__getattribute__篇

问题:实现一个类,要求行为如同namedtuple:只存在给定名称的属性,不允许动态添加实例属性. 主要知识点在于: __setattr__,__getattr__,getattribute__,__delattr__特殊方法的实现使用. 代码如下: 1 """ 2 运行环境 3 python 3.7+ 4 """ 5 from collections OrderedDict, namedtuple 6 #以下为要包装的对象:1个命名元组,用于存

python定制类(以Fib类为例)

class Fib(object): def __init__(self): self.a, self.b = 0, 1 def __iter__(self): return self def __next__(self): #如果一个类想被用于for ... in循环,类似list或tuple那样,就必须实现一个__iter__()方法,该方法返回一个迭代对象, # 然后,Python的for循环就会不断调用该迭代对象的__next__()方法拿到循环的下一个值,直到遇到StopIterati

Python元类__prepare__方法深入理解

学习元类的时候,对__prepare__不是很理解,书上讲解的也不是很详细,最后通过查看stackoverflow的一些帖子对该方法有了一些理解,记录如下: 先看代码: class member_table(dict): def __init__(self): self.member_names = [] def __setitem__(self, key, value): if key not in self: self.member_names.append(key) dict.__seti

Python定制类

1.__str__ class Student(object): def __init__(self, name): self._name = name m = Student("michael") print m <__main__.Student object at 0x0000000003280EF0> 为了使print类的结果格式更加方便阅读 重写__str__(self) class Student(object): def __init__(self, name

【Python】+类内部方法相互调用

1 class TestDate: 2 a = 1 3 4 # "__init__"为类的构造函数 5 def __init__(self): 6 self.a = 666 7 pass 8 9 def a_1(self): 10 print("a_1") 11 self.a_2() 12 13 def a_2(self): 14 print(self.a) 15 print("a_2") 16 17 18 if __name__ == &quo

一个关于python定制类的例子

class Time(object): def __init__(self, hr, min): self.hr = hr self.min = min def __str__(self): return '%d:%d' %(self.hr, self.min) __repr__ = __str__ # 重载加法 def __add__(self, other): m = self.min + other.min h = self.hr + other.hr if m>=60: m -= 60

python基本知识(八):定制类,双下划线方法

'''定制类: 1. 双下划线属性__attr__ 2. 元类metaclass''' # 综述 '''iterable/iterator:1. __iter__(): return iterable_obj 1) 实现了该方法的对象叫iterable 2) iter(obj)会调用该方法, 生成一个迭代器iterator 2. __next__(): 指明迭代器怎么返回值 1) next(iterator)会返回一个值, 直到所有的值都返回了报错StopIteration 2) for.. i

Python学习之定制类

本文和大家分享的主要是 python开发中定制类的相关内容,一起来看看吧,希望对大家学习和使用这部分内容有所帮助. 1. python中什么是特殊方法 任何数据类型的实例都有一个特殊方法:  __str__() ·  用于 print 的  __str__ ·  用于 len 的  __len__ ·  用于 cmp 的  __cmp__ ·  特殊方法定义在 class 中 ·  不需要直接调用 · Python 的某些函数或操作符会调用对应的特殊方法 file:///C:\Users\wlc