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

这篇具有很好参考价值的文章主要介绍了北大肖臻老师《区块链技术与应用》系列课程学习笔记[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,这个后面会解释是什么意思。

北大肖臻老师《区块链技术与应用》系列课程学习笔记[21]以太坊-智能合约-1
图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所示。

北大肖臻老师《区块链技术与应用》系列课程学习笔记[21]以太坊-智能合约-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函数。

北大肖臻老师《区块链技术与应用》系列课程学习笔记[21]以太坊-智能合约-1
图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,表明这个调用是失败的,但发起调用的这个函数不会抛出异常,可以继续执行。

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

3.代理调用 delegatecall()

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

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


 文章来源地址https://www.toymoban.com/news/detail-449844.html

到了这里,关于北大肖臻老师《区块链技术与应用》系列课程学习笔记[21]以太坊-智能合约-1的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

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

    智能合约-1 智能合约-2 智能合约-3 智能合约-4 网上竞拍第二版:由投标者自己取回出价 (1)存在的问题         重入攻击,如果有黑客写了一个如下方程序会怎么样?         这个hack_bid跟前面的那个黑客合约hack_bid合约是一样的,通过调用拍卖bid函数参与竞拍,ha

    2024年03月11日
    浏览(50)
  • 北大肖臻老师《区块链技术与应用》系列课程学习笔记[17]以太坊-GHOST协议

    目录 一、以太坊的出块时间及可能带来的问题         1.以太坊的出块时间         2.以太坊与比特币系统的平均出块时间对比         3.带来的问题 二、GHOST协议         1.GHOST协议的核心思想         2.GHOST协议的缺陷         3.改进后的GHOST协议    

    2024年02月09日
    浏览(35)
  • 北大肖臻老师《区块链技术与应用》系列课程学习笔记[15]以太坊-交易树和收据树

     目录 一、以太坊中的三种树 二、状态树、交易树和收据树的区别 三、交易树和收据树的用途         1.交易树和收据树的用途         2.如何实现复杂的查询操作         3.以太坊中Bloom Filter的用途 四、以太坊的运行过程        在以太坊中,存在三种基于树的

    2024年02月05日
    浏览(39)
  • 《区块链技术与应用》北大肖臻老师——课程笔记【6-8】

    提示:以下内容只是个人在学习过程中记录的笔记,图片均是肖老师课程的截图,可供参考。如有错误或不足之处,请大家指正。 比特币网络传播的工作原理(the BitCoin network): 比特币工作在应用层(application layer),底层是P2P的overlay network(覆盖网络)。 比特币的 P2P网络

    2024年01月18日
    浏览(34)
  • 《区块链技术与应用》北大肖臻老师——课程笔记【19-20】

    提示:以下内容只是个人在学习过程中记录的笔记,图片均是肖老师课程的截图,可供参考。如有错误或不足之处,请大家指正。 Block chain is secured by mining. 对于基于工作量证明的系统来说,挖矿是保障区块链安全的一个重要手段。 比特币的挖矿算法总的来说比较成功,没有

    2024年02月09日
    浏览(27)
  • 《区块链技术与应用》北大肖臻老师——课程笔记【21-23】

    提示:以下内容只是个人在学习过程中记录的笔记,图片均是肖老师课程的截图,可供参考。如有错误或不足之处,请大家指正。 权益证明proof of stake 比特币和 以太坊目前都是使用基于工作量的证明,这种共识机制受到一个普遍的批评——浪费电。 Y轴是TWH=Terawatt hours 10的

    2023年04月08日
    浏览(32)
  • 《区块链技术与应用》北大肖臻老师——课程笔记【11-12】

    提示:以下内容只是个人在学习过程中记录的笔记,图片均是肖老师课程的截图,可供参考。如有错误或不足之处,请大家指正。 1. 转账交易时如果接收者不在线(没有连接到比特币网络上)怎么办? 转账交易不需要接收者在线,这个交易只是在区块链上记录一下,把发送

    2024年01月22日
    浏览(36)
  • 北大肖臻老师<<区块链技术>>笔记1

    课程的大纲 密码学基础 比特币的数据结构 共识协议和系统实现 挖矿算法和难度调整 比特币的脚本 软分叉和硬分叉 匿名和隐私保护 以太坊是后面的 首先是密码学基础的学习: crypto-currency(虚拟货币) 是不加密的,区块链上所有的教以都是公开的。其中有转账金额和地址。

    2024年02月02日
    浏览(29)
  • [北大肖臻-区块链技术与应用笔记]第八节课——BTC 脚本

    比特币系统中使用的脚本语言很简单, 唯一能访问的内存空间就是一个栈 ,这点和通用脚本语言的区别很大。 这个交易有一个输入和两个输出,其中一个输出已经被花出去了,另一个没有被花出去。 输入脚本 输入脚本包含两个操作,分别将两个很长的数压入栈中。 输出脚

    2024年01月21日
    浏览(57)
  • [北大肖臻-区块链技术与应用笔记]第八节课——BTC 分叉

    state fork 如果两个节点差不多同时挖到一个区块,这两个区块都是挂在当前的区块上的,不同节点先收到的区块不同,就会各自沿着先收到的区块往下扩展,这种时候就会出现临时性的分叉,称为 state fork ,即由于对区块链当前的状态有意见分歧而产生的分叉。 分叉攻击(

    2024年02月08日
    浏览(22)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包