21026 large

058 孤荷凌寒从零开始学区块链第 58 天 DAPP013

941xue · 于 发布 · 181 次阅读

孤荷凌寒自学第144天区块链第058天DAPP013
【主要内容】
今天继续学习实战,做一个真正的依托于智能合约的DAPP。今天继续独立完善一个众筹的合约。共耗时38分钟。
(此外整理作笔记花费了约62分钟)
详细学习过程见文末学习过程屏幕录像。

【解决了不能部署合约的问题】
仿佛只是灵光一闪,我突然想到会不会是solidity解释器的版本选择错误造成的呢?
于是凭借感觉,将现在的版本要求:
^0.4.7
修改为:
^0.4.25
当然有一些语法结构也给修改了一下。
现在的合约如下:

pragma solidity ^0.4.25;

contract ZongChouCeshi{
    address owner; //此变量记录下合约部署者(拥有者)的地址
    uint public goal; //要众筹的目标金额(当然这儿指的是eth)
    uint public endtime; //是指此众筹的时长(是一个无符号整数,单位是秒),如果在从部署合约之日加上此天数后的日期期限内达到目的金额,则说明众筹目标达到,合约部署者可以取走所有众筹金额,否则就由参与众筹者自己退回自己那份钱。
    uint public total=0; //这是实际众筹到的金额
    uint public giftcount=0; //记录下有多少节点参与了,不知道怎么检查去重。对映射类型的变量了解不够

    address[] public alladd=new address[](1); //这儿必须写上(1),不然报错如下:
    //Type function (uint256) returns (address[] memory) is not implicitly convertible to expected type address[] storage ref.

    mapping(address=>uint) gift; //每个参与众筹的节点地址实际捐赠的金额,即地址对应的金额

    //下面是合约的建构函数(只在部署时执行一次)
    //部署合约时,手动指定众筹的目标金额与众筹的时长(总数)
    constructor() public{
        owner = msg.sender;
        goal = 50000000;
        endtime = now + 100800;
        //变量 now 将返回当前的unix时间戳(自1970年1月1日以来经过的秒数)。
        //来自博文:https://www.jianshu.com/p/4b8e943ce7f2
    }

    //调用合约的用户节点参与众筹,向合约地址转账一定数量的eth
    //这个函数给我的疑惑是,没有体现(像自己发的代币合约中的那样)登记eth余额双方的变化
    //初步理解是:
    //因为使用的代币正是eth本身,因此当调用此合约的节点在进行转移eth操作时,eth网络本身已经记录并广播完成了eth代币余额在两个节点之间的移动
    function donate(uint intvalue) payable public{
        require(now < endtime);
        require(total < goal);
        require(intvalue >=0.3 * (10 ** 18));
        //每个节点至少捐赠众筹0.3个eth
        if(gift[msg.sender]==0){ //判断此节点是否之前已经捐过款项
            giftcount+=1; //统计已经有多少节点参与捐助,而且要去重节点。
            //alladd[alladd.length++] = msg.sender; //在地址数组中记录下当前地址,
            alladd.push(msg.sender);
        }
        gift[msg.sender] += msg.value; //在gift表中记录下当前节点增加了一定的捐赠金额(这意味着一个节点可以多次调用合约进行捐赠)
        total += msg.value; //让目前的实际众筹余额增加            

    }

    //如果众筹成功,即在规定的时间内,众筹了大于等于goal预设数量金额的eth,那么证明成功,则合约的部署节点可以取走这笔eth到合约部署节点的账户上。
    function draw() public{
        //首先要保证本次调用合约及调用本函数的节点是合约部署者。
        //确实判断众筹目标已经达成。
        require(msg.sender == owner);
        require(total >= goal);
        bool isok=owner.send(address(this).balance); //执行转账,从合约地址转到部署合约的节点地址
        //return isok;

        //eth代币的转账书写方法是非常简洁的:
        //接收转账金额的节点.transfer(转账金额)
        //这样指定【转账金额】的eth就会转移给【接收转账金额的节点】
    }

    //如果众筹失败,则参与众筹的节点可以调用合约的本方法来取回自己的捐赠
    function withdraw() public{
        //只有当众筹时间已过,本函数才可被调用
        //确定判断众筹已经失败。
        require(msg.sender == owner);
        require(total > goal);
        uint amount = gift[msg.sender]; //获取当前调用合约的节点之前总共捐赠的eth的总金额
        total -= amount; //从总金额中减少当前调用合约节点准备取回的eth金额
        gift[msg.sender] = 0; //在gift表中,登记当前调用合约节点新的捐赠金额为0
        bool isok=address(msg.sender).send(amount); //当前调用合约的节点之前总共捐赠的eth的总金额转移回当前调用合约的节点            

    }

    //查询当前已筹集资金总额:
    function getTotal() public constant returns(uint){
        return total;
    }

    //查询当前合约节点自己的余额(这实际上就是total的数量,不过额外增加一个函数来查询)
    function getContractBalance() public constant returns(uint){
        return address(this).balance;
    }

    //查询截止日期,返回的是结束 日期那天的时间戳的一个无符号整数(32位的),需要调用此合约函数的调用方自己用前端 语言来进行处理
    function getEndTimeStamp() public constant returns(uint){
        return endtime;
    }

    //查询当前调用合约的节点已经总共捐助了多少eth
    function getGiftBalance() public constant returns(uint){
        return gift[msg.sender];
    }

    //查询已经有多少人捐助
    function getGiftCount() public constant returns(uint){
        return giftcount; 
    }

    //查询指定的节点已经总共捐助了多少eth
    function getGiftBalanceFrom(address a) public constant returns(uint){
        return gift[a];
    }

    //返回已参与节点的数组
    function getalladdress() public constant returns(address[]){
        return alladd;
    }

}

特别值得一提的是,gas的设置的第二个参数居然设置为0才行,总之是成功部署了。
并且在remix在线编辑器中测试还通过了合约交互的测试,只有合约中的函数:
donate失败。

【发现合约中的donate函数的写法不知道怎样用js调用并与之交互】
通过js代码像调用 合约中其它方法那样调用 donate()函数总是提示错误。
然后参考其它众筹合约代码,发现别的合约都是直接通过回退函数来直接接受众筹eth的。而当前使用的智能合约却专门写了一个donate函数,反而无法设置要发送的eth数量。
参考的博文是:
https://my.oschina.net/u/3790537/blog/1808769
https://www.jianshu.com/p/84cf2f12daed

【用JS处理时间戳】
智能合约可以向与之交互的合约调用方返回一个Uint类型的数值,solidity中的时间戳就是这样表示的,但这个uint类型数值的单位是:秒。
js中也有时间戳概念,但它的时间戳默认单位是:毫秒。
于是,从以下博文中获得帮助,今天 改写的获取众筹结束时间的代码如下:
https://www.cnblogs.com/jf-67/p/8007008.html

            //查询本次授权的结束日期和时间
            function getEndTimeStamp(){
                try{
                    cc.getEndTimeStamp(function(error, result){  //https://www.jianshu.com/p/15ff9da4dd8d
                        if(!error)
                    {
                        t=formatTime(result * 1000,"Y-M-D h:m:s");
                        $("#endtime").html("本次众筹截止时间:" + t);
                       //alert(result);
                    }
                    else{
                        //alert(error);
                        $("#endtime").html('获取出错:' + error);
                    }});
                    //return "ok"
                    $("#endtime").html('正在获取。。。');
                }catch(err){
                    alert("使用授权代币转移时出错 :" + err)
                    return err;
                }
            }

【今天修改完成的前端Html代码】

<!DOCTYPE html>
<html>
<head>
    <meta charset="UTF-8">
    <title>我要众筹</title>
    <script language="javascript" type="text/javascript" src="jquery.min.js"></script>
    <script language="javascript" type="text/javascript" src="web3.min.js"></script>
    <!-- 1. Include cryptozombies_abi.js here -->
    <script language="javascript" type="text/javascript" src="zongchou_abi.js"></script>

    <script>
// 格式化日期,如月、日、时、分、秒保证为2位数
function formatNumber (n) {
    n = n.toString()
    return n[1] ? n : '0' + n;
}
// 参数number为毫秒时间戳,format为需要转换成的日期格式
function formatTime (number, format) {
    let time = new Date(number)
    let newArr = []
    let formatArr = ['Y', 'M', 'D', 'h', 'm', 's']
    newArr.push(time.getFullYear())
    newArr.push(formatNumber(time.getMonth() + 1))
    newArr.push(formatNumber(time.getDate()))

    newArr.push(formatNumber(time.getHours()))
    newArr.push(formatNumber(time.getMinutes()))
    newArr.push(formatNumber(time.getSeconds()))

    for (let i in newArr) {
        format = format.replace(formatArr[i], newArr[i])
    }
    return format;
}

            //声明一些钱包地址:
            //下面一行定义的是部署合约的节点(创世节点)的信息,公钥
            var wallet_address="0x5227C3EF48B5A1bcF784593e46D9579D26a3b592"; //狐狸钱包的公钥,就是钱包地址,是eth网络上的一个节点。
            //下面一行定义的是节点2的信息
            var w2add="0xe2d6c2f289c53B5aEA44C47293Ba179a3bfa21f0"; //公钥

            //下面一行定义的是节点3 的信息
            var w3add="0xb40599fB0366DCf0ffe86677b005b3f20Dfa29aE"; //公钥

            //下面一行定义的是节点4 的信息
            var w4add="0x70c8461366d5368B1E79CBFc2Acf4ba56C745977"; //公钥



            // 2. Start code here
            var cc;
            var web3;

            function startApp() {
                try {
                    var cryptoZombiesAddress = "0x410594034bd867416011937e86d72d398afede83"; //要连接的合约地址
                    var ccc=web3.eth.contract(zongchouABI);
                    cc=ccc.at("0x410594034bd867416011937e86d72d398afede83"); //https://www.cnblogs.com/tinyxiong/p/9046626.html
                    //cc =new web3.eth.contract(cryptozombiesABI, cryptoZombiesAddress); //如果是另一个版本可能还得加上new关键字。
                    alert(typeof cc);

                } catch (err) {
                    alert(err);
                }
                alert("加载成功");

            }

            //async () => 

        //现在这种通过we3.min.js来加载钱包连接的方法,在metamask钱包和麦子钱包中都测试通过。
        window.addEventListener('load',function() {
        try{
            if (typeof web3 !== 'undefined') {
                web3 = new Web3(web3.currentProvider);
                startApp();
            } else {
                //$('#app_loading').hide();
                //alert(jQuery.i18n.prop('lrn_error_alert'));
                //mathWallet.closePage();
                alert("这儿没有钱包环境。");
            }

        }catch(err){
            alert(err);
        }
        });


            //----------------下面是自定义的与合约交互的函数-------------------------
            //--查询总金额---
            function gettotal(){
                try{
                    cc.getTotal(function(error, result){  //https://www.jianshu.com/p/15ff9da4dd8d
                    if(!error)
                    {
                        $("#total").html("已获得众筹总金额:" + result);
                       //alert(result);
                    }
                    else{
                        //alert(error);
                        $("#total").html('获取出错:' + error);
                    }});
                    //return "ok"
                    $("#total").html('正在获取。。。');
                }catch(err){
                    $("#total").html('出错 :' + err);
                }
            }

            //--查询总金额---
            function gettotal2(){
                try{
                    cc.getContractBalance(function(error, result){  //https://www.jianshu.com/p/15ff9da4dd8d
                    if(!error)
                    {
                        $("#total2").html("合约地址代币余额:" + result);
                       //alert(result);
                    }
                    else{
                        //alert(error);
                        $("#total2").html('获取出错:' + error);
                    }});
                    //return "ok"
                    $("#total2").html('正在获取。。。');
                }catch(err){
                    $("#total2").html('出错 :' + err);
                }
            }


            //查询本次授权的结束日期和时间
            function getEndTimeStamp(){
                try{
                    cc.getEndTimeStamp(function(error, result){  //https://www.jianshu.com/p/15ff9da4dd8d
                        if(!error)
                    {
                        t=formatTime(result * 1000,"Y-M-D h:m:s");
                        $("#endtime").html("本次众筹截止时间:" + t);
                       //alert(result);
                    }
                    else{
                        //alert(error);
                        $("#endtime").html('获取出错:' + error);
                    }});
                    //return "ok"
                    $("#endtime").html('正在获取。。。');
                }catch(err){
                    alert("使用授权代币转移时出错 :" + err)
                    return err;
                }
            }

            //参与众筹---
            function dodate(intvalue){
                try{
                    cc.donate(intvalue,function(error, result){
                    if(!error)
                    {

                        //$("#endtime").html("本次众筹截止时间:" + result);
                       //alert(result);
                       alert("参与成功");
                    }else{
                        alert(error);
                        //$("#endtime").html('获取出错:' + error);
                    }});
                    return "ok"
                    //$("#endtime").html('正在获取。。。');
                }catch(err){
                    alert("参与众筹出错 :" + err)
                    return err;
                }
            }



            //--------------------------调用测试---------------------------------------
            //--查询total
            function cmdone_click(){
                try{
                    gettotal();
                }catch(err){
                    alert(err);
                }
            }

            //--查询total2
            function cmdone2_click(){
                try{
                    gettotal2();
                }catch(err){
                    alert(err);
                }
            }

            //--查询本次众筹的结束日期和时间----
            function cmdtwo_click(){
                try{
                    getEndTimeStamp();
                }catch(err){
                    alert(err);
                }
            }

            //--参与众筹-------------
            function cmddo_click(){
                try{
                    var strls=document.getElementById("txtvalue").value;
                    if(/^[0-9]+\.?[0-9]+?$/.test(strls)){
                        var fls=parseFloat(strls);
                        var intls=fls * (10**18);
                        dodate(intls);
                    }else{
                        alert('只能输入数字');
                    }
                }catch(err){
                    alert(err);
                }
            }




        </script>

</head>
<body>
    <div>
        <input type="button" value="查询当前众筹总金额" id="cmdone" name="cmdone" onclick="cmdone_click();" />
        <span id="total" name="total"></span>
        <br /><br />
        <input type="button" value="查询当前众筹总金额" id="cmdone2" name="cmdone2" onclick="cmdone2_click();" />
        <span id="total2" name="total2"></span>
        <br /><br />
        <input type="button" value="查询此众筹的截止日期和时间" id="cmdtwo" name="cmdtwo" onclick="cmdtwo_click();" />
        <span id="endtime" name="endtime"></span>
        <br /><br />
        <input type="button" value="我要入筹" id="cmddo" name="cmddo" onclick="cmddo_click();" />
        <input type="text" value="0.3" id="txtvalue" name="txtvalue" />eth
        <br /><br />


    </div>

</body>
</html>

【学习后记】
昨天的时候,似乎感觉完全没有希望了,历尽千辛万苦,部署一个合约,居然是”假”的,而且后来就干脆不让我部署了,那种感觉就是——前面真的没有 路可走了, 只能回头放弃了!
事实上过去多少失败都是这样造成的,而如今我的耐受力真的提高了不是一星半点,我知道我可以坚持,更重要 的是,我明白,我相信自己一定可以成功——只要有耐心,只要继续做下去,很多时候,失败的原因就是从根本上就不相信自己罢了!
果然今天我就从昨天的“山穷水尽疑无路”中走出来了,真正见到了“柳暗花明又一村”。

github: https://github.com/lhghroom/Self-learning-blockchain-from-scratch
原文地址:https://www.941xue.com/content.aspx?id=1597

【欢迎大家加入[就是要学]社群】
如今,这个世界的变化与科技的发展就像一个机器猛兽,它跑得越来越快,跑得越来越快,在我们身后追赶着我们。
很多人很早就放弃了成长,也就放弃了继续奔跑,多数人保持终身不变的样子,原地不动,成为那猛兽的肚中餐——当然那也不错,在猛兽的逼迫下,机械的重复着自我感觉还良好地稳定工作与生活——而且多半感觉不到这有什么不正常的地方,因为在猛兽肚子里的是大多数人,就好像大多数人都在一个大坑里,也就感觉不出来这是一个大坑了,反而坑外的世界显得有些不大正常。
为什么我们不要做坑里的大多数人?
因为真正的人生,应当有百万种可能 ;因为真正的一生可以有好多辈子组成,每一辈子都可以做自己喜欢的事情;因为真正的人生,应当有无数种可以选择的权利,而不是总觉得自己别无选择。因为我们要成为一九法则中为数不多的那个一;因为我们要成为自己人生的导演而不是被迫成为别人安排的戏目中的演员。
【请注意】
就是要学社群并不会告诉你怎样一夜暴富!也不会告诉你怎样不经努力就实现梦想!
【请注意】
就是要学社群并没有任何可以应付未来一切变化的独门绝技,也没有值得吹嘘的所谓价值连城的成功学方法论!
【请注意】
社群只会互相帮助,让每个人都看清自己在哪儿,自己是怎样的,重新看见心中的梦想,唤醒各自内心中的那个英雄,然后勇往直前,成为自己想要成为的样子!
期待与你并肩奔赴未来!
www.941xue.com
QQ群:646854445 (【就是要学】终身成长)

【同步语音笔记】
https://www.ximalaya.com/keji/19103006/336057171

【学习过程屏幕录屏】
https://www.bilibili.com/video/BV1BV411677i/

  • 暂无回复。