国产乱子伦高清露脸对白-国产精品欧美久久久久天天影视-国产91视频一区-亚洲欧美日产综合在线网-黄视频网站在线看-国产欧美亚洲精品第1页-亚洲www在线-大学生女人三级在线播放-日本在线视频www鲁啊鲁-国产成人精品一区二区仙踪林-69精品欧美一区二区三区-成人欧美亚洲-日本污污网站-中国妞xxxhd露脸偷拍视频-国产精品aⅴ在线观看-精品中文字幕在线

極客小將

您現(xiàn)在的位置是:首頁 » python編程資訊

資訊內(nèi)容

單例模式中不同語言的不同實現(xiàn)

極客小將2021-01-11-
簡介今天python視頻教程欄目介紹單例模式中不同語言的不同實現(xiàn)。前言前段時間在用Python實現(xiàn)業(yè)務的時候發(fā)現(xiàn)一個坑,準確的來說是對于Python門外漢容易踩的坑;大概代碼如下:classMom(object):name=''sons=[]if__name__=='__main
今天python視頻教程欄目介紹單例模式中不同語言的不同實現(xiàn)。前言

前段時間在用 Python 實現(xiàn)業(yè)務的時候發(fā)現(xiàn)一個坑,準確的來說是對于 Python 門外漢容易踩的坑;LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

大概代碼如下:LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

class Mom(object): name = '' sons = []if __name__ == '__main__': m1 = Mom() m1.name = 'm1' m1.sons.append(['s1', 's2']) print '{} sons={}'.format(m1.name, m1.sons) m2 = Mom() m2.name = 'm2' m2.sons.append(['s3', 's4']) print '{} sons={}'.format(m2.name, m2.sons)復制代碼

首先定義了一個 Mom 的類,它包含了一個字符串類型的 name 與列表類型的 sons 屬性;LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

在使用時首先創(chuàng)建了該類的一個實例 m1 并往 sons 中寫入一個列表數(shù)據(jù);緊接著又創(chuàng)建了一個實例 m2 ,也往 sons 中寫入了另一個列表數(shù)據(jù)。LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

如果是一個 Javaer 很少寫 Python 看到這樣的代碼首先想到的輸出應該是:LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

m1 sons=[['s1', 's2']] m2 sons=[['s3', 's4']]復制代碼

但其實**終的輸出結(jié)果是:LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

m1 sons=[['s1', 's2']] m2 sons=[['s1', 's2'], ['s3', 's4']]復制代碼

如果想要達到期望值需要稍微修改一下:LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

class Mom(object): name = '' def __init__(self): self.sons = []復制代碼

只需要修改類的定義就可以了,我相信即使沒有 Python 相關經(jīng)驗對比這兩個代碼應該也能猜到原因:LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

在 Python 中如果需要將變量作為實例變量(也就是每個我們期望的輸出)時,需要將變量定義到構造函數(shù)中,通過 self 訪問。LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

如果只放在類中,和 Java 中的 static 靜態(tài)變量效果類似;這些數(shù)據(jù)由類共享,也就能解釋為什么會出現(xiàn)第一種情況,因為其中的 sons 是由 Mom 類共享,所以每次都會累加。LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

Python 單例

既然 Python 可以通過類變量達到變量在同一個類中共享的效果,那是否可以實現(xiàn)單例模式呢?LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

可以利用 Python 的 metaclass 的特性,動態(tài)的控制類的創(chuàng)建。LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

class Singleton(type): _instances = {} def __call__(cls, *args, **kwargs): if cls not in cls._instances: cls._instances[cls] = super(Singleton, cls).__call__(*args, **kwargs) return cls._instances[cls]復制代碼

首先創(chuàng)建一個 Singleton 的基類,然后我們在我們需要實現(xiàn)單例的類中將其作為 metaclassLR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

class MySQLDriver: __metaclass__ = Singleton def __init__(self): print 'MySQLDriver init.....'復制代碼

這樣Singleton 就可以控制 MySQLDriver 這個類的創(chuàng)建了;其實在 Singleton 中的 __call__ 可以很容易理解這個單例創(chuàng)建的過程:LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

定義一個私有的類屬性 _instances 的字典(也就是 Java 中的 map)可以做到在整個類中共享,無論創(chuàng)建多少個實例。當我們自定義類使用了 __metaclass__ = Singleton 后,便可以控制自定義類的創(chuàng)建了;如果已經(jīng)創(chuàng)建了實例,那就直接從 _instances 取出對象返回,不然就創(chuàng)建一個實例并寫回到 _instances ,有點 Spring 容器的感覺。if __name__ == '__main__': m1 = MySQLDriver() m2 = MySQLDriver() m3 = MySQLDriver() m4 = MySQLDriver() print m1 print m2 print m3 print m4 MySQLDriver init..... <__main__.MySQLDriver object at 0x10d848790> <__main__.MySQLDriver object at 0x10d848790> <__main__.MySQLDriver object at 0x10d848790> <__main__.MySQLDriver object at 0x10d848790>復制代碼

**后我們通過實驗結(jié)果可以看到單例創(chuàng)建成功。LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

Go 單例

由于**近團隊中有部分業(yè)務開始在用 go ,所以也想看看在 go 中如何實現(xiàn)單例。LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

type MySQLDriver struct { username string}復制代碼

在這樣一個簡單的結(jié)構體(可以簡單理解為 Java 中的 class)中是沒法類似于 Python 和 Java 一樣可以聲明類共享變量的;go 語言中不存在 static 的概念。LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

但我們可以在包中聲明一個全局變量來達到同樣的效果:LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

import "fmt"type MySQLDriver struct { username string}var mySQLDriver *MySQLDriverfunc GetDriver() *MySQLDriver { if mySQLDriver == nil { mySQLDriver = &MySQLDriver{} } return mySQLDriver }復制代碼

這樣在使用時:LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

func main() { driver := GetDriver() driver.username = "cj" fmt.Println(driver.username) driver2 := GetDriver() fmt.Println(driver2.username) }復制代碼

就不需要直接構造 MySQLDriver ,而是通過GetDriver() 函數(shù)來獲取,通過 debug 也能看到 driver 和 driver1 引用的是同一個內(nèi)存地址。LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

這樣的實現(xiàn)常規(guī)情況是沒有什么問題的,機智的朋友一定能想到和 Java 一樣,一旦并發(fā)訪問就沒那么簡單了。LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

在 go 中,如果有多個 goroutine 同時訪問GetDriver() ,那大概率會創(chuàng)建多個 MySQLDriver 實例。LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

這里說的沒那么簡單其實是相對于 Java 來說的,go 語言中提供了簡單的 api 便可實現(xiàn)臨界資源的訪問。LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

var lock sync.Mutexfunc GetDriver() *MySQLDriver { lock.Lock() defer lock.Unlock() if mySQLDriver == nil { fmt.Println("create instance......") mySQLDriver = &MySQLDriver{} } return mySQLDriver }func main() { for i := 0; i < 100; i++ { go GetDriver() } time.Sleep(2000 * time.Millisecond) }復制代碼

稍加改造上文的代碼,加入了LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

lock.Lock()defer lock.Unlock()復制代碼

代碼就能簡單的控制臨界資源的訪問,即便我們開啟了100個協(xié)程并發(fā)執(zhí)行,mySQLDriver 實例也只會被初始化一次。LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

這里的 defer 類似于 Java 中的 finally ,在方法調(diào)用前加上 go 關鍵字即可開啟一個協(xié)程。

雖說能滿足并發(fā)要求了,但其實這樣的實現(xiàn)也不夠優(yōu)雅;仔細想想這里LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

mySQLDriver = &MySQLDriver{}復制代碼

創(chuàng)建實例只會調(diào)用一次,但后續(xù)的每次調(diào)用都需要加鎖從而帶來了不必要的開銷。LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

這樣的場景每個語言都是相同的,拿 Java 來說是不是經(jīng)常看到這樣的單例實現(xiàn):LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

public class Singleton { private Singleton() {} private volatile static Singleton instance = null; public static Singleton getInstance() { if (instance == null) { synchronized (Singleton.class){ if (instance == null) { instance = new Singleton(); } } } return instance; } }復制代碼

這是一個典型的雙重檢查的單例,這里做了兩次檢查便可以避免后續(xù)其他線程再次訪問鎖。LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

同樣的對于 go 來說也類似:LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

func GetDriver() *MySQLDriver { if mySQLDriver == nil { lock.Lock() defer lock.Unlock() if mySQLDriver == nil { fmt.Println("create instance......") mySQLDriver = &MySQLDriver{} } } return mySQLDriver }復制代碼

和 Java 一樣,在原有基礎上額外做一次判斷也能達到同樣的效果。LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

但有沒有覺得這樣的代碼非常繁瑣,這一點 go 提供的 api 就非常省事了:LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

var once sync.Oncefunc GetDriver() *MySQLDriver { once.Do(func() { if mySQLDriver == nil { fmt.Println("create instance......") mySQLDriver = &MySQLDriver{} } }) return mySQLDriver }復制代碼

本質(zhì)上我們只需要不管在什么情況下 MySQLDriver 實例只初始化一次就能達到單例的目的,所以利用 once.Do() 就能讓代碼只執(zhí)行一次。LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

查看源碼會發(fā)現(xiàn) once.Do() 也是通過鎖來實現(xiàn),只是在加鎖之前利用底層的原子操作做了一次校驗,從而避免每次都要加鎖,性能會更好。LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

總結(jié)

相信大家日常開發(fā)中很少會碰到需要自己實現(xiàn)一個單例;首先大部分情況下我們都不需要單例,即使是需要,框架通常也都有集成。LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

類似于 go 這樣框架較少,需要我們自己實現(xiàn)時其實也不需要過多考慮并發(fā)的問題;摸摸自己肚子左上方的位置想想,自己寫的這個對象真的同時有幾百上千的并發(fā)來創(chuàng)建嘛?LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

不過通過這個對比會發(fā)現(xiàn) go 的語法確實要比 Java 簡潔太多,同時輕量級的協(xié)程以及簡單易用的并發(fā)工具支持看起來都要比 Java 優(yōu)雅許多;后續(xù)有機會再接著深入。LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

相關免費學習推薦:python視頻教程LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

以上就是單例模式中不同語言的不同實現(xiàn)的詳細內(nèi)容,更多請關注少兒編程網(wǎng)其它相關文章!LR4少兒編程網(wǎng)-Scratch_Python_教程_免費兒童編程學習平臺

預約試聽課

已有385人預約都是免費的,你也試試吧...

国产乱子伦高清露脸对白-国产精品欧美久久久久天天影视-国产91视频一区-亚洲欧美日产综合在线网-黄视频网站在线看-国产欧美亚洲精品第1页-亚洲www在线-大学生女人三级在线播放-日本在线视频www鲁啊鲁-国产成人精品一区二区仙踪林-69精品欧美一区二区三区-成人欧美亚洲-日本污污网站-中国妞xxxhd露脸偷拍视频-国产精品aⅴ在线观看-精品中文字幕在线

        黄色三级视频片| 久久国产这里只有精品| 色18美女社区| 天天综合网久久| 亚洲精品一二三四五区| 国产精品亚洲a| 好吊妞无缓冲视频观看| 国产va亚洲va在线va| 欧洲精品在线播放| 男人和女人啪啪网站| 香港三级韩国三级日本三级| 欧美成人三级在线视频| 91精品国产91久久久久麻豆 主演| 一区二区三区四区免费观看| 国产福利片一区二区| 国产资源第一页| 国产av国片精品| 三年中国国语在线播放免费| 中文字幕av专区| 男同互操gay射视频在线看| 欧美一区二区三区综合| 黄色片视频在线播放| 2025韩国大尺度电影| 国产精品国产三级国产专区51| 国产男女免费视频| 欧美黑人又粗又大又爽免费| 九色porny自拍| 国产911在线观看| 777精品久无码人妻蜜桃| 宅男噜噜噜66国产免费观看| 亚洲免费视频播放| 色综合av综合无码综合网站| 中文字幕第22页| 黄网站色视频免费观看| 老熟妇仑乱视频一区二区| 牛夜精品久久久久久久| av久久久久久| 国产熟人av一二三区| 色哟哟免费网站| 日韩精品免费播放| 91免费国产精品| 玖玖爱视频在线| 国产一区二区在线视频播放| 二级片在线观看| 亚洲综合在线网站| 久久久久久久中文| 国产欧美综合一区| 国内外成人免费在线视频| 国产精品无码av在线播放| 亚洲精品国产久| 杨幂毛片午夜性生毛片| 国产最新免费视频| 国产aaa免费视频| 无码人妻精品一区二区三区99v| 亚洲综合在线网站| 熟女人妇 成熟妇女系列视频| 国产a级黄色大片| 久久久精品视频国产| 国产黄色特级片| 97xxxxx| 中文字幕视频三区| 依人在线免费视频| 91国产精品视频在线观看| 日本熟妇人妻中出| 日本xxxx黄色| 狠狠热免费视频| 一级在线免费视频| 日韩av片专区| 中文 日韩 欧美| 在线观看日本www| 特级西西444| 国产美女永久无遮挡| 欧美一区二区视频在线播放| 91看片淫黄大片91| 丁香婷婷综合激情| 免费看日本毛片| 黄色www网站| 九九九九免费视频| 波多野结衣50连登视频| 人妻av中文系列| 亚洲成熟丰满熟妇高潮xxxxx| 日韩精品一区二区三区色欲av| 国产xxxx振车| 97xxxxx| 色乱码一区二区三区在线| 久久久国产精华液999999 | 欧美特黄aaa| 亚洲综合伊人久久| 无码av天堂一区二区三区| 免费毛片小视频| 亚欧激情乱码久久久久久久久| 久久久久久久高清| 男人插女人视频在线观看| 91在线视频观看免费| 老司机av福利| 国产第一页视频| 亚洲激情免费视频| 777久久久精品一区二区三区 | 日韩视频在线观看一区二区三区| 91在线第一页| 一二三四视频社区在线| 国产裸体舞一区二区三区| 91香蕉国产线在线观看| 中文字幕无码精品亚洲资源网久久| 欧美视频第一区| 国产精品h视频| 五月婷婷狠狠操| 婷婷五月综合缴情在线视频| www.cao超碰| 国产成人精品视频免费看| 熟女视频一区二区三区| 欧美男女交配视频| 日本www在线视频| 国产成年人视频网站| 波多野结衣家庭教师视频| 黑人粗进入欧美aaaaa| 欧美无砖专区免费| 强伦女教师2:伦理在线观看| 麻豆av免费在线| 91视频最新入口| 日韩小视频网站| 性生活免费观看视频| 国产原创精品在线| 手机看片福利盒子久久| 一本大道熟女人妻中文字幕在线| 日韩av自拍偷拍| 91精品999| 丰满少妇在线观看| 日本精品久久久久中文字幕| 久无码久无码av无码| 99热久久这里只有精品| mm131午夜| 麻豆映画在线观看| 国产精品8888| 中文字幕日韩精品无码内射| 超碰超碰超碰超碰超碰| 永久免费在线看片视频| 9l视频自拍9l视频自拍| 五月天色婷婷综合| 欧妇女乱妇女乱视频| 国产精品久久久久7777| 黄色片视频在线免费观看| 99热成人精品热久久66| 天天碰免费视频| 日韩精品视频网址| 日韩欧美视频免费在线观看| 免费cad大片在线观看| 丰满的少妇愉情hd高清果冻传媒| 99在线免费视频观看| 亚洲色成人一区二区三区小说| 欧美色图另类小说| 可以免费观看av毛片| av中文字幕网址| 久操手机在线视频| 国产精品97在线| 免费黄频在线观看| 国产一二三在线视频| 欧美黄色一级片视频| 久久久国产精华液999999 | 视色视频在线观看| 岛国毛片在线播放| 国产欧美精品aaaaaa片| 国产又猛又黄的视频| 精品人妻人人做人人爽| 成人中文字幕av| 日本国产中文字幕| 中文字幕第80页| 91欧美一区二区三区| 国产超级av在线| 怡红院亚洲色图| 丰满的少妇愉情hd高清果冻传媒| 国产高潮免费视频| 久无码久无码av无码| 国内国产精品天干天干| 黄色大片中文字幕| 亚洲热在线视频| 天天操天天爽天天射| 国产女主播自拍| 国产精品88久久久久久妇女| 欧美午夜性生活| 久久精品免费一区二区| 欧美黑人在线观看| 国产性生活一级片| 精品久久久久av| www.射射射| 波多野结衣免费观看| 牛夜精品久久久久久久| 奇米精品一区二区三区| 老司机午夜网站| 免费在线观看的av网站| 777精品久无码人妻蜜桃| 久久久久久av无码免费网站下载| 一二三av在线| 热久久久久久久久| 一道本视频在线观看| www.色就是色| 狠狠躁狠狠躁视频专区| 亚洲黄色a v| 九九九九九伊人| 国产欧美精品一二三|