佛萨奇DAPP开发丨佛萨奇2.0系统开发详情丨佛萨奇2.0智能合约系统开发原理及源码
8.1. solidity合约内创建合约
ChainMaker evm虚拟机支持solidity合约静态创建solidity合约,solidity合约动态创建solidity合约,以及solidity合约动态创建其他类型合约。
8.1.1. 静态创建
所谓静态创建,是指在合约创建之前,创建者合约必须掌握被创建合约的代码。二者在编译之前就已经被绑定,并一同编译生成可部署的字节码,不能在运行时改变被创建者。
静态创建示例
合约static_create.sol,实现功能:
1、部署Creator合约后,可调用Creator合约的createSalted方法创建Creature合约;
2、创建时,可通过name参数,为Creature合约指定合约名。注:部署合约时,ChainMaker会根据合约名计算合约地址,并将地址返回给用户,合约部署完成后,用户即可使用合约名调用合约,也可使用合约地址调用合约。
// SPDX-License-Identifier: GPL-3.0
pragma solidity >=0.7.0 <0.9.0;
//被创建合约
contract Creature {
uint public x;
constructor(uint a) payable {
x = a;
}
function get() public view returns(uint){
return x;
}
}
//创建者合约
contract Creator {
function bytesCopy(bytes memory src, bytes memory dst, uint begin) internal pure returns (bytes memory){
uint len = src.length < 31 ? src.length : 31;
for(uint i = 0; i < len; i++){
dst[begin + i] = src[i];
}
return dst;
}
function createSalted(uint arg, string calldata name) public {
bytes memory bytesname = bytes(name);
bytes memory custom = new bytes(32);
uint8 rtTypeEvm = 5; //evm的虚拟机类型是5,所以该值必须是5
custom[0] = bytes1(rtTypeEvm); //虚拟机类型必须放入custom的首字节,custom将赋值给创建选项salt
custom = bytesCopy(bytesname, custom, 1); //将合约名拷贝到后31字节内
bytes32 n;
assembly{
n := mload(add(custom, 32))
}
//花括号内的salt为创建选项,salt为solidity语法关键字,赋值类型为byte32数据的值
//如果不指定为salt选项,写法为: Creature d = new Creature(arg)
Creature d = new Creature{salt: n}(arg);
}
}
如示例所示,创建者合约与被创建者合约位于同一文件内(也可以import导入),一同编译,执行创建动作时,无法更改被创建合约。
创建时,可通过salt选项指定被创建合约名,salt为byte32类型,但第一个字节必须赋值为evm虚拟机类型的枚举值5,后31字节才用于赋值合约名,所以合约名长度不得超过31字节。evm虚拟机会根据合约名计算地址,对于Chainmaker来说,所有类型的合约,都同时支持通过合约名和合约地址两种方式调用。
创建时,如果不指定salt选项,则evm虚拟机将为被创建者合约自动生成合约名和合约地址。
8.1.2. 动态创建
所谓动态创建,是指创建者合约不需要提前知道被创建合约的代码,只需要在创建时,告诉创建者被创建合约编译后的字节码即可,相比静态创建,动态创建更加灵活。
动态创建示例
合约dynamic_create.sol,实现功能:
1、部署Factory合约后,可调用Factory合约的create方法创建任意合约;
2、创建时,可指定合约虚拟机类型和合约名,并根据提供的合约字节码创建合约。
// SPDX-License-Identifier: GPL-3.0
pragma solidity >=0.8.0;
contract Factory {
function bytesCopy(bytes memory src, bytes memory dst, uint begin) internal pure returns (bytes memory){
uint len = src.length < 31 ? src.length : 31;
for(uint i = 0; i < len; i++){
dst[begin + i] = src[i];
}
return dst;
}
function create(uint8 rtType, string calldata name, bytes calldata code) public returns (address addr){
assert(0 < rtType && rtType < 8);//ChainMaker支持7种虚拟机
bytes memory bytesname = bytes(name);
bytes memory bytesCode = code;
bytes memory custom = new bytes(32);
custom[0] = bytes1(rtType);
custom = bytesCopy(bytesname, custom, 1);
assembly {
//create2指令参数1为转账额,参数2为合约字节码长度,参数3为字节码,参数4为虚拟机类型+合约名
addr := create2(0, add(bytesCode,0x20), mload(bytesCode), mload(add(custom, 32)))
}
}
}
如示例所示,创建者合约不需要事先知道被创建合约,只需要在调用create方法时,以参数将被创建合约字节码传入即可。因为是动态创建,被创建合约不需要在编译前写入同一文件,所以,非solidity合约同样进入了被创建者的可选范围。
8.1.3. 异构创建
异构创建是指,solidity合约创建其他虚拟机类型的合约。异构创建方式和动态创建相同,只需要为调用者合约提供虚拟机类型、被调合约名和合约字节码即可,注意,这三个参数都是必不可少的。
相关文章
- Node学习笔记 - Koa源码阅读
- Spring Framework 源码学习笔记(七)- AOP
- java进销存ERP管理系统源码[通俗易懂]
- 网站在线客服系统源码|在线客服代码下载 (2021最新版)「建议收藏」
- 在线客服系统源码开发实战总结:动态加载js文件实现粘贴一段js的sdk代码,直接引入插件效果
- Golang(五)Context 的使用和源码分析
- 【说站】健身类小程序前后端源码
- 【说站】易校网校园综合跑腿小程序修复版源码
- Spring MVC注解Controller源码流程解析--定位HandlerMethod
- ArrayList 源码笔记
- 全新开发悬赏任务系统源码
- 佛萨奇2.0开发系统逻辑源码
- 基于SpringCloudAlibaba和uni-app的前后端分离的医疗论坛系统(附源码)
- 海外的在线客服系统源码推荐
- Redis 7.0 源码调试环境搭建与源码导读技巧
- ReactDOM.render在react源码中执行之后发生了什么?
- java基于ssm框架开发的公交查询系统源码公交系统源码公交路线查询项目有论文
- Redis源码设计剖析之事件处理示例详解
- Linux系统安装源码步骤详解,轻松get编译安装自己的软件(linux 安装源码)
- 编译解锁MySQL性能C语言源码编译实战(c mysql 源码)
- jQuery源码分析-03构造jQuery对象-源码结构和核心函数