__dict__
方法遇到這樣一個狀況,要打印出一個對象的各類屬性。可是不一樣對象的屬性名都不同,結構也不一樣,沒法寫一個代碼來實現。而後我找到了__dict__
,使用這個屬性,能夠動態獲取到對象的全部屬性,不包括公用屬性。python
class Teacher(object): display = "教師" # 有公有屬性 def __init__(self,name,age,course,salary): self.name = name self.age = age self.course = course self.__salary = salary # 也有私有屬性 # 在定義一個別的類 class Student(object): display = "學生" def __init__(self,name,sid,classes,score): self.name = name self.sid = sid self.calssed = classes self.__score = score def print_obj(obj): "打印對象的全部屬性" print(obj.__dict__) t1 = Teacher("Jerry",36,"Python",20000) s1 = Student('Barry',12,"python01","B") print_obj(t1) print_obj(s1)
經過__dict__
,就能夠動態的獲取到對象的所有屬性。得到的是一個字典,屬性名是字典的key,屬性值是字典的value。從輸出看,私有屬性也能夠得到,只是不包括公有屬性。
而後,若是隻想要屬性值的話,能夠對字典再加工。複習一下字典的3個方法:ide
dict.items()
:用元祖來存放key和vlauedict.keys()
: 只包含keydict.values()
: 只包含value上面3個方法返回的都是一個可迭代對象,能夠用for遍歷,但不是迭代器,不能用next方法。
而後用下面的方法打印code
# 直接用字典了 dict1 = {'name': 'Jerry', 'age': 36, 'course': 'Python', '_Teacher__salary': 20000} # 先看一下3個方法返回的可迭代對象 print(dict1.items()) print(dict1.keys()) print(dict1.values()) # 用下面的方法輸出 print('\n'.join(('%s:%s' % item for item in dict1.items()))) # 每行一對key和value,中間是分號 print(' '.join(('%s' % item for item in dict1.keys()))) # 全部的key打印一行 print(' '.join(('%s' % item for item in dict1.values()))) # 全部的value打印一行
最後拆分一下打印的時候用到的方法a = dict1.items()
這個是可迭代對象,能夠用for遍歷b = ("%s:%s"%item for item in a)
用for循環遍歷a,每一項是個元祖,把元祖轉成"%s:%s"的字符串形式。最外面的( )就是轉成一個迭代器。也能夠用[ ],轉成列表。c = ‘\n’.join(b)
最後用join()方法完成字符串的拼接對象
__str__
方法又發現一個更好用的方法,而且能夠獲取到公有屬性了。__str__
方法是在打印這個對象的時候,再也不打印對象的內存地址,而是打印__str__
方法的返回值:內存
class Teacher(object): display = "教師" # 有公有屬性 def __init__(self,name,age,course,salary): self.name = name self.age = age self.course = course self.__salary = salary # 也有私有屬性 def __str__(self): # 定義打印對象時打印的字符串 return " ".join(str(item) for item in ( self.display,self.name,self.age,self.course,self.__salary)) class Student(object): display = "學生" def __init__(self,name,sid,classes,score): self.name = name self.sid = sid self.calssed = classes self.__score = score def __str__(self): # 其實通常可能都是這樣簡單用一下的 return self.name t1 = Teacher("Jerry",36,"Python",20000) s1 = Student('Barry',12,"python01","B") print(t1) print(s1)
這裏要注意,返回值必須是字符串,因此得傳一個數據類型return " ".join(str(item) for item in (self.display,self.name,self.age,self.course,self.__salary))
這個也能夠這麼寫return "%s %s %s %s %s"%(self.display,self.name,self.age,self.course,self.__salary)
這麼寫雖然好理解,可是前面的%s的數量必須和後面的變量一致,若是要加1個或減1個變量,先後都得改。__str__
方法能夠徹底自定義本身對象的輸出格式,既然是自定義的方法,那麼還能夠加上參數控制。可是調用的時候彷佛並無地方填參數。
實際上是在print調用對象的時候,系統已經幫咱們自動將print指向了__str__
方法,也就是說 print(t1)
其實執行的是 print(t1.__str__())
,這個時候咱們就能夠本身寫全,而後加上參數。字符串
class Teacher(object): display = "教師" # 有公有屬性 def __init__(self,name,age,course,salary): self.name = name self.age = age self.course = course self.__salary = salary # 也有私有屬性 def __str__(self,print_all=False): # 定義打印對象時打印的字符串 if print_all: return " ".join(str(item) for item in ( self.display,self.name,self.age,self.course,self.__salary)) else: return self.name class Student(object): display = "學生" def __init__(self,name,sid,classes,score): self.name = name self.sid = sid self.calssed = classes self.__score = score def __str__(self): # 其實通常可能都是這樣簡單用一下的 return self.name t1 = Teacher("Jerry",36,"Python",20000) s1 = Student('Barry',12,"python01","B") print(t1) print(s1) print(t1.__str__()) # 這個和上面的效果是同樣的 print(t1.__str__(True)) # 如今能夠帶上參數了
其實這裏並沒不是打印了全部的屬性,而是咱們自定義了打印內容。可是自定義的位置是在類中的,這個位置是能夠獲取到所有屬性的。it
其實就是把上面2個方法一塊兒用。其實有上面2個方法應該就能夠了,不過既然都搞明白了,留個記錄也好。
先提一個點,在定義了__str__
方法後,雖然打印出來是字符串,可是在其餘時候傳的值仍是對象。若是想獲取就是打印的值而不是對象,那麼仍是用對象__str__()
來傳遞,下面就是最終的例子:for循環
class Teacher(object): display = "教師" # 有公有屬性 def __init__(self,name,age,course,salary): self.name = name self.age = age self.course = course self.__salary = salary # 也有私有屬性 def __str__(self,print_all=False): # 定義打印對象時打印的字符串 if print_all: return ' '.join(('%s' % item for item in self.__dict__.values())) else: return self.name t1 = Teacher("Jerry",36,"Python",20000) print(t1) print(t1.__str__()) print(t1.__str__(True)) t1_obj = t1 # 雖然print的時候打印的是字符串,可是別的時候仍是會把對象傳過去的 print(t1_obj,type(t1_obj)) # 直接打印看不出來,可是能夠看看數據類型 print(t1_obj.name,t1_obj.age) # 確實內取到對象裏的屬性 t1_str = t1.__str__() # 要傳字符串,還差直接用__str__來獲取 print(t1_str,type(t1_str)) # 這裏獲取到的就是字符串類型了