truffle开发一个简单的Dapp
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
- pragma solidity ^0.4.17;
- contract Adoption {
- address[16] public adopters; // 保存领养者的地址
- // 领养宠物
- function adopt(uint petId) public returns (uint) {
- require(petId >= 0 && petId <= 15); // 确保id在数组长度内
- adopters[petId] = msg.sender; // 保存调用这地址
- return petId;
- }
- // 返回领养者
- function getAdopters() public view returns (address[16]) {
- return adopters;
- }
- }
5.编译合约:truffle compile
6.创建一个自己的部署脚本2_deploy_contracts.js
- var Adoption = artifacts.require("Adoption");
- module.exports = function(deployer) {
- deployer.deploy(Adoption);
- };
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
- pragma solidity ^0.4.17;
- import "truffle/Assert.sol"; // 引入的断言
- import "truffle/DeployedAddresses.sol"; // 用来获取被测试合约的地址
- import "../contracts/Adoption.sol"; // 被测试合约
- contract TestAdoption {
- Adoption adoption = Adoption(DeployedAddresses.Adoption());
- // 领养测试用例
- function testUserCanAdoptPet() public {
- uint returnedId = adoption.adopt(8);
- uint expected = 8;
- Assert.equal(returnedId, expected, "Adoption of pet ID 8 should be recorded.");
- }
- // 宠物所有者测试用例
- function testGetAdopterAddressByPetId() public {
- // 期望领养者的地址就是本合约地址,因为交易是由测试合约发起交易,
- address expected = this;
- address adopter = adoption.adopters(8);
- Assert.equal(adopter, expected, "Owner of pet ID 8 should be recorded.");
- }
- // 测试所有领养者
- function testGetAdopterAddressByPetIdInArray() public {
- // 领养者的地址就是本合约地址
- address expected = this;
- address[16] memory adopters = adoption.getAdopters();
- Assert.equal(adopters[8], expected, "Owner of pet ID 8 should be recorded.");
- }
- }
10.运行测试用例:truffle test
11.Truffle Box的pet-shop中,已经包含了应用的前端代码,都在src/文件夹中,更改app.js来实现应用的功能。初始化web3,修改initWeb3(),删除注释,修改为
- initWeb3: function() {
- // Is there an injected web3 instance?
- if (typeof web3 !== 'undefined') {
- App.web3Provider = web3.currentProvider;
- } else {
- // If no injected web3 instance is detected, fall back to Ganache
- App.web3Provider = new Web3.providers.HttpProvider('http://localhost:7545');
- }
- web3 = new Web3(App.web3Provider);
- return App.initContract();
- }
12.实例化合约,修改initContract()函数
- initContract: function() {
- // 加载Adoption.json,保存了Adoption的ABI(接口说明)信息及部署后的网络(地址)信息,它在编译合约的时候生成ABI,在部署的时候追加网络信息
- $.getJSON('Adoption.json', function(data) {
- // 用Adoption.json数据创建一个可交互的TruffleContract合约实例。
- var AdoptionArtifact = data;
- App.contracts.Adoption = TruffleContract(AdoptionArtifact);
- // Set the provider for our contract
- App.contracts.Adoption.setProvider(App.web3Provider);
- // Use our contract to retrieve and mark the adopted pets
- return App.markAdopted();
- });
- return App.bindEvents();
- }
13.处理领养,修改markAdopted()
- markAdopted: function(adopters, account) {
- var adoptionInstance;
- App.contracts.Adoption.deployed().then(function(instance) {
- adoptionInstance = instance;
- // 调用合约的getAdopters(), 用call读取信息不用消耗gas
- return adoptionInstance.getAdopters.call();
- }).then(function(adopters) {
- for (i = 0; i < adopters.length; i++) {
- if (adopters[i] !== '0x0000000000000000000000000000000000000000') {
- $('.panel-pet').eq(i).find('button').text('Success').attr('disabled', true);
- }
- }
- }).catch(function(err) {
- console.log(err.message);
- });
- }
14.修改handleAdopt()
- handleAdopt: function(event) {
- event.preventDefault();
- var petId = parseInt($(event.target).data('id'));
- var adoptionInstance;
- // 获取用户账号
- web3.eth.getAccounts(function(error, accounts) {
- if (error) {
- console.log(error);
- }
- var account = accounts[0];
- App.contracts.Adoption.deployed().then(function(instance) {
- adoptionInstance = instance;
- // 发送交易领养宠物
- return adoptionInstance.adopt(petId, {from: account});
- }).then(function(result) {
- return App.markAdopted();
- }).catch(function(err) {
- console.log(err.message);
- });
- });
- }
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的更多相关文章
- 如何开发一个简单的HTML5 Canvas 小游戏
原文:How to make a simple HTML5 Canvas game 想要快速上手HTML5 Canvas小游戏开发?下面通过一个例子来进行手把手教学.(如果你怀疑我的资历, A Wiz ...
- 重新想象 Windows 8 Store Apps (64) - 后台任务: 开发一个简单的后台任务
[源码下载] 重新想象 Windows 8 Store Apps (64) - 后台任务: 开发一个简单的后台任务 作者:webabcd 介绍重新想象 Windows 8 Store Apps 之 后 ...
- Cocos2d-x-Lua 开发一个简单的游戏(记数字步进白色块状)
Cocos2d-x-Lua 开发一个简单的游戏(记数字步进白色块状) 本篇博客来给大家介绍怎样使用Lua这门语言来开发一个简单的小游戏-记数字踩白块. 游戏的流程是这种:在界面上生成5个数1~5字并显 ...
- Python开发一个简单的BBS论坛
项目:开发一个简单的BBS论坛 需求: 整体参考“抽屉新热榜” + “虎嗅网” 实现不同论坛版块 帖子列表展示 帖子评论数.点赞数展示 在线用户展示 允许登录用户发贴.评论.点赞 允许上传文件 帖子可 ...
- 作业1开发一个简单的python计算器
开发一个简单的python计算器 实现加减乘除及拓号优先级解析 用户输入 1 - 2 * ( (60-30 +(-40/5) * (9-2*5/3 + 7 /3*99/4*2998 +10 * 568 ...
- django学习-11.开发一个简单的醉得意菜单和人均支付金额查询页面
1.前言 刚好最近跟技术部门的[产品人员+UI人员+测试人员],组成了一桌可以去公司楼下醉得意餐厅吃饭的小team. 所以为了实现这些主要点餐功能: 提高每天中午点餐效率,把点餐时间由20分钟优化为1 ...
- 自己动手模拟开发一个简单的Web服务器
开篇:每当我们将开发好的ASP.NET网站部署到IIS服务器中,在浏览器正常浏览页面时,可曾想过Web服务器是怎么工作的,其原理是什么?“纸上得来终觉浅,绝知此事要躬行”,于是我们自己模拟一个简单的W ...
- 【UI插件】开发一个简单日历插件(上)
前言 最近开始整理我们的单页应用框架了,虽然可能比不上MVVM模式的开发效率,也可能没有Backbone框架模块清晰,但是好歹也是自己开发出来 而且也用于了这么多频道的东西,如果没有总结,没有整理,没 ...
- 30 分钟开发一个简单的 watchOS 2 app <oneVcat>
Apple Watch 和 watchOS 第一代产品只允许用户在 iPhone 设备上进行计算,然后将结果传输到手表上进行显示.在这个框架下,手表充当的功能在很大程度上只是手机的另一块小一些的显示器 ...
随机推荐
- SQL3120W 不能将xx的字段值转换成 INTEGER值
一次用DB2 Load/Import导入数据时,报错,提示SQL3120W 不能将xx的字段值转换成 INTEGER值,但目标列不可为空.未装入该行. 目标表: CREATE TABLE TEST( ...
- NEC 框架规范 css function
/* function */.f-cb:after,.f-cbli li:after{display:block;clear:both;visibility:hidden;height:0;overf ...
- call、apply和bind的用法
在改变 this 指向的时候,经常会把这三个方法混淆,下面就详细的整理一下三者的用法和区别 call() 方法 call() 方法可以有无数个参数 第一个参数是改变 this 指向的对象 后面的参数直 ...
- 原生js的常见封装
)); } ;;;;]){ ]; ] = ;;;,) ,) ,) ,) ,) , ...
- ant-design-pro使用服务器数据接口代理配置
因为是新入门antd-pro这个的小白,所以在mock数据和服务器数据切换这里搞了将近2天才弄好,配置如下,供各位初学者参考,如有错误的地方,请大神指出~叩谢!! 下面开始干货: 1..roadhog ...
- Qt基于model/view数据库编程3
QSqlQueryModel和QSqlQuery类: 工程开发过程中将这两个类合起来使用,用QSqlQueryModel查询展示数据库中的数据,用QSqlQuery类执行sql语言,实现对数据库的操作 ...
- [SDOI2008]仪仗队(欧拉筛裸题)
题目描述 作为体育委员,C君负责这次运动会仪仗队的训练.仪仗队是由学生组成的N * N的方阵,为了保证队伍在行进中整齐划一,C君会跟在仪仗队的左后方,根据其视线所及的学生人数来判断队伍是否整齐(如右图 ...
- Zabbix——部署(DB与web一体)
前提条件: CentOS连接网络并可以正常访问网络 DNS设置完成,可以Ping同外网域名 安装数据库为8.0版本 关闭防火墙 如果需要搭建分离式请见:DB与Web分离 &DB.web.age ...
- springMVC-RESTful约束下dispatcher拦截对象优化
警告: No mapping found for HTTP request with URI [/management/fonts/glyphicons-halflings-regular.woff] ...
- 路由器基础配置之dhcp配置
我们将以上面的拓扑图为例,router9为dhcp的服务器,为pc4,5,6分配三个不同网段的地址,pool为要分配的三个地址池,我们要把pc4设置为12网段,pc5设置成34网段,pc6设置成56网 ...