使用 Browser-solidity 在 Go-Ethereum 上进行简单的智能合约部署

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了使用 Browser-solidity 在 Go-Ethereum 上进行简单的智能合约部署相关的知识,希望对你有一定的参考价值。

1、基本概念

  之前文章中,已经讲过 Go-Ethereum 1.7.2 如何安装及搭建以太坊私有链,这里就不多介绍。

  在本文讲解之前,先介绍以下几点基本概念。

1.1、什么是智能合约?

  智能合约是存储在区块链上的一段代码,它们可以被区块链上的交易所触发,触发后,这段代码可以从区块链上读取数据或者向区块链上写入数据。
  

1.2、什么是Solidity?

  Solidity 是 Ethereum 的一种契约型编程语言,运行在Ethereum虚拟机(EVM)之上。
  

1.2.1、Solidity的语言特性

  它的语法接近于javascript,是一种面向对象的语言。但作为一种真正意义上运行在网络上的去中心合约,它又有很多的不同,下面列举一些:

  • 以太坊底层是基于帐户,而非UTXO的,所以有一个特殊的Address的类型。用于定位用户,定位合约,定位合约的代码(合约本身也是一个帐户)。

  • 由于语言内嵌框架是支持支付的,所以提供了一些关键字,如payable,可以在语言层面直接支持支付。

  • 存储是使用网络上的区块链,数据的每一个状态都可以永久存储,所以需要确定变量使用内存,还是区块链。

  • 运行环境是在去中心化的网络上,会比较强调合约或函数执行的调用的方式。因为原来一个简单的函数调用变为了一个网络上的节点中的代码执行。

  • 最后一个非常大的不同则是它的异常机制,一旦出现异常,所有的执行都将会被回撤,这主要是为了保证合约执行的原子性,以避免中间状态出现的数据不一致。

  

1.3、什么是 Browser-solidity?

  Browser-solidity 是一个官方提供的一个基于浏览器的合约编译器,非常好用,而且build版本会紧跟最新的 Solidity 的 build 版本。但由于网络原因以及GFW的存在,有可能会另一部分人访问很慢,进而影响开发效率。

  可以直接在线使用,访问后面的地址:https://remix.ethereum.org

  

2、Browser-solidity 的源码在哪里?

  https://github.com/ethereum/browser-solidity

  

3、如何在本地安装 Browser-solidity?

  由于国内网络的GFW原因,以及在网络上访问会影响效率,推荐部署到本地开发。
下面的4行代码开始将Browser-solidity安装到本地。

? /Users/lion/my_project/_eth > git clone https://github.com/ethereum/browser-solidity
? /Users/lion/my_project/_eth > cd browser-solidity
? /Users/lion/my_project/_eth/browser-solidity git:(master) > npm install
? /Users/lion/my_project/_eth/browser-solidity git:(master) > npm run prepublish

  
  安装以后,通过以下命令启动Browser-solidity

npm start

  
  启动后,控制台会输出如下信息:

? /Users/lion/my_project/_eth/browser-solidity git:(master) >npm start
>[email protected] start /Users/lion/my_project/_eth/browser-solidity
> npm-run-all -lpr serve watch onchange

[serve   ]
[serve   ] > [email protected] serve /Users/lion/my_project/_eth/browser-solidity
[serve   ] > execr --silent http-server .
[serve   ]
[watch   ]
[watch   ] > [email protected] watch /Users/lion/my_project/_eth/browser-solidity
[watch   ] > watchify src/index.js -dv -p browserify-reload -o build/app.js
[watch   ]
[onchange]
[onchange] > [email protected] onchange /Users/lion/my_project/_eth/browser-solidity
[onchange] > onchange build/app.js -- npm-run-all lint
[onchange]
[watch   ] WS server listening on  58651

  
  然后打开浏览器,在地址栏输入:http://127.0.0.1:8080,可以看到以下效果


  

4、使用 Browser-solidity 编译代码

  下面是一个简单的智能合约代码,输入任何数值,都加上2009。

pragma solidity 0.4.9;contract mshkDemo {

     function mshkadd(uint a) public returns (address, uint b) {  
        uint resutl = a+2009;
        return (msg.sender, resutl);
    }  
  }

    
  第一行是声明使用solidity合约的版本号,在0.4.9版本以前,声明版本号,在版本号前要加上^,如pragma solidity ^0.4.0solidity 中的智能合约是一种类似javascript的语言,所以在语法上很像。

  在本地打开Browser-solidity后,在右侧的Settings选项卡中,在下拉列表中,选择solidity的版本0.4.9


  
  点击Run选项卡,设置好Gas limitGas Price后,点击Create,能够创建一个测试实例,对代码进行调试。这时会在中间下面的窗口中,看到输出的信息。

  在mshkadd的右侧输入100,点击mshkadd,中间下面的窗口会继续出现新的信息,点击Details,能够看到输入的值和输出的值。


  
  点击Compile选项卡,然后点击Publish on,会看到提示层,然后点击Details,可以看到发布后的内容。


  
  发布以后,在上图弹出的层中我们可以看到编译后,能够通过web3部署的代码在WEB3DEPLOY中,代码如下:

var browser_mshk_sol_mshkdemoContract = web3.eth.contract([{"constant":false,"inputs":[{"name":"a","type":"uint256"}],"name":"mshkadd","outputs":[{"name":"","type":"address"},{"name":"b","type":"uint256"}],"payable":false,"type":"function","stateMutability":"nonpayable"}]);var browser_mshk_sol_mshkdemo = browser_mshk_sol_mshkdemoContract.new(   {
     from: web3.eth.accounts[0],
     data: '0x6060604052341561000c57fe5b5b60e48061001b6000396000f30060606040526000357c0100000000000000000000000000000000000000000000000000000000900463ffffffff168063ee64a56014603a575bfe5b3415604157fe5b60556004808035906020019091905050609e565b604051808373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020018281526020019250505060405180910390f35b6000600060006107d9840190503381925092505b509150915600a165627a7a7230582008b5b31ef21a9273b2728793378a837f0b7a4e7517a15041a358e33fbf45899e0029',
     gas: '300000'
   }, function (e, contract){
    console.log(e, contract);
    if (typeof contract.address !== 'undefined') {
         console.log('Contract mined! address: ' + contract.address + ' transactionHash: ' + contract.transactionHash);
    }
 })

  

4.1 将 Browser-solidity 编译后的合约部署到 Geth

  将上面的代码贴到geth javascript console中执行

> var browser_mshk_sol_mshkdemo = browser_mshk_sol_mshkdemoContract.new(    
   {
      from: web3.eth.accounts[0],
      data: '0x6060604052341561000c57fe5b5b60e48061001b6000396000f30060606040526000357c0100000000000000000000000000000000000000000000000000000000900463ffffffff168063ee64a56014603a575bfe5b3415604157fe5b60556004808035906020019091905050609e565b604051808373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020018281526020019250505060405180910390f35b6000600060006107d9840190503381925092505b509150915600a165627a7a7230582008b5b31ef21a9273b2728793378a837f0b7a4e7517a15041a358e33fbf45899e0029',
      gas: '300000'
    }, function (e, contract){
     console.log(e, contract);
     if (typeof contract.address !== 'undefined') {
          console.log('Contract mined! address: ' + contract.address + ' transactionHash: ' + contract.transactionHash);
     }
  })
  null [object Object]
  undefined
> null [object Object]
Contract mined! address: 0xf5f6d56c1c480d9235a4e538f2f2888c683b168c transactionHash: 0xf6203401fd0fe2237b7e53c021622272e7a9aa10d578ac0660c9145e7e700753

from表示调用智能合约的帐户,代码中取的是accounts[0]

data是编译后的代码,你的代码越长,这块的字符串越多

gas调用合约要扣除的gas单位,可以理解为以太币,gas和ether之间有个汇率,汇率受矿机的算率影响会有调整,在公网上,这些gas用于奖励给挖矿者

address表示已经部署智能合约的帐户地址,智能合约也相当于一个帐户

transactionHash的智能合约产生时的hash值,会永久保存到区块链条里面

  
  输入browser_mshk_sol_mshkdemo可以看到合约的一些信息

> browser_mshk_sol_mshkdemo{
  abi: [{
      constant: false,
      inputs: [{...}],
      name: "mshkadd",
      outputs: [{...}, {...}],
      payable: false,
      stateMutability: "nonpayable",
      type: "function"
  }],
  address: "0xf5f6d56c1c480d9235a4e538f2f2888c683b168c",
  transactionHash: "0xf6203401fd0fe2237b7e53c021622272e7a9aa10d578ac0660c9145e7e700753",
  allEvents: function(),
  mshkadd: function()
  }

  
  然后通过browser_mshk_sol_mshkdemo.mshkadd.call调用

> browser_mshk_sol_mshkdemo.mshkadd.call(10)TypeError: Cannot access member 'call' of undefined
    at <anonymous>:1:1

出现TypeError: Cannot access member 'call' of undefined的错误,是因为我们没有挖矿,在之前的章节提到过,如果停止挖矿,是不能进行转帐和智能合约的部署。

  
  重新启动挖矿,调用合约,就可以输出正确的值了。

> miner.start()> browser_mshk_sol_mshkdemo.mshkadd.call(10)
["0xbe323cc4fde114269a9513a27d3e985f82b9e25d", 2019]
> browser_mshk_sol_mshkdemo.mshkadd.call(550)
["0xbe323cc4fde114269a9513a27d3e985f82b9e25d", 2559]


以上是关于使用 Browser-solidity 在 Go-Ethereum 上进行简单的智能合约部署的主要内容,如果未能解决你的问题,请参考以下文章

智能合约开发环境搭建及Hello World合约

智能合约开发环境搭建及Hello World合约

solidity基础知识

Solidity编程 二 之Solidity安装

Solidity开发神器Remix

为啥要使用 Go 语言?Go 语言的优势在哪里?