go語(yǔ)言原始套接字 go接口

Go小知識(shí)新解

1、值接收者和指針接收者

成都創(chuàng)新互聯(lián)專注于保亭黎族網(wǎng)站建設(shè)服務(wù)及定制,我們擁有豐富的企業(yè)做網(wǎng)站經(jīng)驗(yàn)。 熱誠(chéng)為您提供保亭黎族營(yíng)銷(xiāo)型網(wǎng)站建設(shè),保亭黎族網(wǎng)站制作、保亭黎族網(wǎng)頁(yè)設(shè)計(jì)、保亭黎族網(wǎng)站官網(wǎng)定制、成都小程序開(kāi)發(fā)服務(wù),打造保亭黎族網(wǎng)絡(luò)公司原創(chuàng)品牌,更為您提供保亭黎族網(wǎng)站排名全網(wǎng)營(yíng)銷(xiāo)落地服務(wù)。

所謂指針接收者和值接收者這兩個(gè)概念,用GO寫(xiě)了一陣子代碼的人都了解了,這里只做簡(jiǎn)要說(shuō)明一下,也就是對(duì)于一個(gè)給定結(jié)構(gòu),咱們對(duì)結(jié)構(gòu)進(jìn)行方法包裝的時(shí)候,固定必傳的參數(shù),用來(lái)指向這個(gè)對(duì)象結(jié)構(gòu)自身的一個(gè)參數(shù),在go中也就是形式如下:

我們對(duì)結(jié)構(gòu)體testStruct進(jìn)行了包裝,提供了兩個(gè)方法,sum和modify,其中sum的方法接收者為a testStruct,這個(gè)就是值接收者,而modify的接收者為a *testStruct就是指針接收者,也就是說(shuō)固定對(duì)象指針,一個(gè)傳遞的是指針地址,而另外一個(gè)直接傳遞的是結(jié)構(gòu)值拷貝了

對(duì)指針有一定了解的,都可以知道,指針傳遞過(guò)去的,可以直接修改結(jié)構(gòu)內(nèi)部?jī)?nèi)容,而值傳遞過(guò)去的,無(wú)論如何修改這個(gè)接收者的數(shù)據(jù),不會(huì)對(duì)原對(duì)象結(jié)構(gòu)產(chǎn)生影響。而對(duì)于咱們包裝結(jié)構(gòu)對(duì)象的時(shí)候,到底是使用指針還是使用值接收者,這個(gè)實(shí)際上沒(méi)有太大的定論,就我個(gè)人的觀點(diǎn)來(lái)說(shuō),如果結(jié)構(gòu)體占有的內(nèi)存空間不大(kb級(jí)別),而又不需要修改內(nèi)部的,同時(shí)結(jié)構(gòu)對(duì)象內(nèi)部沒(méi)有同步對(duì)象比如(sync包中的mutex,rwlock,waitgroup等之類(lèi)的結(jié)構(gòu)的話,可以直接值傳遞,實(shí)際上值copy也沒(méi)有咱們想象的那么慢,很多時(shí)候,都用指針,最后的gc回收掃描可能都比咱們這個(gè)傳遞copy的消耗大) p="" /kb級(jí)別),而又不需要修改內(nèi)部的,同時(shí)結(jié)構(gòu)對(duì)象內(nèi)部沒(méi)有同步對(duì)象比如(sync包中的mutex,rwlock,waitgroup等之類(lèi)的結(jié)構(gòu)的話,可以直接值傳遞,實(shí)際上值copy也沒(méi)有咱們想象的那么慢,很多時(shí)候,都用指針,最后的gc回收掃描可能都比咱們這個(gè)傳遞copy的消耗大)

2、實(shí)現(xiàn)接口的值接收者和指針接收者有啥區(qū)別

也就是比如定義如下

這里面的值接收者和指針接收者有什么區(qū)別,這里咱來(lái)寫(xiě)一個(gè)測(cè)試

通過(guò)這個(gè)測(cè)試用例可以發(fā)現(xiàn),指針接收者實(shí)現(xiàn)的接口可以同時(shí)支持轉(zhuǎn)移到值接收者接口和指針接收者接口,而用值接收者實(shí)現(xiàn)的接口,則無(wú)法轉(zhuǎn)移到使用指針接收者實(shí)現(xiàn)的接口,為啥子呢?目前網(wǎng)上或者各類(lèi)資料上都是給的一個(gè)很官方很官方,而且很書(shū)面話難以理解的說(shuō)明,大致意思如下:

這是目前網(wǎng)絡(luò)或者各種資料上都是差不多是這樣說(shuō)的,看似講了,實(shí)際上就說(shuō)了一個(gè)結(jié)果,根本就沒(méi)說(shuō)出來(lái)一個(gè)為什么。這樣的總結(jié)出來(lái),一個(gè)初學(xué)者的角度來(lái)看,是很不好理解的,初學(xué)者要么就是死記硬背,要么就是生搬硬套,甚至直到寫(xiě)了好多好多代碼了,都還沒(méi)有搞明白一個(gè)為啥子,只是會(huì)用了而已,從長(zhǎng)遠(yuǎn)來(lái)說(shuō)這是不利于自身提高的。

有這兩個(gè)本質(zhì)點(diǎn),咱們自己來(lái)思考一下,如果你來(lái)實(shí)現(xiàn)這個(gè)編譯器的時(shí)候,用指針接收的時(shí)候,指針接收者,默認(rèn)就能直接獲取支持,而值接收者實(shí)現(xiàn)接口的咱們可以直接來(lái)一個(gè)解指針就變成了值,就能匹配上值接收者實(shí)現(xiàn)的接口了,反過(guò)來(lái)說(shuō),如果值接收者,此時(shí)要匹配指針接收者,如何匹配呢,取一個(gè)地址就變成了指針了,此時(shí)數(shù)據(jù)類(lèi)型確實(shí)是匹配了,但是,地址指向的數(shù)據(jù)區(qū)不對(duì)了,因?yàn)槲覀儎倓傉f(shuō)了值接收者拷貝了一個(gè)新值之后是完全的一個(gè)新的對(duì)象,這個(gè)新對(duì)象和原始對(duì)象一點(diǎn)關(guān)系都沒(méi)有,咱們?nèi)〉刂?,取的也是這個(gè)新對(duì)象地址,對(duì)這個(gè)地址進(jìn)行操作,也是這個(gè)新對(duì)象的內(nèi)部數(shù)據(jù),和原始數(shù)據(jù)內(nèi)部沒(méi)有任何關(guān)系,所以由此就能推斷出,這個(gè)是為啥子值接收者不能匹配上指針接收者,而指針接收者卻可以匹配上值接收者了。

1、在某個(gè)作用域內(nèi)部,所有定義的字符串的數(shù)據(jù)區(qū)相同

這個(gè)很好驗(yàn)證,代碼如下:

2、字符串相加會(huì)產(chǎn)生一個(gè)新串

這個(gè)也很好驗(yàn)證

3、字符串真的是不可變的嗎

實(shí)際上從字符串的結(jié)構(gòu)

從這個(gè)結(jié)構(gòu),就能大致的推斷出來(lái),字符串設(shè)計(jì)成這樣就不具備直接擴(kuò)容+來(lái)增加新數(shù)據(jù),而如果咱們直接使用string[index] = 'a',用這種方式,就不能編譯通過(guò),官方也確定說(shuō)字符串是不可變的。那么真的是不可變的嗎?

通過(guò)上面的結(jié)構(gòu),在加上go的slice切片的數(shù)據(jù)結(jié)構(gòu)

由此可見(jiàn),咱們可以將字符串通過(guò)指針?lè)绞綇?qiáng)轉(zhuǎn)為一個(gè)byte數(shù)組指針,然后通過(guò)byte切片來(lái)修改,試試

編譯通過(guò),運(yùn)行報(bào)錯(cuò)

unexpected fault address 0xae2e27

fatal error: fault

這個(gè)錯(cuò)誤,基本上就是一個(gè)內(nèi)存的保護(hù)錯(cuò)誤,是寫(xiě)異常,所以說(shuō)明了,這個(gè)肯定做了內(nèi)存寫(xiě)保護(hù),那么直接修改一下內(nèi)存區(qū)的屬性,去掉他的寫(xiě)保護(hù),就能寫(xiě)了

以下代碼都是在Win平臺(tái),Go1.18,Win上修改內(nèi)存權(quán)限屬性,使用VirtualProtect,代碼如下

此時(shí)運(yùn)行,就能發(fā)現(xiàn)tstr的內(nèi)容被咱們變了,這種情況實(shí)際上在實(shí)際開(kāi)發(fā)中不具有實(shí)際意義,因?yàn)楸旧碓谡Z(yǔ)言層面,已經(jīng)做了層層限制,咱們這是屬于非法強(qiáng)制的操作方式,是流氓行為,那么是否有比較溫和一點(diǎn)的操作方式呢?答案是有的,且往下看。

通過(guò)上面,我們已經(jīng)用到了字符串結(jié)構(gòu),切片結(jié)構(gòu),要想字符串內(nèi)容可變,那么咱們自己構(gòu)造字符串的數(shù)據(jù)內(nèi)容區(qū)域,且讓這個(gè)數(shù)據(jù)區(qū)木有內(nèi)存寫(xiě)保護(hù)不就行了,內(nèi)容區(qū)可變,GO原生態(tài)的byte數(shù)組不就行嘛,所以咱們自己構(gòu)造一下

此時(shí)我們直接修改buffer的內(nèi)容,就是直接修改了str的數(shù)據(jù)內(nèi)容了。而又不會(huì)像前面的一樣遇到內(nèi)存寫(xiě)保護(hù)

4、字符串轉(zhuǎn)換優(yōu)化時(shí)可能碰到的坑

通過(guò)前面討論的字符串的可變性的方法,咱們可以知道,很多時(shí)候,[]byte到字符串的轉(zhuǎn)變,可以直接構(gòu)造其結(jié)構(gòu),而共享數(shù)據(jù),從而達(dá)到減少數(shù)據(jù)內(nèi)存copy的方式來(lái)進(jìn)行優(yōu)化,再使用這些優(yōu)化的時(shí)候,一定需要注意,字符串或者數(shù)組的生命周期,是否會(huì)存在被改寫(xiě)的情況,從而導(dǎo)致前后不一致的問(wèn)題。

比如下面這段代碼:

大家可以猜想一下,這個(gè)最后里面的數(shù)據(jù)mmp中,"test"的value是多少,"abcd"的value是多少,然后想想為什么,且等端午之后,再來(lái)分解

如何使用Go語(yǔ)言是操作Docker

Docker 提供了一個(gè)與 Docker 守護(hù)進(jìn)程交互的 API (稱為Docker Engine API),我們可以使用官方提供的 Go 語(yǔ)言的 SDK 進(jìn)行構(gòu)建和擴(kuò)展 Docker 應(yīng)用程序和解決方案。

轉(zhuǎn)自:

整理:地鼠文檔

通過(guò)下面的命令就可以安裝 SDK 了:

該部分會(huì)介紹如何使用 Golang + Docker API 進(jìn)行管理本地的 Docker。

第一個(gè)例子將展示如何運(yùn)行容器,相當(dāng)于 docker run docker.io/library/alpine echo "hello world" :

還可以在后臺(tái)運(yùn)行容器,相當(dāng)于 docker run -d bfirsh/reticulate-splines :

列出正在運(yùn)行的容器,就像使用 docker ps 一樣:

如果是 docker ps -a ,我們可以通過(guò)修改 types.ContainerListOptions 中的 All 屬性達(dá)到這個(gè)目的:

通過(guò)上面的例子,我們可以獲取容器的列表,所以在這個(gè)案例中,我們可以去停止所有正在運(yùn)行的容器。

通過(guò)指定容器的 ID,我們可以獲取對(duì)應(yīng) ID 的容器的日志:

獲取本地所有的鏡像,相當(dāng)于 docker image ls 或 docker images :

拉取指定鏡像,相當(dāng)于 docker pull alpine :

除了公開(kāi)的鏡像,我們平時(shí)還會(huì)用到一些私有鏡像,可以是 DockerHub 上私有鏡像,也可以是自托管的鏡像倉(cāng)庫(kù),比如 harbor 。這個(gè)時(shí)候,我們需要提供對(duì)應(yīng)的憑證才可以拉取鏡像。

值得注意的是:在使用 Docker API 的 Go SDK 時(shí),憑證是以明文的方式進(jìn)行傳輸?shù)?,所以如果是自建的鏡像倉(cāng)庫(kù),請(qǐng)務(wù)必使用 HTTPS !

我們可以將一個(gè)已有的容器通過(guò) commit 保存成一個(gè)鏡像:

當(dāng)然,除了可以管理本地的 Docker , 我們同樣也可以通過(guò)使用 Golang + Docker API 管理遠(yuǎn)程的 Docker 。

默認(rèn) Docker 是通過(guò)非網(wǎng)絡(luò)的 Unix 套接字運(yùn)行的,只能夠進(jìn)行本地通信( /var/run/docker.sock ),是不能夠直接遠(yuǎn)程連接 Docker 的。

我們需要編輯配置文件 /etc/docker/daemon.json ,并修改以下內(nèi)容(把 192.168.59.3 改成你自己的 IP 地址),然后重啟 Docker :

創(chuàng)建 client 的時(shí)候需要指定遠(yuǎn)程 Docker 的地址,這樣就可以像管理本地 Docker 一樣管理遠(yuǎn)程的 Docker 了:

現(xiàn)在已經(jīng)有很多可以管理 Docker 的產(chǎn)品,它們便是這樣進(jìn)行實(shí)現(xiàn)的,比如: portainer 。

如何看待go語(yǔ)言泛型的最新設(shè)計(jì)?

Go 由于不支持泛型而臭名昭著,但最近,泛型已接近成為現(xiàn)實(shí)。Go 團(tuán)隊(duì)實(shí)施了一個(gè)看起來(lái)比較穩(wěn)定的設(shè)計(jì)草案,并且正以源到源翻譯器原型的形式獲得關(guān)注。本文講述的是泛型的最新設(shè)計(jì),以及如何自己嘗試泛型。

例子

FIFO Stack

假設(shè)你要?jiǎng)?chuàng)建一個(gè)先進(jìn)先出堆棧。沒(méi)有泛型,你可能會(huì)這樣實(shí)現(xiàn):

type?Stack?[]interface{}func?(s?Stack)?Peek()?interface{}?{

return?s[len(s)-1]

}

func?(s?*Stack)?Pop()?{

*s?=?(*s)[:

len(*s)-1]

}

func?(s?*Stack)?Push(value?interface{})?{

*s?=?

append(*s,?value)

}

但是,這里存在一個(gè)問(wèn)題:每當(dāng)你 Peek 項(xiàng)時(shí),都必須使用類(lèi)型斷言將其從 interface{} 轉(zhuǎn)換為你需要的類(lèi)型。如果你的堆棧是 *MyObject 的堆棧,則意味著很多 s.Peek().(*MyObject)這樣的代碼。這不僅讓人眼花繚亂,而且還可能引發(fā)錯(cuò)誤。比如忘記 * 怎么辦?或者如果您輸入錯(cuò)誤的類(lèi)型怎么辦?s.Push(MyObject{})` 可以順利編譯,而且你可能不會(huì)發(fā)現(xiàn)到自己的錯(cuò)誤,直到它影響到你的整個(gè)服務(wù)為止。

通常,使用 interface{} 是相對(duì)危險(xiǎn)的。使用更多受限制的類(lèi)型總是更安全,因?yàn)榭梢栽诰幾g時(shí)而不是運(yùn)行時(shí)發(fā)現(xiàn)問(wèn)題。

泛型通過(guò)允許類(lèi)型具有類(lèi)型參數(shù)來(lái)解決此問(wèn)題:

type?Stack(type?T)?[]Tfunc?(s?Stack(T))?Peek()?T?{

return?s[len(s)-1]

}

func?(s?*Stack(T))?Pop()?{

*s?=?(*s)[:

len(*s)-1]

}

func?(s?*Stack(T))?Push(value?T)?{

*s?=?

append(*s,?value)

}

這會(huì)向 Stack 添加一個(gè)類(lèi)型參數(shù),從而完全不需要 interface{}?,F(xiàn)在,當(dāng)你使用 Peek() 時(shí),返回的值已經(jīng)是原始類(lèi)型,并且沒(méi)有機(jī)會(huì)返回錯(cuò)誤的值類(lèi)型。這種方式更安全,更容易使用。(譯注:就是看起來(lái)更丑陋,^-^)

此外,泛型代碼通常更易于編譯器優(yōu)化,從而獲得更好的性能(以二進(jìn)制大小為代價(jià))。如果我們對(duì)上面的非泛型代碼和泛型代碼進(jìn)行基準(zhǔn)測(cè)試,我們可以看到區(qū)別:

type?MyObject?struct?{

X?

int

}

var?sink?MyObjectfunc?BenchmarkGo1(b?*testing.B)?{

for?i?:=?0;?i??b.N;?i++?{

var?s?Stack

s.Push(MyObject{})

s.Push(MyObject{})

s.Pop()

sink?=?s.Peek().(MyObject)

}

}

func?BenchmarkGo2(b?*testing.B)?{

for?i?:=?0;?i??b.N;?i++?{

var?s?Stack(MyObject)

s.Push(MyObject{})

s.Push(MyObject{})

s.Pop()

sink?=?s.Peek()

}

}

結(jié)果:

BenchmarkGo1BenchmarkGo1-16?????12837528?????????87.0?ns/op???????48?B/op????????2?allocs/opBenchmarkGo2BenchmarkGo2-16?????28406479?????????41.9?ns/op???????24?B/op????????2?allocs/op

在這種情況下,我們分配更少的內(nèi)存,同時(shí)泛型的速度是非泛型的兩倍。

合約(Contracts)

上面的堆棧示例適用于任何類(lèi)型。但是,在許多情況下,你需要編寫(xiě)僅適用于具有某些特征的類(lèi)型的代碼。例如,你可能希望堆棧要求類(lèi)型實(shí)現(xiàn) String() 函數(shù)

GO語(yǔ)言(十六):模糊測(cè)試入門(mén)(上)

本教程介紹了 Go 中模糊測(cè)試的基礎(chǔ)知識(shí)。通過(guò)模糊測(cè)試,隨機(jī)數(shù)據(jù)會(huì)針對(duì)您的測(cè)試運(yùn)行,以嘗試找出漏洞或?qū)е卤罎⒌妮斎搿?梢酝ㄟ^(guò)模糊測(cè)試發(fā)現(xiàn)的一些漏洞示例包括 SQL 注入、緩沖區(qū)溢出、拒絕服務(wù)和跨站點(diǎn)腳本攻擊。

在本教程中,您將為一個(gè)簡(jiǎn)單的函數(shù)編寫(xiě)一個(gè)模糊測(cè)試,運(yùn)行 go 命令,并調(diào)試和修復(fù)代碼中的問(wèn)題。

首先,為您要編寫(xiě)的代碼創(chuàng)建一個(gè)文件夾。

1、打開(kāi)命令提示符并切換到您的主目錄。

在 Linux 或 Mac 上:

在 Windows 上:

2、在命令提示符下,為您的代碼創(chuàng)建一個(gè)名為 fuzz 的目錄。

3、創(chuàng)建一個(gè)模塊來(lái)保存您的代碼。

運(yùn)行g(shù)o mod init命令,為其提供新代碼的模塊路徑。

接下來(lái),您將添加一些簡(jiǎn)單的代碼來(lái)反轉(zhuǎn)字符串,稍后我們將對(duì)其進(jìn)行模糊測(cè)試。

在此步驟中,您將添加一個(gè)函數(shù)來(lái)反轉(zhuǎn)字符串。

a.使用您的文本編輯器,在 fuzz 目錄中創(chuàng)建一個(gè)名為 main.go 的文件。

獨(dú)立程序(與庫(kù)相反)始終位于 package 中main。

此函數(shù)將接受string,使用byte進(jìn)行循環(huán) ,并在最后返回反轉(zhuǎn)的字符串。

此函數(shù)將運(yùn)行一些Reverse操作,然后將輸出打印到命令行。這有助于查看運(yùn)行中的代碼,并可能有助于調(diào)試。

e.該main函數(shù)使用 fmt 包,因此您需要導(dǎo)入它。

第一行代碼應(yīng)如下所示:

從包含 main.go 的目錄中的命令行,運(yùn)行代碼。

可以看到原來(lái)的字符串,反轉(zhuǎn)它的結(jié)果,然后再反轉(zhuǎn)它的結(jié)果,就相當(dāng)于原來(lái)的了。

現(xiàn)在代碼正在運(yùn)行,是時(shí)候測(cè)試它了。

在這一步中,您將為Reverse函數(shù)編寫(xiě)一個(gè)基本的單元測(cè)試。

a.使用您的文本編輯器,在 fuzz 目錄中創(chuàng)建一個(gè)名為 reverse_test.go 的文件。

b.將以下代碼粘貼到 reverse_test.go 中。

這個(gè)簡(jiǎn)單的測(cè)試將斷言列出的輸入字符串將被正確反轉(zhuǎn)。

使用運(yùn)行單元測(cè)試go test

接下來(lái),您將單元測(cè)試更改為模糊測(cè)試。

單元測(cè)試有局限性,即每個(gè)輸入都必須由開(kāi)發(fā)人員添加到測(cè)試中。模糊測(cè)試的一個(gè)好處是它可以為您的代碼提供輸入,并且可以識(shí)別您提出的測(cè)試用例沒(méi)有達(dá)到的邊緣用例。

在本節(jié)中,您將單元測(cè)試轉(zhuǎn)換為模糊測(cè)試,這樣您就可以用更少的工作生成更多的輸入!

請(qǐng)注意,您可以將單元測(cè)試、基準(zhǔn)測(cè)試和模糊測(cè)試保存在同一個(gè) *_test.go 文件中,但對(duì)于本示例,您將單元測(cè)試轉(zhuǎn)換為模糊測(cè)試。

在您的文本編輯器中,將 reverse_test.go 中的單元測(cè)試替換為以下模糊測(cè)試。

Fuzzing 也有一些限制。在您的單元測(cè)試中,您可以預(yù)測(cè)Reverse函數(shù)的預(yù)期輸出,并驗(yàn)證實(shí)際輸出是否滿足這些預(yù)期。

例如,在測(cè)試用例Reverse("Hello, world")中,單元測(cè)試將返回指定為"dlrow ,olleH".

模糊測(cè)試時(shí),您無(wú)法預(yù)測(cè)預(yù)期輸出,因?yàn)槟鸁o(wú)法控制輸入。

但是,Reverse您可以在模糊測(cè)試中驗(yàn)證函數(shù)的一些屬性。在這個(gè)模糊測(cè)試中檢查的兩個(gè)屬性是:

(1)將字符串反轉(zhuǎn)兩次保留原始值

(2)反轉(zhuǎn)的字符串將其狀態(tài)保留為有效的 UTF-8。

注意單元測(cè)試和模糊測(cè)試之間的語(yǔ)法差異:

(3)確保新包unicode/utf8已導(dǎo)入。

隨著單元測(cè)試轉(zhuǎn)換為模糊測(cè)試,是時(shí)候再次運(yùn)行測(cè)試了。

a.在不進(jìn)行模糊測(cè)試的情況下運(yùn)行模糊測(cè)試,以確保種子輸入通過(guò)。

如果您在該文件中有其他測(cè)試,您也可以運(yùn)行g(shù)o test -run=FuzzReverse,并且您只想運(yùn)行模糊測(cè)試。

b.運(yùn)行FuzzReverse模糊測(cè)試,查看是否有任何隨機(jī)生成的字符串輸入會(huì)導(dǎo)致失敗。這是使用go test新標(biāo)志-fuzz執(zhí)行的。

模糊測(cè)試時(shí)發(fā)生故障,導(dǎo)致問(wèn)題的輸入被寫(xiě)入將在下次運(yùn)行的種子語(yǔ)料庫(kù)文件中g(shù)o test,即使沒(méi)有-fuzz標(biāo)志也是如此。要查看導(dǎo)致失敗的輸入,請(qǐng)?jiān)谖谋揪庉嬈髦写蜷_(kāi)寫(xiě)入 testdata/fuzz/FuzzReverse 目錄的語(yǔ)料庫(kù)文件。您的種子語(yǔ)料庫(kù)文件可能包含不同的字符串,但格式相同。

語(yǔ)料庫(kù)文件的第一行表示編碼版本。以下每一行代表構(gòu)成語(yǔ)料庫(kù)條目的每種類(lèi)型的值。由于 fuzz target 只需要 1 個(gè)輸入,因此版本之后只有 1 個(gè)值。

c.運(yùn)行沒(méi)有-fuzz標(biāo)志的go test; 新的失敗種子語(yǔ)料庫(kù)條目將被使用:

由于我們的測(cè)試失敗,是時(shí)候調(diào)試了。

本文名稱:go語(yǔ)言原始套接字 go接口
當(dāng)前路徑:http://muchs.cn/article32/doshssc.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供ChatGPT、電子商務(wù)、用戶體驗(yàn)、云服務(wù)器、手機(jī)網(wǎng)站建設(shè)、企業(yè)建站

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)

商城網(wǎng)站建設(shè)