場景
餐飲店的賬單由多個(gè)部分組成唠摹,需要為每一桌客人計(jì)算賬單總價(jià)盏档。王叔叔要給這個(gè)店做一個(gè)結(jié)賬系統(tǒng)捞稿,由于這很簡單,于是他上來就設(shè)計(jì)一個(gè)類來計(jì)算總價(jià),每計(jì)算一桌就新建一個(gè)對象弧轧。
小王覺得這個(gè)方案有問題,他就提出了幾點(diǎn)想法:
- 每次計(jì)算都需要實(shí)例化碗殷,計(jì)算次數(shù)越多劣针,創(chuàng)建的實(shí)例就越多。造成了效率低亿扁,內(nèi)存占用高的結(jié)果
- 我們需要自己創(chuàng)建對象和銷毀對象捺典。
考慮到這些問題,小王重新設(shè)計(jì)一個(gè)類來計(jì)算總價(jià)从祝,實(shí)例化這個(gè)類襟己,每次都只使用同一個(gè)對象。
- 私有變量牍陌,引用對象實(shí)例
- 一個(gè)公共方法擎浴,實(shí)例化對象或者返回已有的對象
- 繞過靜態(tài)構(gòu)造函數(shù)機(jī)制。
以下是偽代碼:
class CalculatePayment:
drink = None
meal = None
dessert = None
fruit = None
__payment_instance = None
def get_payment_instance(self):
if self.__payment_instance:
return self.__payment_instance
else:
payment = CalculatePayment()
self.__payment_instance = payment
def set_bill(self, drink, meal, dessert, fruit):
self.drink = drink
self.meal = meal
self.fruit = fruit
self.dessert = dessert
def _calculate_drink_bill(self):
if not self.drink:
return 0
drink_price_list = price_list['drink']
payment = drink_price_list[self.drink]
return payment
def _calculate_meal_bill(self):
if not self.meal:
return 0
meal_price_list = price_list['meal']
payment = meal_price_list[self.meal]
return payment
def _calculate_dessert_bill(self):
if not self.dessert:
return 0
dessert_price_list = price_list['dessert']
payment = dessert_price_list[self.dessert]
return payment
def _calculate_fruit_bill(self):
if not self.fruit:
return 0
fruit_price_list = price_list['fruit']
payment = fruit_price_list[self.fruit]
return payment
def total_payment(self):
fruit_payment = self._calculate_fruit_bill()
meal_payment = self._calculate_meal_bill()
drink_payment = self._calculate_drink_bill()
dessert_payment = self._calculate_dessert_bill()
return fruit_payment + meal_payment + drink_payment + dessert_payment
for bill in bill_list:
calculator = CalculatePayment.get_payment_instance()
calculator.set_bill(bill['drink'], bill['meal'], bill['fruit'], bill['dessert'])
bill['amount'] = calculator.total_payment()
總結(jié)
意圖:希望對象只有一個(gè)實(shí)例毒涧,但沒有控制對象實(shí)例化的的全局對象贮预。還希望確保所有實(shí)體使用該對象相同的實(shí)例,而無需將引用傳給它們。
問題:多個(gè)不同客戶對象要引用同一個(gè)對象仿吞,希望這個(gè)對象的實(shí)例只有一個(gè)滑频。
解決方案:確保只有一個(gè)實(shí)例。
好處:
- 一次實(shí)例化唤冈,多次使用
- 不用關(guān)心該對象是不是已經(jīng)被實(shí)例化了峡迷,該對象自己負(fù)責(zé)。
- 不需要考慮對實(shí)例的銷毀你虹。
- 配合Adapter模式可以起到很好的效果