当前位置: 首页 > news >正文

Fisco Bcos学习 - 开发第一个区块链应用

文章目录

    • 一、前言
    • 二、业务场景分析:简易资产管理系统
    • 三、智能合约设计与实现
      • 3.1 存储结构设计
      • 3.2 接口设计
      • 3.3 完整合约代码
    • 四、合约编译与Java接口生成
    • 五、SDK配置与项目搭建
      • 5.1 获取Java工程项目
      • 5.2 项目目录结构
      • 5.3 引入Web3SDK
      • 5.4 证书与配置文件
    • 六、业务开发:Java客户端实现
      • 6.1 核心类设计:AssetClient
        • 6.1.1 初始化
        • 6.1.2 合约对象创建
        • 6.1.3 接口调用
    • 七、项目运行与功能验证
      • 7.1 编译项目
      • 7.2 部署合约
      • 7.3 注册资产
      • 7.4 查询资产
      • 7.5 资产转移
    • 八、参考资料

一、前言

在区块链技术快速发展的今天,如何将其应用于实际业务场景成为开发者关注的重点。FISCO BCOS作为国产优秀的联盟链平台,为企业级区块链应用开发提供了强大支持。本文将跟随官方教程,详细记录基于FISCO BCOS构建第一个区块链应用的全过程,涵盖从业务分析到最终实现的完整流程。

二、业务场景分析:简易资产管理系统

区块链技术因其防篡改、可追溯的特性,在金融领域有着天然优势。本次实践选择开发一个简易的资产管理系统,主要实现以下核心功能:

  • 资产注册:在区块链上登记资产账户及初始金额
  • 资产转账:实现不同账户间的资产转移
  • 资产查询:查询指定账户的资产余额

这个场景虽然简单,但涵盖了区块链应用开发的核心流程,非常适合作为入门案例。

三、智能合约设计与实现

3.1 存储结构设计

FISCO BCOS提供了合约CRUD接口开发模式,允许通过合约创建表结构并进行数据操作。针对资产管理需求,设计了名为t_asset的表:

  • account:资产账户,作为主键(string类型)
  • asset_value:资产金额(uint256类型)

该表结构示例如下:

accountasset_value
Alice10000
Bob20000

3.2 接口设计

根据业务目标,定义了三个核心接口:

  • select(string account):查询资产金额
  • register(string account, uint256 amount):资产注册
  • transfer(string from_asset_account, string to_asset_account, uint256 amount):资产转移

3.3 完整合约代码

pragma solidity ^0.4.24;
import "./Table.sol";contract Asset {// 事件定义,用于记录关键操作event RegisterEvent(int256 ret, string account, uint256 asset_value);event TransferEvent(int256 ret, string from_account, string to_account, uint256 amount);constructor() public {// 构造函数中创建表createTable();}function createTable() private {TableFactory tf = TableFactory(0x1001);// 创建t_asset表,指定主键和字段tf.createTable("t_asset", "account", "asset_value");}function openTable() private returns (Table) {TableFactory tf = TableFactory(0x1001);Table table = tf.openTable("t_asset");return table;}// 查询资产金额function select(string account) public constant returns (int256, uint256) {Table table = openTable();Entries entries = table.select(account, table.newCondition());uint256 asset_value = 0;if (0 == uint256(entries.size())) {return (-1, asset_value);} else {Entry entry = entries.get(0);return (0, uint256(entry.getInt("asset_value")));}}// 资产注册function register(string account, uint256 asset_value) public returns (int256) {int256 ret_code = 0;int256 ret = 0;uint256 temp_asset_value = 0;(ret, temp_asset_value) = select(account);if (ret != 0) {Table table = openTable();Entry entry = table.newEntry();entry.set("account", account);entry.set("asset_value", int256(asset_value));int count = table.insert(account, entry);if (count == 1) {ret_code = 0;} else {ret_code = -2;}} else {ret_code = -1;}emit RegisterEvent(ret_code, account, asset_value);return ret_code;}// 资产转移function transfer(string from_account, string to_account, uint256 amount) public returns (int256) {int ret_code = 0;int256 ret = 0;uint256 from_asset_value = 0;uint256 to_asset_value = 0;// 检查转出账户是否存在(ret, from_asset_value) = select(from_account);if (ret != 0) {ret_code = -1;emit TransferEvent(ret_code, from_account, to_account, amount);return ret_code;}// 检查转入账户是否存在(ret, to_asset_value) = select(to_account);if (ret != 0) {ret_code = -2;emit TransferEvent(ret_code, from_account, to_account, amount);return ret_code;}// 检查余额是否充足if (from_asset_value < amount) {ret_code = -3;emit TransferEvent(ret_code, from_account, to_account, amount);return ret_code;}// 检查金额是否溢出if (to_asset_value + amount < to_asset_value) {ret_code = -4;emit TransferEvent(ret_code, from_account, to_account, amount);return ret_code;}Table table = openTable();Entry entry0 = table.newEntry();entry0.set("account", from_account);entry0.set("asset_value", int256(from_asset_value - amount));int count = table.update(from_account, entry0, table.newCondition());if (count != 1) {ret_code = -5;emit TransferEvent(ret_code, from_account, to_account, amount);return ret_code;}Entry entry1 = table.newEntry();entry1.set("account", to_account);entry1.set("asset_value", int256(to_asset_value + amount));table.update(to_account, entry1, table.newCondition());emit TransferEvent(ret_code, from_account, to_account, amount);return ret_code;}
}

注意:该合约依赖FISCO BCOS提供的系统合约Table.sol,实现对表的CRUD操作

四、合约编译与Java接口生成

Solidity合约无法被Java程序直接调用,需要通过控制台工具将其编译为Java类。具体步骤如下:

  1. Asset.solTable.sol放在console/contracts/solidity目录
  2. 执行编译脚本:
# 切换到console目录
cd ~/fisco/console/
# 编译合约,指定Java包名
./sol2java.sh org.fisco.bcos.asset.contract

编译成功后会在console/contracts/sdk目录生成:

  • abi目录:存放合约ABI文件
  • bin目录:存放合约字节码文件
  • java目录:存放生成的Java合约类

生成的Asset.java包含了与Solidity合约对应的Java接口:

package org.fisco.bcos.asset.contract;
public class Asset extends Contract {// 转账接口public RemoteCall<TransactionReceipt> transfer(String from_account, String to_account, BigInteger amount);// 注册接口public RemoteCall<TransactionReceipt> register(String account, BigInteger asset_value);// 查询接口public RemoteCall<Tuple2<BigInteger, BigInteger>> select(String account);// 加载合约public static Asset load(String contractAddress, Web3j web3j, Credentials credentials, ContractGasProvider contractGasProvider);// 部署合约public static RemoteCall<Asset> deploy(Web3j web3j, Credentials credentials, ContractGasProvider contractGasProvider);
}

五、SDK配置与项目搭建

5.1 获取Java工程项目

# 下载项目压缩包
cd ~
curl -LO https://github.com/FISCO-BCOS/LargeFiles/raw/master/tools/asset-app.tar.gz
# 解压项目
tar -zxf asset-app.tar.gz

5.2 项目目录结构

asset-app/
├── build.gradle          # Gradle配置文件
├── gradle/
│   ├── wrapper/
│   │   ├── gradle-wrapper.jar
│   │   └── gradle-wrapper.properties
├── gradlew               # Linux执行脚本
├── gradlew.bat           # Windows执行脚本
├── src/
│   ├── main/
│   │   ├── java/
│   │   │   └── org/
│   │   │       └── fisco/
│   │   │           └── bcos/
│   │   │               └── asset/
│   │   │                   ├── client/      # 客户端调用类
│   │   │                   │   └── AssetClient.java
│   │   │                   └── contract/    # 合约Java类
│   │   │                       └── Asset.java
│   │   └── resources/
│   │       ├── applicationContext.xml       # 项目配置文件
│   │       ├── contract.properties          # 合约地址配置
│   │       ├── log4j.properties             # 日志配置
│   │       └── contract/
│   │           ├── Asset.sol                # Solidity合约
│   │           └── Table.sol
├── tool/
│   └── asset_run.sh     # 运行脚本

5.3 引入Web3SDK

项目已在build.gradle中配置好Web3SDK依赖:

repositories {maven { url "http://maven.aliyun.com/nexus/content/groups/public/" }maven { url "https://dl.bintray.com/ethereum/maven/" }mavenCentral ()
}dependencies {compile('org.fisco-bcos:web3sdk:2.1.0')
}

5.4 证书与配置文件

拷贝区块链节点的SDK证书到项目资源目录:

cp fisco/nodes/127.0.0.1/sdk/* asset-app/src/test/resources/

六、业务开发:Java客户端实现

6.1 核心类设计:AssetClient

AssetClient.java是业务逻辑的核心,负责合约的部署与调用,主要包含以下功能模块:

6.1.1 初始化
// 初始化Web3j和Credentials对象
ApplicationContext context = new ClassPathXmlApplicationContext("classpath:applicationContext.xml");
Service service = context.getBean(Service.class);
service.run();ChannelEthereumService channelEthereumService = new ChannelEthereumService();
channelEthereumService.setChannelService(service);
Web3j web3j = Web3j.build(channelEthereumService, 1);Credentials credentials = Credentials.create(Keys.createEcKeyPair());
6.1.2 合约对象创建
// 部署合约
Asset asset = Asset.deploy(web3j, credentials, new StaticGasProvider(gasPrice, gasLimit)).send();// 加载已部署合约
Asset asset = Asset.load(contractAddress, web3j, credentials, new StaticGasProvider(gasPrice, gasLimit));
6.1.3 接口调用
// 查询资产
Tuple2<BigInteger, BigInteger> result = asset.select(assetAccount).send();// 注册资产
TransactionReceipt registerReceipt = asset.register(assetAccount, amount).send();// 资产转账
TransactionReceipt transferReceipt = asset.transfer(fromAccount, toAccount, amount).send();

七、项目运行与功能验证

7.1 编译项目

# 切换到项目目录
cd ~/asset-app
# 编译项目
./gradlew build

编译成功后在dist目录生成运行脚本。

7.2 部署合约

cd dist
bash asset_run.sh deploy
# 输出示例:Deploy Asset successfully, contract address is 0xd09ad04220e40bb8666e885730c8c460091a4775

7.3 注册资产

bash asset_run.sh register Alice 100000
# 输出:Register account successfully => account: Alice, value: 100000bash asset_run.sh register Bob 100000
# 输出:Register account successfully => account: Bob, value: 100000

7.4 查询资产

bash asset_run.sh query Alice
# 输出:account Alice, value 100000bash asset_run.sh query Bob
# 输出:account Bob, value 100000

7.5 资产转移

bash asset_run.sh transfer Alice Bob 50000
# 输出:Transfer successfully => from_account: Alice, to_account: Bob, amount: 50000bash asset_run.sh query Alice
# 输出:account Alice, value 50000bash asset_run.sh query Bob
# 输出:account Bob, value 150000

八、参考资料

  • FISCO BCOS文档:https://www.bookstack.cn/read/fisco-bcos-2.4-zh
http://www.lryc.cn/news/575098.html

相关文章:

  • PAC 学习框架:机器学习的可靠性工程
  • 通俗易懂解读BPE分词算法实现
  • 回归预测 | Matlab实现KAN神经网络多输入单输出回归预测模型
  • 轻巧灵动,智启未来 ——Kinova Gen3 Lite 机器人轻松解锁各行业自动化新姿势
  • 领域驱动设计(DDD)【13】之重构中的坏味道:深入理解依恋特性(Feature Envy)与表意接口模式
  • 香港电讯携手Palo Alto Networks,护航企业跨区域数字化之旅
  • 第8章项目进度管理归纳总结补充
  • 英飞凌高性能BMS解决方案助力汽车电动化
  • python学智能算法(十六)|机器学习支持向量机简单示例
  • 基于MATLAB图像特征识别及提取实现图像分类
  • spring event(spring事件)
  • 从 HLS 到 Verilog 的转变解析1:以 AXI 接口为例
  • 云原生灰度方案对比:服务网格灰度(Istio ) 与 K8s Ingress 灰度(Nginx Ingress )
  • jenkins 越用越卡,打开网页缓慢
  • CLion 调试时 Command Timed Out 问题解决方案
  • 深入剖析 Spring AOP
  • 红外图像增强(dde):基于“基础层-细节层”分解的增强算法
  • 5. Pytest失败重跑机制pytest-rerunfailures
  • LE AUDIO---Chapter 2. The Bluetooth® LE Audio architecture
  • AR/VR 显示画质失真?OAS 体全息光栅案例来解决
  • Linux系统之Nginx反向代理与缓存
  • 鸿蒙Next仓颉开发语言中的数据类型总结分享
  • 【计算机网络】第二章:物理层
  • 掌握多门计算机语言之后,如何高效学习新语言与切换编程思维
  • 在 GitLab CI 中配置多任务
  • 《从0到1:C/C++音视频开发自学指南》
  • SQL学习笔记2
  • 论文阅读:arxiv 2025 ThinkSwitcher: When to Think Hard, When to Think Fast
  • 通过 HTML 子图和多尺度卷积 BERT 的双向融合实现可解释的恶意 URL 检测
  • npm 报错:“无法加载文件 ...npm.ps1,因为在此系统上禁止运行脚本” 解决方案(附执行策略说明)