北大肖臻老师《区块链技术与应用》系列课程学习笔记[21]以太坊-智能合约-1

目录

一、什么是智能合约

二、智能合约的代码结构

        1.Solidity语言

        2.bid函数

        3.fallback()函数

 二、外部账户如何调用智能合约

三、一个合约如何调用另一个合约中的函数

        1.直接调用

        2.使用address类型的call()函数

        3.代理调用 delegatecall()

        智能合约是以太坊的精髓,也是以太坊和比特币一个最大的区别。

一、什么是智能合约

1.智能合约的本质是运行在区块链上的一段代码,代码的逻辑定义了智能合约的内容。
2.智能合约的账户保存了合约当前的运行状态
(1)balance:当前余额;
(2)nonce:交易次数;
(3)code:合约代码;
(4)storage:存储,数据结构一是一颗MPT;
3.Solidity是智能合约最常用的语言,语法上与JavaScript很接近。

二、智能合约的代码结构

pragma solidity ^0.4.21;
contract SimpleAuction {
    address public beneficiary;//拍卖受益人
    uint public  auctionEnd;//结束时间
    address public highestBidder;//当前的最高出价人
    mapping( address => uint) bids;//所有竞拍者的出价
    address[] bidders;//所有竞拍者

    //需要记录的事件
    event HighestBidIncreased(address bidder,uint amount);
    event -Pay2Beneficiary( address - winner , uint amount);

    //以受益者地址 `_beneficiary` 的名义,
    //创建一个简单的拍卖,拍卖时间为 `_biddingTime` 秒。
    constructor(uint _biddingTime,address _beneficiary
        )public {
        beneficiary = _beneficiary;
        auctionEnd = now + biddingTime;
    }

    //对拍卖进行出价,随交易一起发送的ether与之前已经发送的ether的和为本次出价。
    function bid() public payable {…
    }

    //使用withdraw模式
    //由投标者自己取回出价,返回是否成功
    function withdraw() public returns (bool) {…
    }

    //结束拍卖,把最高的出价发送给受益人
    function pay2Beneficiary() public returns (bool) {…
    }
}

1.Solidity语言

        Solidity是面向对象的编程语言,这里的contract类似于C++当中的类(class),定义了很多状态变量。Solidity是强类型语言,大部分跟普通的编程语言(如C++等)比较接近,如:uint,即unsigned int(无符号整数)。address类型是Solidity语言所特有的。

        上述代码段中的event事件,是用来记录日志的。第一个事件是HighestBidIncreased,拍卖的最高出价增加了,代码中是一个网上拍卖的例子,记录一下最新高价的参数(address bidder),金额是amount;第二个事件是Pay2Beneficiary,参数是赢得拍卖的人的地址及最后出价amount。

        Solidity语言跟其他普通编程语言相比,有一些特别之处。如:mapping,mapping是一个哈希表,保存了从地址到unit的一个映射。Solidity语言中哈希表不支持遍历,如果想遍历哈希表里的所有元素,需要想办法记录哈希表中有哪些元素,用bidders数组记录下来,Solidity语言中的数组可以是固定长度的,也可以是动态改变长度的。上述代码是一个动态改变长度的数组,如果要在数组里增加一个元素,就用push操作,即bidders.push(bidder):新增加一个出价人在数组的末尾;想知道这个数组有多少个元素,可以用bidders.length;如果是固定长度的数组,就要写明数组的长度,比如address[1024],这个就是长度为1024的数组。

        Solidity语言中定义构造函数有两种方法,构造函数只能有一个。一种方法就是像c++构造函数一样,定一个与contract同名的函数,这个函数可以有参数,但是不能有返回值。实际上新版本Solidity语言更推荐用这里使用的方法:用一个constructor来定义一个构造函数,这个构造函数只有在合约创建的时候会被调用一次。

        最后是三个成员函数,三个函数都是public,说明其他账户可以调用这些函数,bid这个函数,这里标志有一个payable,这个后面会解释是什么意思。

图1-1

2.bid函数

        在bid函数中,有一个payable(另外两个函数都没有),以太坊中规定如果这个合约账户要能接收外部转账的话,那么必须标注成payable,这个例子中bid函数是什么意思?这是网上拍卖的合约,bid函数是用来进行竞拍出价的。比如说你要参与拍卖,你说你出100个以太币,那么就调用合约当中的bid函数,所以拍卖规则是,调用bid函数时要把拍卖的出价100个以太币也发送过去,存储到这个合约里,锁定到拍卖结束。避免有人凭空出价,说出1万个以太币,实际上你没那么多钱,所以要拍卖的时候,要把你发的价钱放到合约里锁定起来,所以bid函数要有能够接收外部转账的能力,所以才标注一个payable。withdraw函数就没有payable,withdraw就是拍卖结束了,出价最高的那个人赢得了拍卖,其他人没有拍到想要的东西,可以调用withdraw把自己当初出的价钱,就是原来bid的时候锁定在智能合约里的以太币再取回来,因为这个的目的不是为了真的转账,不是要把钱转给智能合约,而仅仅是调用withdraw函数把当初锁定在智能合约里的那一部分钱取回来,所以没必要弄payable。图2-1中的交易就属于不需要payable。

3.fallback()函数

function()public [payable]{
……
}

        这个函数既没有参数也没有返回值,而且也没有函数名,是个匿名函数,fallback这个关键字也没有出现在这个函数名里。调用这个合约的时候,A调用B这个合约,然后要在转账交易的data域说明你调用的是B当中的哪个函数。如果A给合约B转账了一笔钱,没有说明调用的是哪个函数,data域是空的,那么这个时候就是调用这个fallback()函数,没有别的函数可调了,就调他。还有一种情况是要调的函数不存在,在那个data域里,你说要调这个函数,而实际这个合约当中没有这个函数,那也是调用这个fallback()函数,这就是为什么这个函数没有参数也没有返回值,因为他没法提供参数。

        对于fallback()函数来说,也可能需要标注payable关键字,如果fallback()函数需要有接收转账的能力的话,也需要写成是payable,一般情况下,都是写上payable的,如果合约账户没有任何函数标识为payable,包括fallback()函数也没有标识成payable,那么这个合约没有任何能力接受外部的转账。就是如果这个合约没有fallback()函数或者是有fallback()函数 但是没有写payable,那么其他人往这个合约里转一笔钱,别的都不说,data域是空的就会引发异常。

        fallback()函数和payable都是在合约定义的时候写的,我给你转账时候不用写payable,也不用写fallback(),如果转账的时候,别的什么都不写,没有调用任何一个函数,那么就自动调用这个fallback()函数。

        fallback()函数不是必须定义的,一个合约可以没有fallback()函数,如果没有fallback()函数的话,出现前面说的几种情况,就会抛出异常。比如给一个合约转账,没有说调哪个函数,那个合约也没有定义fallback()函数,那么这个转账就是错误的,就会引发错误处理。另外只有合约账户才有这些东西,外部账户跟这个都没有关系,外部账户都没有代码。

        另外,转账金额可以是0,但汽油费是要给的,这是两码事,转账金额是给收款人的,汽油费是给发布这个区块的矿工的,如果汽油费不给的话,矿工不会把你这个交易打包发布到区块链。

 二、外部账户如何调用智能合约

        调用智能合约其实跟转账是类似的。如A发起一个交易转账给B,如果B是一个普通的账户,那么这就是一个普通的转账交易,与比特币当中的转账交易是一样的。如果B是一个合约账户的话,那么这个转账实际上是发起一次对B这个合约的调用,那么具体是调用合约中的哪个函数,是在数据域data域中说明的,如图2-1所示。

图2-1

三、一个合约如何调用另一个合约中的函数

1.直接调用

contract A {
    event LogCallFoo(string str);
    function foo(string str) returns (uint){
        emit LogCallFoo( str) ;
        return 123;
    }
}

contract B {
    uint ua;
    function cal1AFooDirectly(address addr) public{
        Aa = A(addr) ;
        ua = a.foo("call foo directly");
    }
}

        有A,B两个合约,A这个合约就只是写成log,event定义事件LogCallFoo,emit LogCallFoo():用emit这个操作来调用这个事件,作用就是写一个log,对于程序的运行逻辑是没有影响的。在B合约中,函数参数是一个地址(A合约的地址),然后把这个地址转换成A这个合约的一个实例,然后调用其中的foo这个函数。

        以太坊中规定一个交易只有外部账户才能够发起,合约账户不能自己主动发起一个交易。这个例子当中需要有一个外部账户调用了合约B当中的这个callAFooDirectly函数,然后这个函数再调用合约A当中的foo函数。

图3-1

 2.使用address类型的call()函数

contract c {
    function callAFooByCall(address addr) public returns (bool){
        bytes4 funcsig = bytes4(keccak256("foo(string)"));
        if ( addr.call(funcsig,"call foo by func call"))
            return true;
        return false;
    }
}

        funcsing:要调用函数的签名,然后后面跟的是调用的参数。该方法与直接调用方法相比,区别是对于错误处理的不同:直接调用方法,如果调用的合约在执行过程中出现错误,那么会导致发起调用的合约也跟着一起回滚,如果在直接调用方法中A在执行过程出现异常,B这个合约也跟着一起出错。而address.call()这种形式,如果在调用过程中,被调用的合约抛出异常,那么这个call函数会返回false,表明这个调用是失败的,但发起调用的这个函数不会抛出异常,可以继续执行。

图3-2

3.代理调用 delegatecall()

        与address.call()方法基本上是一样的,一个主要的区别是delegatecall不需要切换到被调用的合约的环境中去执行,而是在当前合约环境中执行就可以了,比如就用当前账户的账户余额存储之类的,如图3-3所示。

图3-3


 

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以太坊是一个平台,它上面提供各种模块让用户来搭建应用,如果将搭建应用比作造房子,那么以太坊就提供了墙面、屋顶、地板等模块,用户只需像搭积木一样把房子搭起来,因此在以太坊上建立应用的成本和速度都大大改善。具体来说,以太坊通过一套图灵完备的脚本语言(Ethereum Virtual Machinecode,简称EVM语言)来建立应用,它类似于汇编语言。我们知道,直接用汇编语言编程是非常痛苦的,但以太坊里的编程并不需要直接使用EVM语言,而是类似C语言、Python、Lisp等高级语言,再通过编译器转成EVM语言。上面所说的平台之上的应用,其实就是合约,这是以太坊的核心。合约是一个活在以太坊系统里的自动代理人,他有一个自己的以太币地址,当用户向合约的地址里发送一笔交易后,该合约就被激活,然后根据交易中的额外信息,合约会运行自身的代码,最后返回一个结果,这个结果可能是从合约的地址发出另外一笔交易。需要指出的是,以太坊中的交易,不单只是发送以太币而已,它还可以嵌入相当多的额外信息。如果一笔交易是发送给合约的,那么这些信息就非常重要,因为合约将根据这些信息来完成自身的业务逻辑。合约所能提供的业务,几乎是无穷无尽的,它的边界就是你的想象力,因为图灵完备的语言提供了完整的自由度,让用户搭建各种应用。白皮书举了几个例子,如储蓄账户、用户自定义的子货币等。 2013年年末,以太坊创始人Vitalik Buterin发布了以太坊初版白皮书,启动了项目。2014年7月24日起,以太坊进行了为期42天的以太币预售。2016年初,以太坊的技术得到市场认可,价格开始暴涨,吸引了大量开发者以外的人进入以太坊的世界。中国三大比特币交易所之二的火币网及OKCoin币行都于2017年5月31日正式上线以太坊。 [1] 自从进入2016年以来,那些密切关注数字货币产业的人都急切地观察着第二代加密货币平台以太坊的发展动向。作为一种比较新的利用比特币技术的开发项目,以太坊致力于实施全球去中心化且无所有权的的数字技术计算机来执行点对点合约。简单来说就是,以太坊是一个你无法关闭的世界计算机。加密架构与图灵完整性的创新型结合可以促进大量的新产业的出现。反过来,传统行业的创新压力越来越大,甚至面临淘汰的风险。比特币网络事实上是一套分布式的数据库,而以太坊则更进一步,她可以看作是一台分布式的计算机:区块链是计算机的ROM,合约是程序,而以太坊的矿工们则负责计算,担任CPU的角色。这台计算机不是、也不可能是免费使用的,不然任何人都可以往里面存储各种垃圾信息和执行各种鸡毛蒜皮的计算,使用它至少需要支付计算费和存储费,当然还有其它一些费用。最为知名的是2017年初以摩根大通、芝加哥交易所集团、纽约梅隆银行、汤森路透、微软、英特尔、埃森哲等20多家全球top金融机构和科技公司成立的企业以太坊联盟。而以太坊催生的加密货币以太币近期又成了继比特币之后受追捧的资产。  智能合约的潜在应用很多。彭博社商业周刊称它是“所有人共享但无法篡改的软件”。更高级的软件有可能用以太坊创建网络商店。区块链程序以太坊可以用来创建去中心化的程序、自治组织和智能合约,据纽约时报的报导,在2016年5月已经有数十个可用的程序。预期的应用目标涵盖金融、物联网、农田到餐桌(farm-to-table)、智能电网、体育,菠菜等。去中心化自治组织有潜力让许多原本无法运行或成本过高的营运模型成为可能。较知名的应用有:去中心化创业投资:The DAO用以太币资金创立,目标是为商企业和非营利机构创建新的去中心化营业模式、The Rudimental让独立艺术家在区块链上进行群众募资。社会经济平台:Backfeed。去中心化预测市场:Augur。物联网:Ethcore(一间以太坊公司)研发的客户端、Chronicled(一间区块链公司)发表了以太坊区块链的实物资产验证平台;芯片公司、物理IP创建者和生产者可以用植入的蓝牙或近场通信进行验证。Slock.It开发的智能锁可以在付费后自动打开,让用户在付费后可以帮电动车充电、或是打开租屋的房门。虚拟宝物交易平台:FreeMyVunk。版权授权:Ujo Music平台让创作人用智能合约发布音乐,消费者可以直接付费给创作人。伊莫珍·希普用此平台发布了一首单曲。智能电网:TransActive Grid让用户可以和邻居买卖能源。去中心化期权市场:Etheropt。钉住汇率的代币:DigixDAO提供与黄金挂钩的代币,在2016年四月正式营运。Decentralized Capital提供和各种货币挂钩的代币。移动支付:Everex让外劳汇款回家乡。客户端软件以太坊的两个主要的客户端软件是Geth和Parity。企业软件企业软件公司也正测试用以太坊作为各种用途。已知有兴趣的公司包括微软、
智能合约是一种在区块链上执行的自动化合约。Python是一种常用的编程语言,也可以用于编写智能合约。在Python中,可以使用一些库来编写智能合约,例如Web3.py和Solidity.py。下面是一个使用Web3.py库编写智能合约的示例: ```python from web3 import Web3 # 连接到以太坊节点 w3 = Web3(Web3.HTTPProvider('https://mainnet.infura.io/v3/your-infura-project-id')) # 加载智能合约ABI contract_abi = [ { "constant": False, "inputs": [ { "name": "x", "type": "uint256" } ], "name": "set", "outputs": [], "payable": False, "stateMutability": "nonpayable", "type": "function" }, { "constant": True, "inputs": [], "name": "get", "outputs": [ { "name": "", "type": "uint256" } ], "payable": False, "stateMutability": "view", "type": "function" } ] # 部署智能合约 contract_address = '0x1234567890abcdef1234567890abcdef12345678' contract = w3.eth.contract(address=contract_address, abi=contract_abi) # 调用智能合约方法 transaction = contract.functions.set(42).buildTransaction({ 'from': w3.eth.accounts[0], 'gas': 100000, 'gasPrice': w3.toWei('1', 'gwei'), 'nonce': w3.eth.getTransactionCount(w3.eth.accounts[0]) }) signed_transaction = w3.eth.account.signTransaction(transaction, private_key='your-private-key') transaction_hash = w3.eth.sendRawTransaction(signed_transaction.rawTransaction) transaction_receipt = w3.eth.waitForTransactionReceipt(transaction_hash) # 获取智能合约状态 result = contract.functions.get().call() print("智能合约状态:", result) ``` 请注意,上述示例中的合约ABI和地址是虚构的,您需要根据您自己的合约来替换它们。此外,您还需要替换连接到以太坊节点的URL和私钥。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值