区块链钱包—BTC Java版离线签名交易

article/2025/9/11 21:46:53

对于离线交易不做过多解释~,说白了就是拿上一笔未发出交易记录进行私钥的签名然后广播到链上。

主要是对区块链离线交易进行utxo上链。

代码参考: https://gitee.com/DHing/signature-transaction

UnspentUtxo交易查询参考:https://blog.csdn.net/u011663149/article/details/91855077

 

 离线签名(非隔离见证地址):

import java.util.ArrayList;
import java.util.List;import org.apache.commons.codec.binary.Hex;
import org.apache.commons.configuration2.Configuration;
import org.bitcoinj.core.Address;
import org.bitcoinj.core.Coin;
import org.bitcoinj.core.Context;
import org.bitcoinj.core.DumpedPrivateKey;
import org.bitcoinj.core.ECKey;
import org.bitcoinj.core.NetworkParameters;
import org.bitcoinj.core.Sha256Hash;
import org.bitcoinj.core.Transaction;
import org.bitcoinj.core.TransactionOutPoint;
import org.bitcoinj.core.UTXO;
import org.bitcoinj.core.Utils;
import org.bitcoinj.params.MainNetParams;
import org.bitcoinj.params.TestNet3Params;
import org.bitcoinj.script.Script;
import org.omg.CORBA.UNKNOWN;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;import com.alibaba.fastjson.JSON;
import com.bscoin.coldwallet.cointype.common.ConfigUtil;
import com.bscoin.coldwallet.cointype.common.UnSpentUtxo;import org.bitcoinj.core.TransactionConfidence;/**  * @ClassName: RawTransaction  * @author DHing  *    
*/  
public class RawTransaction {private static Logger LOG = LoggerFactory.getLogger(RawTransaction.class);static NetworkParameters params;static {try {Configuration config = ConfigUtil.getInstance();params = config.getBoolean("bitcoin.testnet") ? TestNet3Params.get() : MainNetParams.get();LOG.info("=== [BTC] bitcoin  client networkID:{} ===", params.getId());} catch (Exception e) {LOG.info("=== [BTC] com.bscoin.coldwallet.cointype.btc.rawtransaction:{} ===", e.getMessage(), e);}}/**  * @Title: signTransaction* @param @param privKey 私钥* @param @param recevieAddr 收款地址* @param @param formAddr 发送地址* @param @param amount 金额* @param @param fee 手续费(自定义 或者 默认)* @param @param unUtxos 未交易的utxo* @param @return    参数  * @return char[]    返回类型  * @throws  */  public static String signTransaction(String privKey, String recevieAddr, String formAddr, long amount, long fee, List<UnSpentUtxo> unUtxos) {if(!unUtxos.isEmpty() && null != unUtxos){List<UTXO> utxos = new ArrayList<UTXO>();// String to a private keyDumpedPrivateKey dumpedPrivateKey = DumpedPrivateKey.fromBase58(params, privKey);ECKey key = dumpedPrivateKey.getKey();// 接收地址Address receiveAddress = Address.fromBase58(params, recevieAddr);// 构建交易Transaction tx = new Transaction(params);tx.addOutput(Coin.valueOf(amount), receiveAddress); // 转出// 如果需要找零 消费列表总金额 - 已经转账的金额 - 手续费long value = unUtxos.stream().mapToLong(UnSpentUtxo::getValue).sum();Address toAddress = Address.fromBase58(params, formAddr);long leave  = value - amount - fee;if(leave > 0){tx.addOutput(Coin.valueOf(leave), toAddress);}// utxos is an array of inputs from my walletfor (UnSpentUtxo unUtxo : unUtxos) {utxos.add(new UTXO(Sha256Hash.wrap(unUtxo.getHash()),unUtxo.getTxN(),Coin.valueOf(unUtxo.getValue()), unUtxo.getHeight(), false,new Script(Utils.HEX.decode(unUtxo.getScript())),unUtxo.getAddress()));}for (UTXO utxo : utxos) {TransactionOutPoint outPoint = new TransactionOutPoint(params, utxo.getIndex(), utxo.getHash());// YOU HAVE TO CHANGE THIStx.addSignedInput(outPoint, utxo.getScript(), key, Transaction.SigHash.ALL, true);}Context context = new Context(params);tx.getConfidence().setSource(TransactionConfidence.Source.NETWORK);tx.setPurpose(Transaction.Purpose.USER_PAYMENT);LOG.info("=== [BTC] sign success,hash is :{} ===",tx.getHashAsString());return new String(Hex.encodeHex(tx.bitcoinSerialize()));}return null;}public static void main(String[] args) {List<UnSpentUtxo> us = new ArrayList<UnSpentUtxo>();UnSpentUtxo u = new UnSpentUtxo();u.setAddress("mifiHFYFPk5cri4oneXVsRZJZKovvdDcjo");u.setHash("2bc6ac92468c2b4f1fcd2349822dc4663dfc0705b30131087a20ed8d17de8274");u.setHeight(	1413239);u.setScript("76a914a1806613a51a81966779e2fa1537013cf4cd2b1788ac");u.setTxN(1);u.setValue(100000);UnSpentUtxo u1 = new UnSpentUtxo();u1.setAddress("mvEtuEqYPMrLaKjJ5nTZ57vQAoYUtVmMaQ");u1.setHash("1893b6ff8ef2bd6f5d652937ffbaed5bb669c5d9ab450066253d6692f2d4d972");u1.setHeight(1413334);u1.setScript("76a914a1806613a51a81966779e2fa1537013cf4cd2b1788ac");u1.setTxN(1);u1.setValue(400000);us.add(u);us.add(u1);System.out.println(JSON.toJSONString(us));String c = signTransaction("cNRE3D1pbPPvGs9wpZd3X9NuLsuUQPzPa7ktQyF1nhqBabraocU9", "mifiHFYFPk5cri4oneXVsRZJZKovvdDcjo", "mvEtuEqYPMrLaKjJ5nTZ57vQAoYUtVmMaQ", 400000, 10000, us);System.out.println(c);}
}
package com.bscoin.coldwallet.cointype.common;import java.io.Serializable;public class UnSpentUtxo implements Serializable {private static final long serialVersionUID = -7417428486644921613L;private String hash;//交易hashprivate long txN; //private long value;//金额private int height;//区块高度private String script;//hexprivate String address;//钱包地址public String getHash() {return hash;}public void setHash(String hash) {this.hash = hash;}public long getTxN() {return txN;}public void setTxN(long txN) {this.txN = txN;}public long getValue() {return value;}public void setValue(long value) {this.value = value;}public int getHeight() {return height;}public void setHeight(int height) {this.height = height;}public String getScript() {return script;}public void setScript(String script) {this.script = script;}public String getAddress() {return address;}public void setAddress(String address) {this.address = address;}}

签名成功返回Hex,使用https://live.blockcypher.com/btc-testnet/decodetx/ 进行解码查看交易详情:

最后:

      调用节点钱包api: sendrawtransaction(hex:object) 传入生成的Hex即可进行广播交易了。


http://chatgpt.dhexx.cn/article/sP6S2Bgo.shtml

相关文章

Zcash钱包(ZEC钱包)官方客户端常见命令实测(一)

一、如何查询链接节点 $ ./src/zcash-cli getpeerinfo 二、如何获取一个 t- 开头的钱包地址 $ ./src/zcash-cli getnewaddress 三、如何获取一个 z- 开头的钱包地址 $ ./src/zcash-cli z_getnewaddress

通过代码生成以太坊助记词、根据钱包地址获取私钥

生成助记词、公私钥 新建一个目录 在目录下执行 npm init &#xff08;需安装nodejs&#xff09;nodejs中文官网 安装依赖 &#xff08;没有yarn的需要安装&#xff0c;直接执行 npm install -g yarn&#xff09; yarn add bip39 ethereum-hdwallet 在目录下新建js文件 将下…

windows下的Zcash钱包(ZEC钱包)-zcash4win 1.0.12

官方下载地址&#xff1a;https://zcash4win.com/ 百度云盘下载地址&#xff1a;https://pan.baidu.com/s/1i46zcSt 密码: by6b QQ群交流分享&#xff1a;480956296

windows下的Zcash钱包(ZEC钱包)-zcash4win 1.0.11

官方下载地址&#xff1a;https://zcash4win.com/ 下载完后&#xff0c;直接下一步下一步&#xff0c;安装即可 技巧&#xff1a; 1、安装好后打开&#xff0c;开始下载文件&#xff0c;然后关闭 2、下载这两个文件&#xff1a;https://s3.amazonaws.com/zcashfinalmpc/sprou…

可能最详细的教程,新手如何获取Zcash钱包(ZEC钱包)官方客户端地址的方法

折腾了2天&#xff0c;终于折腾出了Zcash的官方地址了&#xff08;非第三方交易平台的&#xff09; 先上个图&#xff0c;有图有真相嘛&#xff0c;哈哈&#xff0c;图片有水印&#xff0c;看下面那个命令也是获取钱包地址的 好了&#xff0c;开始上教程&#xff0c;我这边使用…

今天终于找到了一款windows下的Zcash钱包(ZEC钱包),推荐给大家

折腾了好几天&#xff0c;想说下linux真心不适合我这类新人&#xff0c;所以导出百度&#xff0c;终于在官网看到一个链接 MacOS&#xff0c;想来这个应该是苹果系统的&#xff0c;有苹果&#xff0c;肯定也会有Windows的&#xff0c;于是点进去继续淘宝 后台请教朋友&#xff…

可能最详细的教程,新手如何获取Zcash钱包(ZEC)官方地址的方法

折腾了2天&#xff0c;终于折腾出了Zcash的官方地址了&#xff08;非第三方交易平台的&#xff09; 先上个图&#xff0c;有图有真相嘛&#xff0c;哈哈&#xff0c;图片有水印&#xff0c;看下面那个命令也是获取钱包地址的 好了&#xff0c;开始上教程&#xff0c;我这边使用…

Python中使用中文正则表达式匹配指定的中文字符串

业务场景&#xff1a; 从中文字句中匹配出指定的中文子字符串 .这样的情况我在工作中遇到非常多, 特梳理总结如下. 难点: 处理GBK和utf8之类的字符编码, 同时正则匹配Pattern中包含汉字,要汉字正常发挥作用,必须非常谨慎.推荐最好统一为utf8编码,如果不是这种最优情况,也有酌…

验证中文名字---正则表达式

验证中文名字—正则表达式 效果 正确演示 错误演示 运用知识点 正则表达式 ps http://www.bejson.com/convert/unicode_chinese/ unicode编码&#xff08;了解&#xff09; 打开浏览器 F12 控制台 输入 escape(“淦”) unescape("%u6DE6") 在这里插入图片描…

在js中用正则匹配中文

在JavaScript中&#xff0c;用正则表达式如何匹配中文字符&#xff08;包括繁体、简体&#xff09;&#xff1f; 答案就是用中文字符集来匹配&#xff0c;中文字符集为[\u4e00-\u9fa5] <!DOCTYPE html> <!-- 文档申明&#xff1a;让浏览器用H5的标准解析当前网页 --…

正则表达式 判断是否是中文

&#xff08;一&#xff09;字母、数字、下划线、汉字正则表达式 1. 只有字母、数字和下划线且不能以下划线开头和结尾的正则表达式&#xff1a;^(?!_)(?!.*?_$)[a-zA-Z0-9_]$ 只有字母和数字的&#xff1a; ^[a-zA-Z0-9_]$ 2. 至少一个汉字、数字、字母、下划线: &qu…

正则中文匹配

这篇文章主要讲如何使用正则匹配中文字符&#xff0c;中文正则表达式的匹配规则不像其他正则规则一样容易记住&#xff0c;下面一起看看这个中文正则表达式是怎么样的。 \w匹配的仅仅是中文&#xff0c;数字&#xff0c;字母&#xff0c;对于国人来讲&#xff0c;仅匹配中文…

js正则中文

hi&#xff0c;大家好   今天跟小伙伴们浅谈以下如何用正则表示中文以及如何去运用。众所周知中文在计算机中是不能进行存储的。那我们是以什么办法让我们和计算机进行更好的沟通呢&#xff1f; 常用的几种中文编码格式 utf-8编码   utf-8又称“万国码”&#xff0c;可以同…

python正则如何匹配中文汉字

正则表达式匹配中文汉字&#xff0c;在实际应用中十分常见。 比如&#xff1a;爬虫网页文本提取、验证用户输入标准等。 以下面文本字符串为例&#xff0c;匹配出astr这个字符串中的所有汉字。 import re astr aaaaa何时when 杖尔看see南雪snow&#xff0c;我me与梅花plum bl…

Guava: Joiner

Joiner 字符串拼接对象 on 基本实例 Joiner.on(" ").join("四川省","成都市","金牛区").toString() 1 on 以空格为分隔符join内是拼接的对象:迭代器对象、数组、可变参数等 result 四川省 成都市 金牛区 1 特殊处理 动态拼接…

Google Guava中Joiner用法

Google Guava中Joiner用法 一&#xff1a;Google Guava的下载地址链接&#xff1a; Google Guava的jar包下载地址&#xff1a;https://pan.baidu.com/s/1B2IjhOlrulOgwceG1TehKQ 密码&#xff1a;2qha 1&#xff1a;java application项目引入下载的jar包。 2&#xff1a;we…

【第一章】google guava 之 Joiner学习

Joiner是将数组按照某分隔符分隔返回字符串 com.google.common.base.Joiner#on(java.lang.String) 用户构造函数设置用什么字符分隔 com.google.common.base.Joiner#appendTo(A, java.lang.Iterable<? extends java.lang.Object>) 用于传入一个集合&#xff0c;然后通…

001 Joiner

一.概述 本类的名称的含义就是连接器,可以帮助我们实现数据(对象)的连接. 二 . 对象的创建   Testpublic void test1() {Joiner joiner Joiner.on(",");} 创建一个Joiner对象,我们可以使用on()方法,含义就是会用这个分隔符进行连接数据. 我们可以发现我们实际上都…

mp4 joiner linux,MP4Joiner怎么用?使用MP4Joiner快速合并多个mp4视频文件的方法介绍

如何使用MP4Joiner快速合并多个mp4视频文件&#xff1f;MP4Joiner是一款专业小巧的MP4视频合并软件&#xff0c;支持将多个MP4视频合并为一个&#xff0c;支持自定义设置视频的前后顺序&#xff0c;速度非常快&#xff0c;也不会损坏视频质量&#xff0c;非常不错。接下来小编就…

1、Guava-连接器Joiner使用和源码分析

Guava-连接器Joiner使用和源码分析 1.Guava-连接器Joiner使用和源码分析1.1 使用版本1.2 代码示例1.2.1 基本使用1.2.2 集合中Null导致空指针异常1.2.3 忽略集合中的null1.2.4 用默认值代替集合中的null1.2.5 添加至Appendable中1.2.6 连接Map中的key和value1.2.7 使用Stream流…