智能合约开发solidity编程语言实例

 智能合约开发用solidity编程语言部署在以太坊这个区块链平台,本文提供一个官方实战示例快速入门,用例子深刻浅出智能合约开发,体会以太坊构建去中心化可信交易技术魅力。智能合约实际上是“执行合约条款的计算机交易协议”。区块链上的全部用户均可以看到基于区块链的智能合约。html

维基上说智能合约(英语:Smart contract )是一种旨在以信息化方式传播、验证或执行合同的计算机协议。智能合约容许在没有第三方的状况下进行可信交易。这些交易可追踪且不可逆转。智能合约概念于1994年由Nick Szabo首次提出。智能合同的目的是提供优于传统合同方法的安全,并减小与合同相关的其余交易成本。编程

因为区块链上的全部用户均可以看到基于区块链的智能合约。这也会致使包括安全漏洞在内的全部漏洞均可见,而且可能没法迅速修复。这样的攻击难以迅速解决。数组

插曲,2016年6月The DAOEther的漏洞形成损失5000万美圆,而开发者试图达成共识的解决方案。DAO的程序在黑客删除资金以前有一段时间的延迟。以太坊软件的一个硬分叉在时限到期以前完成了攻击者的资金回收工做。以太坊智能合约中的问题包括合约编程Solidity、编译器错误、以太坊虚拟机错误、对区块链网络的攻击、程序错误的不变性以及其余尚无文档记录的攻击。安全

部署智能合约的经典案例有:网络

1. 以太坊在其区块链上实施了一种近乎图灵完备的语言,这是一个突出的智能合约框架。
2. RootStock (RSK) 是一个智能合约平台,经过侧链技术链接到比特币区块链。 RSK兼容为以太坊创造的智能合约。数据结构

一个典型的智能合约的solidity语言编程示例或者叫实例以下一个委托投票系统,作了一些备注:app

官网示例原文:https://solidity.readthedocs.io/en/develop/solidity-by-example.html#possible-improvements框架

这个例子是最新的,主要用到了以太坊编程语言Solidity的一些特性。例子实现了一个投票智能合约即电子投票系统。解决的主要问题是如何分配合理的权限给正确的人,而且要防止被篡改。这个例子实现了如何去委托投票,整个投票计数过程是自动并且彻底透明。编程语言

功能上它首先为投票建立一个合约,发起者做为所谓的chairperson姑且叫主席来给每个独立的地址分配相应权限。每个参与投票者能够本身投票或者委托本身信任的人。这段代码最后运行结果会返回得票数最多的那个议案或者叫倡议。ide

pragma solidity ^0.4.22;

/// @title Voting with delegation.一个有委托功能的投票系统
contract Ballot {
    // This declares a new complex type which will 定义一个复杂类型
    // be used for variables later. 后面做为变量来使用
    // It will represent a single voter.表明一个投票人
    struct Voter {
        uint weight; // weight is accumulated by delegation weight在表明投票过程当中会累积
        bool voted;  // if true, that person already voted 若是值为true,表明这个投票人已经投过票
        address delegate; // person delegated to 投票人地址
        uint vote;   // index of the voted proposal 当前投票的索引
    }

    // This is a type for a single proposal.表明一份议案的数据结构 
    struct Proposal {
        bytes32 name;   // short name (up to 32 bytes) 议案的名称
        uint voteCount; // number of accumulated votes 议案接受的投票数
    }

    address public chairperson;  // 定义投票发起人

    // This declares a state variable that 
    // stores a `Voter` struct for each possible address. 这个状态变量存储了全部潜在投票人
    mapping(address => Voter) public voters;

    // A dynamically-sized array of `Proposal` structs. 定义动态数组存储议案
    Proposal[] public proposals;

    /// Create a new ballot to choose one of `proposalNames`. 传入议案名称来定义一个投票对象
    function Ballot(bytes32[] proposalNames) public {
        chairperson = msg.sender;
        voters[chairperson].weight = 1;

        // For each of the provided proposal names,
        // create a new proposal object and add it
        // to the end of the array. 按传入的议案名称建立一个议案,并加入到前面定义的议案数组
        for (uint i = 0; i < proposalNames.length; i++) {
            // `Proposal({...})` creates a temporary
            // Proposal object and `proposals.push(...)`
            // appends it to the end of `proposals`.建立一个临时议案对象,加入议案数组
            proposals.push(Proposal({
                name: proposalNames[i],
                voteCount: 0
            }));
        }
    }

    // Give `voter` the right to vote on this ballot.
    // May only be called by `chairperson`. 给投票人分配投票权限,这个操做只有主席才能够
    function giveRightToVote(address voter) public {
        // If the first argument of `require` evaluates
        // to `false`, execution terminates and all
        // changes to the state and to Ether balances
        // are reverted.
        // This used to consume all gas in old EVM versions, but
        // not anymore.
        // It is often a good idea to use `require` to check if
        // functions are called correctly.
        // As a second argument, you can also provide an
        // explanation about what went wrong.
        require(
            msg.sender == chairperson,
            "Only chairperson can give right to vote."
        );
        require(
            !voters[voter].voted,
            "The voter already voted."
        );
        require(voters[voter].weight == 0);
        voters[voter].weight = 1;
    }

    /// Delegate your vote to the voter `to`. 委托投票给另一个投票人
    function delegate(address to) public {
        // assigns reference 找出委托发起人,若是已经投票,终止程序
        Voter storage sender = voters[msg.sender];
        require(!sender.voted, "You already voted.");

        require(to != msg.sender, "Self-delegation is disallowed.");

        // Forward the delegation as long as
        // `to` also delegated.
        // In general, such loops are very dangerous,
        // because if they run too long, they might
        // need more gas than is available in a block.
        // In this case, the delegation will not be executed,
        // but in other situations, such loops might
        // cause a contract to get "stuck" completely.
        while (voters[to].delegate != address(0)) {
            to = voters[to].delegate;

            // We found a loop in the delegation, not allowed. 发起人、委托人不能是同一个,不然终止程序
            require(to != msg.sender, "Found loop in delegation.");
        }

        // Since `sender` is a reference, this
        // modifies `voters[msg.sender].voted`   标识发起人已经投过票
        sender.voted = true;
        sender.delegate = to;
        Voter storage delegate_ = voters[to];
        if (delegate_.voted) {
            // If the delegate already voted,
            // directly add to the number of votes 投票成功,投票总数加上相应的weight
            proposals[delegate_.vote].voteCount += sender.weight;
        } else {
            // If the delegate did not vote yet,
            // add to her weight. 若是还没投票,发起人weight赋值给委托人
            delegate_.weight += sender.weight;
        }
    }

    /// Give your vote (including votes delegated to you)
    /// to proposal `proposals[proposal].name`.投票给某个议案
    function vote(uint proposal) public {
        Voter storage sender = voters[msg.sender];
        require(!sender.voted, "Already voted.");
        sender.voted = true;
        sender.vote = proposal;

        // If `proposal` is out of the range of the array,
        // this will throw automatically and revert all
        // changes.
        proposals[proposal].voteCount += sender.weight;
    }

    /// @dev Computes the winning proposal taking all
    /// previous votes into account.找出投票数最多的议案
    function winningProposal() public view
            returns (uint winningProposal_)
    {
        uint winningVoteCount = 0;
        for (uint p = 0; p < proposals.length; p++) {
            if (proposals[p].voteCount > winningVoteCount) {
                winningVoteCount = proposals[p].voteCount;
                winningProposal_ = p;
            }
        }
    }

    // Calls winningProposal() function to get the index
    // of the winner contained in the proposals array and then
    // returns the name of the winner
    function winnerName() public view
            returns (bytes32 winnerName_)
    {
        winnerName_ = proposals[winningProposal()].name;
    }
}

 

若是这个代码基本可以看明白,那应该是能够直接实战开启以太坊区块链的学习进程了。

安利个教程能够经过在线编程环境实战学习:以太坊DApp实战开发

相关文章
相关标签/搜索