1.安装Truffle:npm install -g truffle

2.建立项目目录并进入:mkdir pet-shop-tutorial

cd pet-shop-tutorial

3.使用truffle unbox创建项目,这种方式的前端直接就是写好的:truffle unbox pet-shop

4.在contracts目录下,添加合约文件Adoption.sol

  1. pragma solidity ^0.4.17;
  2.  
  3. contract Adoption {
  4.  
  5. address[16] public adopters; // 保存领养者的地址
  6.  
  7. // 领养宠物
  8. function adopt(uint petId) public returns (uint) {
  9. require(petId >= 0 && petId <= 15); // 确保id在数组长度内
  10.  
  11. adopters[petId] = msg.sender; // 保存调用这地址
  12. return petId;
  13. }
  14.  
  15. // 返回领养者
  16. function getAdopters() public view returns (address[16]) {
  17. return adopters;
  18. }
  19.  
  20. }

5.编译合约:truffle compile

6.创建一个自己的部署脚本2_deploy_contracts.js

  1. var Adoption = artifacts.require("Adoption");
  2.  
  3. module.exports = function(deployer) {
  4. deployer.deploy(Adoption);
  5. };

7.执行部署之前,确保有一个区块链运行,使用Ganache开启一个私链进行开发测试,默认会在7545端口上。安装Ganache:执行wget https://github.com/trufflesuite/ganache/release/download/v1.0.1/ganache-1.0.1-x86_64.AppImage,再修改权限:chmod +x ganache-1.0.1-x86_64.AppImage,双击图标就可以运行Ganache。

8.部署合约:truffle mimgrate

9.测试,在test目录下新建一个TestAdoption.sol

  1. pragma solidity ^0.4.17;
  2.  
  3. import "truffle/Assert.sol"; // 引入的断言
  4. import "truffle/DeployedAddresses.sol"; // 用来获取被测试合约的地址
  5. import "../contracts/Adoption.sol"; // 被测试合约
  6.  
  7. contract TestAdoption {
  8. Adoption adoption = Adoption(DeployedAddresses.Adoption());
  9.  
  10. // 领养测试用例
  11. function testUserCanAdoptPet() public {
  12. uint returnedId = adoption.adopt(8);
  13.  
  14. uint expected = 8;
  15. Assert.equal(returnedId, expected, "Adoption of pet ID 8 should be recorded.");
  16. }
  17.  
  18. // 宠物所有者测试用例
  19. function testGetAdopterAddressByPetId() public {
  20. // 期望领养者的地址就是本合约地址,因为交易是由测试合约发起交易,
  21. address expected = this;
  22. address adopter = adoption.adopters(8);
  23. Assert.equal(adopter, expected, "Owner of pet ID 8 should be recorded.");
  24. }
  25.  
  26. // 测试所有领养者
  27. function testGetAdopterAddressByPetIdInArray() public {
  28. // 领养者的地址就是本合约地址
  29. address expected = this;
  30. address[16] memory adopters = adoption.getAdopters();
  31. Assert.equal(adopters[8], expected, "Owner of pet ID 8 should be recorded.");
  32. }
  33. }

10.运行测试用例:truffle test

11.Truffle Box的pet-shop中,已经包含了应用的前端代码,都在src/文件夹中,更改app.js来实现应用的功能。初始化web3,修改initWeb3(),删除注释,修改为

  1. initWeb3: function() {
  2. // Is there an injected web3 instance?
  3. if (typeof web3 !== 'undefined') {
  4. App.web3Provider = web3.currentProvider;
  5. } else {
  6. // If no injected web3 instance is detected, fall back to Ganache
  7. App.web3Provider = new Web3.providers.HttpProvider('http://localhost:7545');
  8. }
  9. web3 = new Web3(App.web3Provider);
  10.  
  11. return App.initContract();
  12. }

12.实例化合约,修改initContract()函数

  1. initContract: function() {
  2. // 加载Adoption.json,保存了Adoption的ABI(接口说明)信息及部署后的网络(地址)信息,它在编译合约的时候生成ABI,在部署的时候追加网络信息
  3. $.getJSON('Adoption.json', function(data) {
  4. // 用Adoption.json数据创建一个可交互的TruffleContract合约实例。
  5. var AdoptionArtifact = data;
  6. App.contracts.Adoption = TruffleContract(AdoptionArtifact);
  7.  
  8. // Set the provider for our contract
  9. App.contracts.Adoption.setProvider(App.web3Provider);
  10.  
  11. // Use our contract to retrieve and mark the adopted pets
  12. return App.markAdopted();
  13. });
  14. return App.bindEvents();
  15. }

13.处理领养,修改markAdopted()

  1. markAdopted: function(adopters, account) {
  2. var adoptionInstance;
  3.  
  4. App.contracts.Adoption.deployed().then(function(instance) {
  5. adoptionInstance = instance;
  6.  
  7. // 调用合约的getAdopters(), 用call读取信息不用消耗gas
  8. return adoptionInstance.getAdopters.call();
  9. }).then(function(adopters) {
  10. for (i = 0; i < adopters.length; i++) {
  11. if (adopters[i] !== '0x0000000000000000000000000000000000000000') {
  12. $('.panel-pet').eq(i).find('button').text('Success').attr('disabled', true);
  13. }
  14. }
  15. }).catch(function(err) {
  16. console.log(err.message);
  17. });
  18. }

14.修改handleAdopt()

  1. handleAdopt: function(event) {
  2. event.preventDefault();
  3.  
  4. var petId = parseInt($(event.target).data('id'));
  5.  
  6. var adoptionInstance;
  7.  
  8. // 获取用户账号
  9. web3.eth.getAccounts(function(error, accounts) {
  10. if (error) {
  11. console.log(error);
  12. }
  13.  
  14. var account = accounts[0];
  15.  
  16. App.contracts.Adoption.deployed().then(function(instance) {
  17. adoptionInstance = instance;
  18.  
  19. // 发送交易领养宠物
  20. return adoptionInstance.adopt(petId, {from: account});
  21. }).then(function(result) {
  22. return App.markAdopted();
  23. }).catch(function(err) {
  24. console.log(err.message);
  25. });
  26. });
  27. }

15.安装MetaMask,点击链接https://metamask.io/,将插件安装到浏览器中,点击浏览器右上角的图标,首先我们选择一个测试网络(如粉红色的),进入界面,选择Accept,在登录页面中点击“Import Existing DEN”,还原Ganache创建的钱包,作为开发测试钱包,在输入框中填写candy maple cake sugar pudding cream honey rich smooth crumble sweet treat这几个词,输入自己的密码。

16.登录之后,点击Custom RPC,添加一个网络:http://127.0.0.1:7545,切换之后返回,看到第一个账户的状态

17.运行npm run dev启动服务

自己在完成这个项目中遇到的问题

问题一:执行truffle compile虚拟机卡顿,一直执行不下去

解决方法:将虚拟机的内存改大一点,可以改为3G

问题二:执行了npm run dev,但是不能直接启动到Firefox浏览器

解决办法:直接打开浏览器,在地址栏中输入localhost:3000

问题三:进入到项目后,网页上显示不出pet的信息

解决办法:显示不出信息是因为jQuery的加载问题,直接百度一个jQuery.1.11.1.min.js,替换新的链接

问题四:当点击完Adopt,并且submit之后,相应的按钮应该显示为success,通过打印输出发现是adopters数组有问题,地址全都为0x

解决办法:再执行一次truffle migrate

根据https://learnblockchain.cn/2018/01/12/first-dapp/#more博主的分享总结的。

truffle开发一个简单的Dapp的更多相关文章

  1. 如何开发一个简单的HTML5 Canvas 小游戏

    原文:How to make a simple HTML5 Canvas game 想要快速上手HTML5 Canvas小游戏开发?下面通过一个例子来进行手把手教学.(如果你怀疑我的资历, A Wiz ...

  2. 重新想象 Windows 8 Store Apps (64) - 后台任务: 开发一个简单的后台任务

    [源码下载] 重新想象 Windows 8 Store Apps (64) - 后台任务: 开发一个简单的后台任务 作者:webabcd 介绍重新想象 Windows 8 Store Apps 之 后 ...

  3. Cocos2d-x-Lua 开发一个简单的游戏(记数字步进白色块状)

    Cocos2d-x-Lua 开发一个简单的游戏(记数字步进白色块状) 本篇博客来给大家介绍怎样使用Lua这门语言来开发一个简单的小游戏-记数字踩白块. 游戏的流程是这种:在界面上生成5个数1~5字并显 ...

  4. Python开发一个简单的BBS论坛

    项目:开发一个简单的BBS论坛 需求: 整体参考“抽屉新热榜” + “虎嗅网” 实现不同论坛版块 帖子列表展示 帖子评论数.点赞数展示 在线用户展示 允许登录用户发贴.评论.点赞 允许上传文件 帖子可 ...

  5. 作业1开发一个简单的python计算器

    开发一个简单的python计算器 实现加减乘除及拓号优先级解析 用户输入 1 - 2 * ( (60-30 +(-40/5) * (9-2*5/3 + 7 /3*99/4*2998 +10 * 568 ...

  6. django学习-11.开发一个简单的醉得意菜单和人均支付金额查询页面

    1.前言 刚好最近跟技术部门的[产品人员+UI人员+测试人员],组成了一桌可以去公司楼下醉得意餐厅吃饭的小team. 所以为了实现这些主要点餐功能: 提高每天中午点餐效率,把点餐时间由20分钟优化为1 ...

  7. 自己动手模拟开发一个简单的Web服务器

    开篇:每当我们将开发好的ASP.NET网站部署到IIS服务器中,在浏览器正常浏览页面时,可曾想过Web服务器是怎么工作的,其原理是什么?“纸上得来终觉浅,绝知此事要躬行”,于是我们自己模拟一个简单的W ...

  8. 【UI插件】开发一个简单日历插件(上)

    前言 最近开始整理我们的单页应用框架了,虽然可能比不上MVVM模式的开发效率,也可能没有Backbone框架模块清晰,但是好歹也是自己开发出来 而且也用于了这么多频道的东西,如果没有总结,没有整理,没 ...

  9. 30 分钟开发一个简单的 watchOS 2 app <oneVcat>

    Apple Watch 和 watchOS 第一代产品只允许用户在 iPhone 设备上进行计算,然后将结果传输到手表上进行显示.在这个框架下,手表充当的功能在很大程度上只是手机的另一块小一些的显示器 ...

随机推荐

  1. SQL3120W 不能将xx的字段值转换成 INTEGER值

    一次用DB2 Load/Import导入数据时,报错,提示SQL3120W 不能将xx的字段值转换成 INTEGER值,但目标列不可为空.未装入该行. 目标表: CREATE TABLE TEST( ...

  2. NEC 框架规范 css function

    /* function */.f-cb:after,.f-cbli li:after{display:block;clear:both;visibility:hidden;height:0;overf ...

  3. call、apply和bind的用法

    在改变 this 指向的时候,经常会把这三个方法混淆,下面就详细的整理一下三者的用法和区别 call() 方法 call() 方法可以有无数个参数 第一个参数是改变 this 指向的对象 后面的参数直 ...

  4. 原生js的常见封装

    )); } ;;;;]){                 ];                 ] = ;;;,)     ,)     ,)     ,)         ,)         , ...

  5. ant-design-pro使用服务器数据接口代理配置

    因为是新入门antd-pro这个的小白,所以在mock数据和服务器数据切换这里搞了将近2天才弄好,配置如下,供各位初学者参考,如有错误的地方,请大神指出~叩谢!! 下面开始干货: 1..roadhog ...

  6. Qt基于model/view数据库编程3

    QSqlQueryModel和QSqlQuery类: 工程开发过程中将这两个类合起来使用,用QSqlQueryModel查询展示数据库中的数据,用QSqlQuery类执行sql语言,实现对数据库的操作 ...

  7. [SDOI2008]仪仗队(欧拉筛裸题)

    题目描述 作为体育委员,C君负责这次运动会仪仗队的训练.仪仗队是由学生组成的N * N的方阵,为了保证队伍在行进中整齐划一,C君会跟在仪仗队的左后方,根据其视线所及的学生人数来判断队伍是否整齐(如右图 ...

  8. Zabbix——部署(DB与web一体)

    前提条件: CentOS连接网络并可以正常访问网络 DNS设置完成,可以Ping同外网域名 安装数据库为8.0版本 关闭防火墙 如果需要搭建分离式请见:DB与Web分离 &DB.web.age ...

  9. springMVC-RESTful约束下dispatcher拦截对象优化

    警告: No mapping found for HTTP request with URI [/management/fonts/glyphicons-halflings-regular.woff] ...

  10. 路由器基础配置之dhcp配置

    我们将以上面的拓扑图为例,router9为dhcp的服务器,为pc4,5,6分配三个不同网段的地址,pool为要分配的三个地址池,我们要把pc4设置为12网段,pc5设置成34网段,pc6设置成56网 ...