[关闭]
@qidiandasheng 2018-08-11T20:34:46.000000Z 字数 3270 阅读 1900

编写一个简单的智能合约

区块链


基础知识

Geth

Geth是由以太坊基金会提供的官方客户端软件,用Go编程语言编写的。

Solidity

Solidity是一种合约导向式语言,可被应用于各种不同的区块链平台上,其主要开发者为Gavin Wood,Christian Reitwiessner,Alex Beregszaszi,Liana Husikyan,Yoichi Hirai和其他几位早期以太坊核心贡献者。Solidity 可使程式开发人员能在区块链上(例如以太坊)编写智能合约。

Solidity的语法概念最早是由Gavin Wood在2014年提出,后期则以Christian Reitwiessner所领导的以太坊团队Solidity接手开发。该语言是针对以太坊虚拟机(EVM)所设计的四种语言之一(其他的还有Serpent,LLL,Viper(实验中)和Mutan(已弃用))。

Web3.js

javascript库,可以用来与一个节点进行交互。 由于它是一个 JavaScript 库,您可以使用它来构建基于Web的dapps。

Remix

以太坊官方推荐的智能合约开发IDE,适合新手,可以在浏览器中快速部署测试智能合约。

搭建开发环境

geth安装

Mac下安装geth

  1. brew tap ethereum/ethereum
  2. brew install ethereum

geth使用

geth启动一个以太坊网络节点

  1. geth --datadir testNet --dev console 2>> test.log

启动成功界面如下图所示:

准备账户

查看账户:

  1. > eth.accounts

我们能看到返回一个账户数组,里面有一个默认账户,如:

  1. ["0xcc4164b8535eb7a9291486a1a5c982fd15c8b75d"]

查看余额:

  1. > eth.getBalance(eth.accounts[0])

我们默认取第一个账户的余额,容易看到这个默认分配的账号余额很多,为了更容易看到余额的变化,我们可以新创建一个账户。

创建账户:

personal.newAccount("password123456")

password123456为新账户的密码,这时我们输出账户数组就能看到两个账户了,输出新账户的余额eth.getBalance(eth.accounts[1]),我们能看到值为0,用新账户部署合约是需要有余额的,所以我们需要转一个以太币到新账户里面。

给新账户转币:

  1. eth.sendTransaction({from: '0xcc4164b8535eb7a9291486a1a5c982fd15c8b75d', to: '0x0a9961a0cde47c7e806a2874dc065cb33f6201c5', value: web3.toWei(1, "ether")})

然后你就能在你的test.log日志里看到挖矿记录了。再次查看余额eth.getBalance(eth.accounts[1])就能看到账户里有一个以太币了。

解锁账户:

在部署合约前需要先解锁账户,输入你的账号还有之前设置的密码,使用以下命令:

  1. personal.unlockAccount(eth.accounts[1],"password123456");

编写合约代码

部署合约

以下是一个简单的智能合约代码:

  1. pragma solidity 0.4.24;
  2. contract hello {
  3. function mutiply(uint a) returns (uint result) {
  4. return a*3;
  5. }
  6. }

这段代码就是传入一个数字,然后每次调用mutiply()的时候返回这个数字乘以3的结果。

把这段代码复制进入remix,这个就是一个浏览器ide。这个浏览器ide也可以安装在本地,这里面有教程教你具体怎么安装:github地址。点击Details即可部署智能合约代码,然后再弹框里找到以下这段代码,直接复制,修改里面的web.eth.accounts[0]web.eth.accounts[1]

然后在命令行里面输入以上代码:

  1. > var helloContract = web3.eth.contract([{"constant":true,"inputs":[{"name":"a","type":"uint256"}],"name":"mutiply","outputs":[{"name":"result","type":"uint256"}],"payable":false,"stateMutability":"nonpayable","type":"function"}]);
  2. undefined
  3. > var hello = helloContract.new(
  4. {
  5. from: web3.eth.accounts[1],
  6. data: '0x608060405234801561001057600080fd5b5060bb8061001f6000396000f300608060405260043610603f576000357c0100000000000000000000000000000000000000000000000000000000900463ffffffff168063f70d290d146044575b600080fd5b348015604f57600080fd5b50606c600480360381019080803590602001909291905050506082565b6040518082815260200191505060405180910390f35b60006003820290509190505600a165627a7a7230582002abd0500936c002b085e7f017e1aca58ca5294e5e4b27f0733e430604076c3f0029',
  7. gas: '4700000'
  8. }, function (e, contract){
  9. console.log(e, contract);
  10. if (typeof contract.address !== 'undefined') {
  11. console.log('Contract mined! address: ' + contract.address + ' transactionHash: ' + contract.transactionHash);
  12. }
  13. })

执行代码后输出以下结果表示合约部署成功,address就是合约地址:

  1. > null [object Object]
  2. Contract mined! address: 0xa29b84d8d302820f6ca1ebbf2f159ba12cf82b02 transactionHash: 0x15dd29aa336c8c4bf5868422c202f55153990a04426d5130e454957a87f56b07

调用合约

部署完合约后,我们就可以开始调用合约了,引号里面的值就是合约的地址,这里定义了一个Demo变量,然后通过Demo调用方法multiply()

  1. > Demo=eth.contract(hello.abi).at("0xa29b84d8d302820f6ca1ebbf2f159ba12cf82b02")
  2. {
  3. abi: [{
  4. constant: true,
  5. inputs: [{...}],
  6. name: "mutiply",
  7. outputs: [{...}],
  8. payable: false,
  9. stateMutability: "nonpayable",
  10. type: "function"
  11. }],
  12. address: "0xa29b84d8d302820f6ca1ebbf2f159ba12cf82b02",
  13. transactionHash: null,
  14. allEvents: function(),
  15. mutiply: function()
  16. }
  17. > Demo.mutiply(3)
  18. 9

参考

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

Remix 以太坊Solidity IDE搭建与初步使用

GETH客户端基本操作

添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注