python重寫類
① python類的繼承和多態代碼詳解
Python類的繼承和多態代碼詳解
Python類的繼承
在OOP(ObjectOrientedProgramming)程序設計中,當我們定義一個class的時候,可以從某個現有的class繼承,新的class稱為子類(Subclass),而被繼承的class稱為基類、父類或超類(Baseclass、Superclass)。
我們先來定義一個classPerson,表示人,定義屬性變數name及sex(姓名和性別);
定義一個方法print_title():當sex是male時,printman;當sex是female時,printwoman。參考如下代碼:
class Person(object):
def __init__(self,name,sex):
self.name = name
self.sex = sex
def print_title(self):
if self.sex == "male":
print("man")
elif self.sex == "female":
print("woman")
class Child(Person): # Child 繼承 Person
pass
May = Child("May","female")
Peter = Person("Peter","male")
print(May.name,May.sex,Peter.name,Peter.sex) # 子類繼承父類方法及屬性
May.print_title()
Peter.print_title()
而我們編寫Child類,完全可以繼承Person類(Child就是Person);使用classsubclass_name(baseclass_name)來表示繼承;
繼承有什麼好處?最大的好處是子類獲得了父類的全部屬性及功能。如下Child類就可以直接使用父類的print_title()方法
實例化Child的時候,子類繼承了父類的構造函數,就需要提供父類Person要求的兩個屬性變數name及sex:
在繼承關系中,如果一個實例的數據類型是某個子類,那它也可以被看做是父類(May既是Child又是Person)。但是,反過來就不行(Peter僅是Person,而不是Child)。
繼承還可以一級一級地繼承下來,就好比從爺爺到爸爸、再到兒子這樣的關系。而任何類,最終都可以追溯到根類object,這些繼承關系看上去就像一顆倒著的樹。比如如下的繼承樹:
isinstance()及issubclass()
Python與其他語言不同點在於,當我們定義一個class的時候,我們實際上就定義了一種數據類型。我們定義的數據類型和Python自帶的數據類型,比如str、list、dict沒什麼兩樣。
Python有兩個判斷繼承的函數:isinstance()用於檢查實例類型;issubclass()用於檢查類繼承。參見下方示例:
class Person(object):
pass
class Child(Person): # Child 繼承 Person
pass
May = Child()
Peter = Person()
print(isinstance(May,Child)) # True
print(isinstance(May,Person)) # True
print(isinstance(Peter,Child)) # False
print(isinstance(Peter,Person)) # True
print(issubclass(Child,Person)) # True
Python類的多態
在說明多態是什麼之前,我們在Child類中重寫print_title()方法:若為male,printboy;若為female,printgirl
class Person(object):
def __init__(self,name,sex):
self.name = name
self.sex = sex
def print_title(self):
if self.sex == "male":
print("man")
elif self.sex == "female":
print("woman")
class Child(Person): # Child 繼承 Person
def print_title(self):
if self.sex == "male":
print("boy")
elif self.sex == "female":
print("girl")
May = Child("May","female")
Peter = Person("Peter","male")
print(May.name,May.sex,Peter.name,Peter.sex)
May.print_title()
Peter.print_title()
當子類和父類都存在相同的print_title()方法時,子類的print_title()覆蓋了父類的print_title(),在代碼運行時,會調用子類的print_title()
這樣,我們就獲得了繼承的另一個好處:多態。
多態的好處就是,當我們需要傳入更多的子類,例如新增Teenagers、Grownups等時,我們只需要繼承Person類型就可以了,而print_title()方法既可以直不重寫(即使用Person的),也可以重寫一個特有的。這就是多態的意思。調用方只管調用,不管細節,而當我們新增一種Person的子類時,只要確保新方法編寫正確,而不用管原來的代碼。這就是著名的「開閉」原則:
對擴展開放(Openforextension):允許子類重寫方法函數
對修改封閉(Closedformodification):不重寫,直接繼承父類方法函數
子類重寫構造函數
子類可以沒有構造函數,表示同父類構造一致;子類也可重寫構造函數;現在,我們需要在子類Child中新增兩個屬性變數:mother和father,我們可以構造如下(建議子類調用父類的構造方法,參見後續代碼):
class Person(object):
def __init__(self,name,sex):
self.name = name
self.sex = sex
class Child(Person): # Child 繼承 Person
def __init__(self,name,sex,mother,father):
self.name = name
self.sex = sex
self.mother = mother
self.father = father
May = Child("May","female","April","June")
print(May.name,May.sex,May.mother,May.father)
Person
若父類構造函數包含很多屬性,子類僅需新增1、2個,會有不少冗餘的代碼,這邊,子類可對父類的構造方法進行調用,參考如下:
class Person(object):
def __init__(self,name,sex):
self.name = name
self.sex = sex
class Child(Person): # Child 繼承 Person
def __init__(self,name,sex,mother,father):
Person.__init__(self,name,sex) # 子類對父類的構造方法的調用
self.mother = mother
self.father = father
May = Child("May","female","April","June")
print(May.name,May.sex,May.mother,May.father)
多重繼承
多重繼承的概念應該比較好理解,比如現在需要新建一個類baby繼承Child,可繼承父類及父類上層類的屬性及方法,優先使用層類近的方法,代碼參考如下:
class Person(object):
def __init__(self,name,sex):
self.name = name
self.sex = sex
def print_title(self):
if self.sex == "male":
print("man")
elif self.sex == "female":
print("woman")
class Child(Person):
pass
class Baby(Child):
pass
May = Baby("May","female") # 繼承上上層父類的屬性
print(May.name,May.sex)
May.print_title() # 可使用上上層父類的方法
class Child(Person):
def print_title(self):
if self.sex == "male":
print("boy")
elif self.sex == "female":
print("girl")
class Baby(Child):
pass
May = Baby("May","female")
May.print_title() # 優先使用上層類的方法
總結
以上就是本文關於Python類的繼承和多態代碼詳解的全部內容,希望對大家有所幫助。
② python中重寫==的s方法
Python中對應於虛擬類的概念是抽象基類(ABC),一般類的話沒有必須重寫的東西。
③ 簡述python面向對象編程中函數重載和重寫的區別
這個基本是沒有一點關聯。。。只是名字容易混淆而已 重寫就是對父類的方法重寫,改變方法體中的語句。。。。 重載就是同一個函數名,參數個數、類型、排列順序不同,jvm根據參數來決定調用哪一個方法
④ python怎麼重寫集合方法
class Set(object):
def __init__(self,data=None):
if data == None:
self.__data = []
else:
if not hasattr(data,'__iter__'):
#提供的數據不可以迭代,實例化失敗
raise Exception('必須提供可迭代的數據類型')
temp = []
for item in data:
#集合中的元素必須是可哈希
hash(item)
if not item in temp:
temp.append(item)
self.__data = temp
#析構函數
def __del__(self):
del self.__data
#添加元素,要求元素必須可哈希
def add(self, other):
hash(other)
if other not in self.__data:
self.__data.append(other)
else:
print('元素已存在,操作被忽略')
#刪除元素
def remove(self,other):
if other in self.__data:
self.__data.remove(other)
print('刪除成功')
else:
print('元素不存在,刪除操作被忽略')
#隨機彈出並返回一個元素
def pop(self):
if not self.__dat:
print('集合已空,彈出操作被忽略')
return
import random
item = random.choice(self.__data)
self.__data.remove(item)
return item
#運算符重載,集合差集運算
def __sub__(self, other):
if not isinstance(other,Set):
raise Exception('類型錯誤')
#空集合
result = Set()
#如果一個元素屬於當前集合而不屬於另一個集合,添加
for item in self.__data:
if item not in other.__data:
result.__data.append(item)
return result
#提供方法,集合差集運算,復用上面的代碼
def difference(self,other):
return self - other
#|運算符重載,集合並集運算
def __or__(self, other):
if not isinstance(other,Set):
raise Exception('類型錯誤')
result = Set(self.__data)
for item in other.__data:
if item not in result.__data:
result.__data.append(item)
return result
#提供方法,集合並集運算
def union(self,otherSet):
return self | otherSet
#&運算符重載,集合交集運算
def __and__(self, other):
if not isinstance(other,Set):
raise Exception('類型錯誤')
result = Set()
for item in self.__data:
if item in other.__data:
result.__data.append(item)
return result
#^運算符重載,集合對稱差集
def __xor__(self, other):
return (self-other) | (other-self)
#提供方法,集合對稱差集運算
def symetric_difference(self,other):
return self ^ other
#==運算符重載,判斷兩個集合是否相等
def __eq__(self, other):
if not isinstance(other,Set):
raise Exception('類型錯誤')
if sorted(self.__data) == sorted(other.__data):
return True
return False
#>運算符重載,集合包含關系
def __gt__(self, other):
if not isinstance(other,Set):
raise Exception('類型錯誤')
if self != other:
flag1 = True
for item in self.__data:
if item not in other.__data:
#當前集合中有的元素不屬於另一個集合
flag1 = False
break
flag2 = True
for item in other.__data:
if item not in self.__data:
#另一集合中的元素不屬於當前集合
flag2 = False
break
if not flag1 and flag2:
return True
return False
#>=運算符重載,集合包含關系
def __ge__(self, other):
if not isinstance(other,Set):
raise Exception('類型錯誤')
return self == other or self > other
#提供方法,判斷當前集合是否為另一個集合的真子集
def issubset(self,other):
return self<other
#提供方法,判斷當前集合是否為另一集合的超集
def issuperset(self,other):
return self > other
#提供方法,清空集合所有元素
def clear(self):
while self.__data:
del self.__data[-1]
print('集合已清空')
#運算符重載,使得集合可迭代
def __iter__(self):
return iter(self.__data)
#運算符重載,支持in運算符
def __contains__(self, item):
return item in self.__data
#支持內置函數len()
def __len__(self):
return len(self.__data)
#直接查看該類對象時調用該函數
def __repr__(self):
return '{'+str(self.__data)[1:-1]+'}'
#使用print()函數輸出該類對象時調用該函數
__str__ = __repr__
⑤ Python自定義的類,為什麼需要重寫
首先,自定義的類在不繼承任何基類的情況下,也具有__str__屬性:
[python] view plain
class RoundFloatManual(object):
... def __init__(self, val):
... assert isinstance(val, float), \
... "Value must be a float!"
... self.value = round(val, 2)
rfm = RoundFloatManual(5.590464)
dir(rfm)
返回:
['__class__', '__delattr__', '__dict__', '__doc__', '__format__', '__getattribute__', '__hash__', '__init__', '__mole__', '__new__', '__rece__', '__rece_ex__', '__repr__', '__setattr__', '__sizeof__', '__str__', '__subclasshook__', '__weakref__', 'value']
__str__ 是 Python 類中的特殊方法,他的回傳值就是使用 str(x) 所得到的值, 而 print(x) 其實就等於是print(str(x)).其實再講細一點,當我們呼叫 str(x) 的時候其實是呼叫x.__str__()
也就是說我們可以這樣想像:
print(x) === print(str(x)) === print(x.__str__())
一般我們 自定義的類,__str__ 方法的回傳值是默認的字串,比如說: <__main__.Mylist object at 0x0071A470> 用以說明 namespace, class name 和位置.如果要改變__str__ 的回傳值,我們必須要覆寫他.
⑥ python類方法重寫
從父類繼承中的方法,如果不滿足程序的需求,就需要重寫。
方法重寫指的是在子類中自定義實現父類中的同名方法。
⑦ Python中類的定義規是什麼
類的概念:
類 Class: 用來描述具體相同的屬性和方法的對象的集合。定義了該集合中每個對象所共有的屬性和方法。對象是類的示例。
類變數:類變數在整個實例化的對象中是公用的。類變數定義在類中且在函數體之外。類變數通常不作為實例變數使用。
實例變數:定義在方法中的變數,只作用於當前實例的類。
數據成員:類變數或者實例變數用於處理類及其實例對象的相關數據。
方法:類中定義的函數。在類內部,使用 def 關鍵字來定義一個方法,與一般函數定義不同,類方法必須包含參數 self, 且為第一個參數,self 代表的是類的實例。
構造函數:即__init()__,特殊的方法,在對象創建的時候被自動調用。
析構函數:即__del()__,特殊的方法,在對象被銷毀時被自動調用。
實例化:創建一個類的實例,類的具體對象。就是將創建的類賦值給另一個變數。理解為賦值即可,a = class(),這個過程,就叫做實例化
對象:通過類定義的數據結構實例。對象包括兩個數據成員(類變數和實例變數)和方法。
繼承:即一個派生類(derived class)繼承基類(base class)的欄位和方法。繼承也允許把一個派生類的對象作為一個基類對象對待。例如,有這樣一個設計:一個Dog類型的對象派生自Animal類,這是模擬」是一個(is-a)」關系(例圖,Dog是一個Animal)。
方法重寫:如果從父類繼承的方法不能滿足子類的需求,可以對其 進行改寫,這個過程叫方法的覆蓋(override),也稱為方法的重寫。
————————————————
原文鏈接:https://blog.csdn.net/f156207495/article/details/81166252
網頁鏈接
⑧ 在Python中為什麼重寫__new__方法時,不需要聲明它是靜態方法
這個不需要糾結,就和你寫的 "if" 解釋器如何判斷它是關鍵字?一個意思。
這個是解釋器自己的邏輯,同樣__init__()方法和其他方法也定義也沒有區別,為什麼實例化先會調用它,也是一個意思。
通俗點講,你帶了幾個朋友回家,你要給家裡人介紹一下,其中有一個朋友是你們家親戚,那是不是就可以不用介紹了?
⑨ python怎麼給導入模塊的類重寫其中部分類方法啊
要重寫ca.py中Test類的test方法,可以這樣做:
from ca import *
def t(self,mu):self.b=mu
Test.test=t
當然也可以重寫__init__方法:
from ca import *
def __init__(self):self.a,self.b=0,0
Test.__init__=__init__
⑩ python 重載和重寫的區別
重載和重寫,這是兩個新概念,是兩個令我們容易混淆的概念。方法重載(overloading method)
是在一個類裡面,方法名字相同,而參數不同。返回類型呢?可以相同也可以不同。方法重寫(overiding method)
子類不想原封不動地繼承父類的方法,而是想作一定的修改,這就需要採用方法的重寫。方法重寫又稱方法覆蓋。方法重載是讓類以統一的方式處理不同類型數據的
一種手段。Java的方法重載,就是在類中可以創建多個方法,它們具有相同的名字,但具有不同的參數和不同的定義。調用方法時通過傳遞給它們的不同個數和
類型的參數來決定具體使用哪個方法,
這就是多態性。方法重寫:在Java中,子類可繼承父類中的方法,而不需要重新編寫相同的方法。但有時子類並不想原封不動地繼承父類的方法,而是想作一定
的修改,這就需要採用方法的重寫。方法重寫又稱方法覆蓋。
若子類中的方法與父類中的某一方法具有相同的方法名、返回類型和參數表,則新方法將覆蓋原有的方法。
如需父類中原有的方法,可使用super關鍵字,該關鍵字引用了當前類的父類重寫方法的規則:
參數列表必須完全與被重寫的方法的相同,否則不能稱其為重寫而是重載.
返回的類型必須一直與被重寫的方法的返回類型相同,否則不能稱其為重寫而是重載.訪問修飾符的限制一定要大於被重寫方法的訪問修飾符
(public>protected>default>private)重寫方法一定不能拋出新的檢查異常或者比被重寫方法申明更加寬
泛的檢查型異常.例如,父類的一個方法申明了一個檢查異常IOException,在重寫這個方法是就不能拋出Exception,只能拋出
IOException的子類異常,可以拋出非檢查異常.重載的規則:必須具有不同的參數列表;可以有不同的返回類型,只要參數列表不同就可以
了;可以有不同的訪問修飾符;可以拋出不同的異常;注意,Java的方法重載要求同名的方法必須有不同的參數表,僅有返回類型不同是不足以區分兩
個重載的方法。重寫方法只能存在於具有繼承關系中,重寫方法只能重寫父類非私有的方法。下面分別舉一個例子來說明方法重載:public class
TestOverLoad{ public static void main(String[] args) {Test test = new
Test(); test.print(null); } }class Test{ public void print(String
some){System.out.println("String version print"); } public void
print(Object some){ System.out.println("Object version print");
}}該程序輸出的結果是String version print。