GO語言bytes32 go語言bytes

Go語言和其他語言的不同之基本語法

Go語言作為出現(xiàn)比較晚的一門編程語言,在其原生支持高并發(fā)、云原生等領(lǐng)域的優(yōu)秀表現(xiàn),像目前比較流行的容器編排技術(shù)Kubernetes、容器技術(shù)Docker都是用Go語言寫的,像Java等其他面向?qū)ο蟮恼Z言,雖然也能做云原生相關(guān)的開發(fā),但是支持的程度遠沒有Go語言高,憑借其語言特性和簡單的編程方式,彌補了其他編程語言一定程度上的不足,一度成為一個熱門的編程語言。

10年積累的成都網(wǎng)站設(shè)計、網(wǎng)站建設(shè)經(jīng)驗,可以快速應對客戶對網(wǎng)站的新想法和需求。提供各種問題對應的解決方案。讓選擇我們的客戶得到更好、更有力的網(wǎng)絡(luò)服務。我雖然不認識你,你也不認識我。但先網(wǎng)站制作后付款的網(wǎng)站建設(shè)流程,更有賽罕免費網(wǎng)站建設(shè)讓你可以放心的選擇與我們合作。

最近在學習Go語言,我之前使用過C#、Java等面向?qū)ο缶幊痰恼Z言,發(fā)現(xiàn)其中有很多的編程方式和其他語言有區(qū)別的地方,好記性不如爛筆頭,總結(jié)一下,和其他語言做個對比。這里只總結(jié)差異的地方,具體的語法不做詳細的介紹。

種一棵樹最好的時間是十年前,其次是現(xiàn)在。

3)變量初始化時候可以和其他語言一樣直接在變量后面加等號,等號后面為要初始化的值,也可以使用變量名:=變量值的簡單方式

3)變量賦值 Go語言的變量賦值和多數(shù)語言一致,但是Go語言提供了多重賦值的功能,比如下面這個交換i、j變量的語句:

在不支持多重賦值的語言中,交換兩個變量的值需要引入一個中間變量:

4)匿名變量

在使用其他語言時,有時候要獲取一個值,卻因為該函數(shù)返回多個值而不得不定義很多沒有的變量,Go語言可以借助多重返回值和匿名變量來避免這種寫法,使代碼看起來更優(yōu)雅。

假如GetName()函數(shù)返回3個值,分別是firstName,lastName和nickName

若指向獲得nickName,則函數(shù)調(diào)用可以這樣寫

這種寫法可以讓代碼更清晰,從而大幅降低溝通的復雜度和維護的難度。

1)基本常量

常量使用關(guān)鍵字const 定義,可以限定常量類型,但不是必須的,如果沒有定義常量的類型,是無類型常量

2)預定義常量

Go語言預定義了這些常量 true、false和iota

iota比較特殊,可以被任務是一個可被編譯器修改的常量,在每個const關(guān)鍵字出現(xiàn)時被重置為0,然后在下一個const出現(xiàn)之前每出現(xiàn)一個iota,其所代表的數(shù)字會自動加1.

3)枚舉

1)int 和int32在Go語言中被認為是兩種不同類型的類型

2)Go語言定義了兩個浮點型float32和float64,其中前者等價于C語言的float類型,后者等價于C語言的double類型

3)go語言支持復數(shù)類型

復數(shù)實際上是由兩個實數(shù)(在計算機中使用浮點數(shù)表示)構(gòu)成,一個表示實部(real)、一個表示虛部(imag)。也就是數(shù)學上的那個復數(shù)

復數(shù)的表示

實部與虛部

對于一個復數(shù)z=complex(x,y),就可以通過Go語言內(nèi)置函數(shù)real(z)獲得該復數(shù)的實部,也就是x,通過imag(z)獲得該復數(shù)的虛部,也就是y

4)數(shù)組(值類型,長度在定義后無法再次修改,每次傳遞都將產(chǎn)生一個副本。)

5)數(shù)組切片(slice)

數(shù)組切片(slice)彌補了數(shù)組的不足,其數(shù)據(jù)結(jié)構(gòu)可以抽象為以下三個變量:

6)Map 在go語言中Map不需要引入任何庫,使用很方便

Go循環(huán)語句只支持for關(guān)鍵字,不支持while和do-while

goto語句的語義非常簡單,就是跳轉(zhuǎn)到本函數(shù)內(nèi)的某個標簽

今天就介紹到這里,以后我會在總結(jié)Go語言在其他方面比如并發(fā)編程、面向?qū)ο蟆⒕W(wǎng)絡(luò)編程等方面的不同及使用方法。希望對大家有所幫助。

golang之大端序、小端序

當分別處于大小端模式下的內(nèi)容存放如下

(1)大端模式存儲(存儲地址為16位)

地址 數(shù)據(jù)

0x0004(高地址) 0x44

0x0003 0x33

0x0002 0x22

0x0001(低地址) 0x11

(2)小端模式存儲(存儲地址為16位)

地址 數(shù)據(jù)

0x0004(高地址) 0x11

0x0003 0x22

0x0002 0x33

0x0001(低地址) 0x44

在前面也簡單闡述了大小端序的定義并結(jié)合簡單實例來說明,接下來會給出詳細實例來說明:

1、大端序(Big-Endian):或稱大尾序

一個類型: int32 的數(shù) 0X0A0B0C0D的內(nèi)存存放情況

數(shù)據(jù)是以8bits為單位

2、小端序(little-endian):或稱小尾序

比如0x00000001

大端序:內(nèi)存低比特位 00000000 00000000 00000000 00000001 內(nèi)存高比特位

小端序:內(nèi)存低比特位 10000000 00000000 00000000 00000000 內(nèi)存高比特位

其實在前面羅列出那么東西,最終是為了接下來講述的在golang中涉及到網(wǎng)絡(luò)傳輸、文件存儲時的選擇。一般來說網(wǎng)絡(luò)傳輸?shù)淖止?jié)序,可能是大端序或者小端序,取決于軟件開始時通訊雙方的協(xié)議規(guī)定。TCP/IP協(xié)議RFC1700規(guī)定使用“大端”字節(jié)序為網(wǎng)絡(luò)字節(jié)序,開發(fā)的時候需要遵守這一規(guī)則。默認golang是使用大端序。詳情見golang中包encoding/binary已提供了大、小端序的使用

輸出結(jié)果:

16909060 use big endian:

int32 to bytes: [1 2 3 4] ### [0001 0002 0003 0004]

bytes to int32: 16909060

16909060 use little endian:

int32 to bytes: [4 3 2 1] ### [0004 0003 0002 0001]

bytes to int32: 16909060

在RPCX框架中關(guān)于RPC調(diào)用過程涉及的傳遞消息進行編碼的,采用的就是大端序模式

GO語言運行32位

第一種

1.使用go env命令,查看系統(tǒng)的配置環(huán)境,可以看到GOARCH(當前系統(tǒng))是amd64

2.執(zhí)行 set GOARCH=386 配置go輸出系統(tǒng)平臺為32位,此時再用go env命令查看系統(tǒng)的配置環(huán)境,如圖:

第二種

打開Run Edit Configurations Configuration標簽

為Environment添加兩個設(shè)置項

參考

用Go來做以太坊開發(fā)⑤事件日志

智能合約具有在執(zhí)行期間“發(fā)出”事件的能力。 事件在以太坊中也稱為“日志”。 事件的輸出存儲在日志部分下的事務處理中。 事件已經(jīng)在以太坊智能合約中被廣泛使用,以便在發(fā)生相對重要的動作時記錄,特別是在代幣合約(即ERC-20)中,以指示代幣轉(zhuǎn)賬已經(jīng)發(fā)生。 這些部分將引導您完成從區(qū)塊鏈中讀取事件以及訂閱事件的過程,以便交易事務被礦工打包入塊的時候及時收到通知。

為了訂閱事件日志,我們需要做的第一件事就是撥打啟用websocket的以太坊客戶端。 幸運的是,Infura支持websockets。

下一步是創(chuàng)建篩選查詢。 在這個例子中,我們將閱讀來自我們在之前課程中創(chuàng)建的示例合約中的所有事件。

我們接收事件的方式是通過Go channel。 讓我們從go-ethereum core/types 包創(chuàng)建一個類型為 Log 的channel。

現(xiàn)在我們所要做的就是通過從客戶端調(diào)用 SubscribeFilterLogs 來訂閱,它接收查詢選項和輸出通道。 這將返回包含unsubscribe和error方法的訂閱結(jié)構(gòu)。

最后,我們要做的就是使用select語句設(shè)置一個連續(xù)循環(huán)來讀入新的日志事件或訂閱錯誤。

我們會在下個章節(jié)介紹如何解析日志。

Commands

Store.sol

event_subscribe.go

智能合約可以可選地釋放“事件”,其作為交易收據(jù)的一部分存儲日志。讀取這些事件相當簡單。首先我們需要構(gòu)造一個過濾查詢。我們從go-ethereum包中導入 FilterQuery 結(jié)構(gòu)體并用過濾選項初始化它。我們告訴它我們想過濾的區(qū)塊范圍并指定從中讀取此日志的合約地址。在示例中,我們將從在 智能合約章節(jié) 創(chuàng)建的智能合約中讀取特定區(qū)塊所有日志。

下一步是調(diào)用ethclient的 FilterLogs ,它接收我們的查詢并將返回所有的匹配事件日志。

返回的所有日志將是ABI編碼,因此它們本身不會非常易讀。為了解碼日志,我們需要導入我們智能合約的ABI。為此,我們導入編譯好的智能合約Go包,它將包含名稱格式為 ContractABI 的外部屬性。之后,我們使用go-ethereum中的 accounts/abi 包的 abi.JSON 函數(shù)返回一個我們可以在Go應用程序中使用的解析過的ABI接口。

現(xiàn)在我們可以通過日志進行迭代并將它們解碼為我么可以使用的類型。若您回憶起我們的樣例合約釋放的日志在Solidity中是類型為 bytes32 ,那么Go中的等價物將是 [32]byte 。我們可以使用這些類型創(chuàng)建一個匿名結(jié)構(gòu)體,并將指針作為第一個參數(shù)傳遞給解析后的ABI接口的 Unpack 函數(shù),以解碼原始的日志數(shù)據(jù)。第二個參數(shù)是我們嘗試解碼的事件名稱,最后一個參數(shù)是編碼的日志數(shù)據(jù)。

此外,日志結(jié)構(gòu)體包含附加信息,例如,區(qū)塊摘要,區(qū)塊號和交易摘要。

若您的solidity事件包含 indexed 事件類型,那么它們將成為 主題 而不是日志的數(shù)據(jù)屬性的一部分。在solidity中您最多只能有4個主題,但只有3個可索引的事件類型。第一個主題總是事件的簽名。我們的示例合約不包含可索引的事件,但如果它確實包含,這是如何讀取事件主題。

正如您所見,首個主題只是被哈希過的事件簽名。

這就是閱讀和解析日志的全部內(nèi)容。要學習如何訂閱日志,閱讀上個章節(jié)。

命令

Store.sol

event_read.go

首先,創(chuàng)建ERC-20智能合約的事件日志的interface文件 erc20.sol :

然后在給定abi使用 abigen 創(chuàng)建Go包

現(xiàn)在在我們的Go應用程序中,讓我們創(chuàng)建與ERC-20事件日志簽名類型相匹配的結(jié)構(gòu)類型:

初始化以太坊客戶端

按照ERC-20智能合約地址和所需的塊范圍創(chuàng)建一個“FilterQuery”。這個例子我們會用 ZRX 代幣:

用 FilterLogs 來過濾日志:

接下來我們將解析JSON abi,稍后我們將使用解壓縮原始日志數(shù)據(jù):

為了按某種日志類型進行過濾,我們需要弄清楚每個事件日志函數(shù)簽名的keccak256哈希值。 事件日志函數(shù)簽名哈希始終是 topic [0] ,我們很快就會看到。 以下是使用go-ethereum crypto 包計算keccak256哈希的方法:

現(xiàn)在我們將遍歷所有日志并設(shè)置switch語句以按事件日志類型進行過濾:

現(xiàn)在要解析 Transfer 事件日志,我們將使用 abi.Unpack 將原始日志數(shù)據(jù)解析為我們的日志類型結(jié)構(gòu)。 解包不會解析 indexed 事件類型,因為它們存儲在 topics 下,所以對于那些我們必須單獨解析,如下例所示:

Approval 日志也是類似的方法:

最后,把所有的步驟放一起:

我們可以把解析的日志與etherscan的數(shù)據(jù)對比:

Commands

erc20.sol

event_read_erc20.go

solc version used for these examples

要讀取 0x Protocol 事件日志,我們必須首先將solidity智能合約編譯為一個Go包。

安裝solc版本 0.4.11

為例如 Exchange.sol 的事件日志創(chuàng)建0x Protocol交易所智能合約接口:

Create the 0x protocol exchange smart contract interface for event logs as Exchange.sol :

接著給定abi,使用 abigen 來創(chuàng)建Go exchange 包:

Then use abigen to create the Go exchange package given the abi:

現(xiàn)在在我們的Go應用程序中,讓我們創(chuàng)建與0xProtocol事件日志簽名類型匹配的結(jié)構(gòu)體類型:

初始化以太坊客戶端:

創(chuàng)建一個 FilterQuery ,并為其傳遞0x Protocol智能合約地址和所需的區(qū)塊范圍:

用 FilterLogs 查詢?nèi)罩荆?/p>

接下來我們將解析JSON abi,我們后續(xù)將使用解壓縮原始日志數(shù)據(jù):

為了按某種日志類型過濾,我們需要知曉每個事件日志函數(shù)簽名的keccak256摘要。正如我們很快所見到的那樣,事件日志函數(shù)簽名摘要總是 topic[0] :

現(xiàn)在我們迭代所有的日志并設(shè)置一個switch語句來按事件日志類型過濾:

現(xiàn)在要解析 LogFill ,我們將使用 abi.Unpack 將原始數(shù)據(jù)類型解析為我們自定義的日志類型結(jié)構(gòu)體。Unpack不會解析 indexed 事件類型,因為這些它們存儲在 topics 下,所以對于那些我們必須單獨解析,如下例所示:

對于 LogCancel 類似:

最后是 LogError :

將它們放在一起并運行我們將看到以下輸出:

將解析后的日志輸出與etherscan上的內(nèi)容進行比較:

命令

Exchange.sol

event_read_0xprotocol.go

這些示例使用的solc版本

Go語言中的字節(jié)序

Go中的binary包實現(xiàn)了簡單的數(shù)字與字節(jié)序列的轉(zhuǎn)換以及變長值的編解碼

package main

import ( "fmt" "bytes" "encoding/binary" ) func main(){ n := 0x12345678 bytesBuffer := bytes.NewBuffer([]byte{}) //BigEndian 大端順序存儲 LittleEndian小端順序存儲 binary.Write(bytesBuffer, binary.BigEndian, int32(n)) data:=bytesBuffer.Bytes() fmt.Printf("[0]: %#x addr:%#x\n",data[0],data[0]) fmt.Printf("[0]: %#x addr:%#x\n",data[1],data[1]) fmt.Printf("[0]: %#x addr:%#x\n",data[2],data[2]) fmt.Printf("[0]: %#x addr:%#x\n",data[3],data[3]) }

輸出

[0]: 0x12 addr:0xc042010248 [1]: 0x34 addr:0xc042010249 [2]: 0x56 addr:0xc04201024a [3]: 0x78 addr:0xc04201024b

也可以使用下面的方式

n := 0x12345678 var data []byte = make([]byte,4) //操作的都是無符號整型 binary.BigEndian.PutUint32(data,uint32(n))

可以使用下面的方式判斷當前系統(tǒng)的字節(jié)序類型

const INT_SIZE int = int(unsafe.Sizeof(0))

//判斷我們系統(tǒng)中的字節(jié)序類型 func systemEdian() { var i int = 0x1 bs := (*[INT_SIZE]byte)(unsafe.Pointer(i)) if bs[0] == 0 { fmt.Println("system edian is little endian") } else { fmt.Println("system edian is big endian") } }

用Go來做以太坊開發(fā)④智能合約

在這個章節(jié)中我們會介紹如何用Go來編譯,部署,寫入和讀取智能合約。

與智能合約交互,我們要先生成相應智能合約的應用二進制接口ABI(application binary interface),并把ABI編譯成我們可以在Go應用中調(diào)用的格式。

第一步是安裝 Solidity編譯器 ( solc ).

Solc 在Ubuntu上有snapcraft包。

Solc在macOS上有Homebrew的包。

其他的平臺或者從源碼編譯的教程請查閱官方solidity文檔 install guide .

我們還得安裝一個叫 abigen 的工具,來從solidity智能合約生成ABI。

假設(shè)您已經(jīng)在計算機上設(shè)置了Go,只需運行以下命令即可安裝 abigen 工具。

我們將創(chuàng)建一個簡單的智能合約來測試。 學習更復雜的智能合約,或者智能合約的開發(fā)的內(nèi)容則超出了本書的范圍。 我強烈建議您查看 truffle framework 來學習開發(fā)和測試智能合約。

這里只是一個簡單的合約,就是一個鍵/值存儲,只有一個外部方法來設(shè)置任何人的鍵/值對。 我們還在設(shè)置值后添加了要發(fā)出的事件。

雖然這個智能合約很簡單,但它將適用于這個例子。

現(xiàn)在我們可以從一個solidity文件生成ABI。

它會將其寫入名為“Store_sol_Store.abi”的文件中

現(xiàn)在讓我們用 abigen 將ABI轉(zhuǎn)換為我們可以導入的Go文件。 這個新文件將包含我們可以用來與Go應用程序中的智能合約進行交互的所有可用方法。

為了從Go部署智能合約,我們還需要將solidity智能合約編譯為EVM字節(jié)碼。 EVM字節(jié)碼將在事務的數(shù)據(jù)字段中發(fā)送。 在Go文件上生成部署方法需要bin文件。

現(xiàn)在我們編譯Go合約文件,其中包括deploy方法,因為我們包含了bin文件。

在接下來的課程中,我們將學習如何部署智能合約,然后與之交互。

Commands

Store.sol

solc version used for these examples

如果你還沒看之前的章節(jié),請先學習 編譯智能合約的章節(jié) 因為這節(jié)內(nèi)容,需要先了解如何將智能合約編譯為Go文件。

假設(shè)你已經(jīng)導入從 abigen 生成的新創(chuàng)建的Go包文件,并設(shè)置ethclient,加載您的私鑰,下一步是創(chuàng)建一個有配置密匙的交易發(fā)送器(tansactor)。 首先從go-ethereum導入 accounts/abi/bind 包,然后調(diào)用傳入私鑰的 NewKeyedTransactor 。 然后設(shè)置通常的屬性,如nonce,燃氣價格,燃氣上線限制和ETH值。

如果你還記得上個章節(jié)的內(nèi)容, 我們創(chuàng)建了一個非常簡單的“Store”合約,用于設(shè)置和存儲鍵/值對。 生成的Go合約文件提供了部署方法。 部署方法名稱始終以單詞 Deploy 開頭,后跟合約名稱,在本例中為 Store 。

deploy函數(shù)接受有密匙的事務處理器,ethclient,以及智能合約構(gòu)造函數(shù)可能接受的任何輸入?yún)?shù)。我們測試的智能合約接受一個版本號的字符串參數(shù)。 此函數(shù)將返回新部署的合約地址,事務對象,我們可以交互的合約實例,還有錯誤(如果有)。

就這么簡單:)你可以用事務哈希來在Etherscan上查詢合約的部署狀態(tài):

Commands

Store.sol

contract_deploy.go

solc version used for these examples

這寫章節(jié)需要了解如何將智能合約的ABI編譯成Go的合約文件。如果你還沒看, 前先讀 上一個章節(jié) 。

一旦使用 abigen 工具將智能合約的ABI編譯為Go包,下一步就是調(diào)用“New”方法,其格式為“Newcontractname style="box-sizing: border-box; font-size: 16px; -ms-text-size-adjust: auto; -webkit-tap-highlight-color: transparent;"”,所以在我們的例子中如果你 回想一下它將是 NewStore 。 此初始化方法接收智能合約的地址,并返回可以開始與之交互的合約實例。/contractname

Commands

Store.sol

contract_load.go

solc version used for these examples

這寫章節(jié)需要了解如何將智能合約的ABI編譯成Go的合約文件。如果你還沒看, 前先讀 上一個章節(jié) 。

在上個章節(jié)我們學習了如何在Go應用程序中初始化合約實例。 現(xiàn)在我們將使用新合約實例提供的方法來閱讀智能合約。 如果你還記得我們在部署過程中設(shè)置的合約中有一個名為 version 的全局變量。 因為它是公開的,這意味著它們將成為我們自動創(chuàng)建的getter函數(shù)。 常量和view函數(shù)也接受 bind.CallOpts 作為第一個參數(shù)。了解可用的具體選項要看相應類的 文檔 一般情況下我們可以用 nil 。

Commands

Store.sol

contract_read.go

solc version used for these examples

這寫章節(jié)需要了解如何將智能合約的ABI編譯成Go的合約文件。如果你還沒看, 前先讀 上一個章節(jié) 。

寫入智能合約需要我們用私鑰來對交易事務進行簽名。

我們還需要先查到nonce和燃氣價格。

接下來,我們創(chuàng)建一個新的keyed transactor,它接收私鑰。

然后我們需要設(shè)置keyed transactor的標準交易選項。

現(xiàn)在我們加載一個智能合約的實例。如果你還記得 上個章節(jié) 我們創(chuàng)建一個名為 Store 的合約,并使用 abigen 工具生成一個Go文件。 要初始化它,我們只需調(diào)用合約包的 New 方法,并提供智能合約地址和ethclient,它返回我們可以使用的合約實例。

我們創(chuàng)建的智能合約有一個名為 SetItem 的外部方法,它接受solidity“bytes32”格式的兩個參數(shù)(key,value)。 這意味著Go合約包要求我們傳遞一個長度為32個字節(jié)的字節(jié)數(shù)組。 調(diào)用 SetItem 方法需要我們傳遞我們之前創(chuàng)建的 auth 對象(keyed transactor)。 在幕后,此方法將使用它的參數(shù)對此函數(shù)調(diào)用進行編碼,將其設(shè)置為事務的 data 屬性,并使用私鑰對其進行簽名。 結(jié)果將是一個已簽名的事務對象。

現(xiàn)在我就可以看到交易已經(jīng)成功被發(fā)送到了以太坊網(wǎng)絡(luò)了:

要驗證鍵/值是否已設(shè)置,我們可以讀取智能合約中的值。

搞定!

Commands

Store.sol

contract_write.go

solc version used for these examples

有時您需要讀取已部署的智能合約的字節(jié)碼。 由于所有智能合約字節(jié)碼都存在于區(qū)塊鏈中,因此我們可以輕松獲取它。

首先設(shè)置客戶端和要讀取的字節(jié)碼的智能合約地址。

現(xiàn)在你需要調(diào)用客戶端的 codeAt 方法。 codeAt 方法接受智能合約地址和可選的塊編號,并以字節(jié)格式返回字節(jié)碼。

你也可以在etherscan上查詢16進制格式的字節(jié)碼

contract_bytecode.go

首先創(chuàng)建一個ERC20智能合約interface。 這只是與您可以調(diào)用的函數(shù)的函數(shù)定義的契約。

然后將interface智能合約編譯為JSON ABI,并使用 abigen 從ABI創(chuàng)建Go包。

假設(shè)我們已經(jīng)像往常一樣設(shè)置了以太坊客戶端,我們現(xiàn)在可以將新的 token 包導入我們的應用程序并實例化它。這個例子里我們用 Golem 代幣的地址.

我們現(xiàn)在可以調(diào)用任何ERC20的方法。 例如,我們可以查詢用戶的代幣余額。

我們還可以讀ERC20智能合約的公共變量。

我們可以做一些簡單的數(shù)學運算將余額轉(zhuǎn)換為可讀的十進制格式。

同樣的信息也可以在etherscan上查詢:

Commands

erc20.sol

contract_read_erc20.go

solc version used for these examples

網(wǎng)站題目:GO語言bytes32 go語言bytes
文章地址:http://muchs.cn/article14/doccpge.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供面包屑導航、品牌網(wǎng)站建設(shè)、搜索引擎優(yōu)化、微信公眾號外貿(mào)建站、建站公司

廣告

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

小程序開發(fā)