用工厂模式管理以太坊的多个solidity智能合约
我们写了一份小的计算合约作为Hello World。如果我们可以创建一个允许用户创建自己的计数器的合约怎么办?
让我们创建我们的合约CounterFactory
,它将管理所有其他计数器Counters
。它将包含一个映射,将所有者与其计数器合约的地址相关联。
mapping(address => address) counters;
当用户想要使用我们的计数器系统来拥有他自己的计数器时,他将需要请求创建他的计数器。
function createCounter() public {
if (counters[msg.sender] == 0) {
counters[msg.sender] = new Counter(msg.sender);
}
}
请注意,我们将构造函数的地址传递给构造函数,因此我们将转移调用者的所有权。在新智能合约的构造函数中,msg.sender
将引用我们的合约工厂的地址。这是一个非常重要的要点,因为使用合约与其他合约进行交互是一种常见做法。因此,你应该在复杂的情况下照顾谁是发件人。
现在是增量函数,我们首先检查用户是否已经注册了智能合约并从合约中调用增量函数。由于映射存储了智能的地址,我们需要将地址转换为Counter
合约类型。存储合约的地址而不是直接引用智能合约允许我们通过使用空地址检查合约是否已初始化:0或0x0 ..
function increment() public {
require (counters[msg.sender] != 0);
Counter(counters[msg.sender]).increment(msg.sender);
}
最后,为了读取计数器的值,我们将用户的地址作为参数来获取计数器的值。
function getCount(address account) public constant returns (uint) {
if (counters[account] != 0) {
return (Counter(counters[account]).getCount());
}
}
在这个例子中,我们保持简单但你可以想象几个场景,例如需要将Ether发送到createCounter()
函数,这样合约的初始创建者可以获得一些收入来完成他的工作。我们还可以让原始创建者删除计数器,或将合约与字符串或数字相关联。
对Counter
合约进行了简单编辑,以适应作为参数传递的新地址。
这是完整的代码:
pragma solidity ^0.4.11;
contract Counter {
address owner;
address factory;
uint count = 0;
function Counter(address _owner) {
owner = _owner;
factory = msg.sender
}
modifier isOwner(address _caller) {
require(msg.sender == factory);
require(_caller == owner);
_;
}
function increment(address caller) public isOwner(caller) {
count = count + 1;
}
function getCount() constant returns (uint) {
return count;
}
}
contract CounterFactory {
mapping(address => address) counters;
function createCounter() public {
if (counters[msg.sender] == 0) {
counters[msg.sender] = new Counter(msg.sender);
}
}
function increment() public {
require (counters[msg.sender] != 0);
Counter(counters[msg.sender]).increment(msg.sender);
}
function getCount(address account) public constant returns (uint) {
if (counters[account] != 0) {
return (Counter(counters[account]).getCount());
}
}
}
请注意,如果多次调用,我们的计数器可能会成为溢出的受害者。你应尽可能使用SafeMath库来防止这种情况。
要部署我们的智能合约,您需要提供CounterFactory
和Counter
的代码。部署时,你需要选择CounterFactory
。
从你的一个帐户执行createCounter()
函数并在合约界面的阅读部分调用increment()
函数后,你需要将帐户的地址设置为读取计数器的值。你现在可以为每个帐户设置一个计数器。
在下一个教程中,我们将看到如何使用继承来保持干净的代码并重用现有的和经过测试的区块。
分享一些以太坊、EOS、比特币等区块链相关的交互式在线编程实战教程:
- java以太坊开发教程,主要是针对java和android程序员进行区块链以太坊开发的web3j详解。
- python以太坊,主要是针对python工程师使用web3.py进行区块链以太坊开发的详解。
- php以太坊,主要是介绍使用php进行智能合约开发交互,进行账号创建、交易、转账、代币开发以及过滤器和交易等内容。
- 以太坊入门教程,主要介绍智能合约与dapp应用开发,适合入门。
- 以太坊开发进阶教程,主要是介绍使用node.js、mongodb、区块链、ipfs实现去中心化电商DApp实战,适合进阶。
- C#以太坊,主要讲解如何使用C#开发基于.Net的以太坊应用,包括账户管理、状态与交易、智能合约开发与交互、过滤器和交易等。
- 如何构建智能反垃圾邮件的WordPress插件
- 【深入研究】使用RNN预测股票价格系列一
- 【深入研究】使用RNN预测股票价格系列二
- 教你用一行Python代码实现并行(附代码)
- 在美国国会图书馆标题表的SKOS上运行Apache Spark GraphX算法
- 【精选】破解波动性突破实盘系统
- 从程序员的角度看神经网络的激活功能
- 在线矩阵微积分工具,可以生成 Python/Latex 代码哦!
- 机器学习应用区块链系列(一)——如何开发一套自己的智能合约系统
- 使用Botkit和Rasa NLU构建智能聊天机器人
- 【量化投资】缠论面面观(附Python源码)
- 独家 | 教你用Q学习算法训练神经网络玩游戏(附源码)
- 使用重采样评估Python中机器学习算法的性能
- Autofac正式发布2.1版
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法