嗯,说好的智能合约呢?怎么开始转账了?
因为,UTXO转账是所有合约的基础,你一定听说过交易手续费这个概念,那么交易手续费怎么支付呢?
这就是个问题了对吧。
UTXO这个事情本身就是和转账是一体化的,之前我们在基础知识哪里讲过UTXO,但是那里是讲不清楚的,因为不结合转账来说UTXO那是不可能说的清楚的。
UTXO的名字 unspent transaction output,未花费交易输出
我们之前提到过找零的问题,账户A给B转账,需要输入账户A的n个UTXO,然后一个输出给B多少钱,剩余的钱一个输出给A自己。
如果交易的输出总额和输入总额不匹配的时候,如果输出总额>输入总额,交易会失败。
如果输出总额<输入总额会怎样?你的钱捐给系统了,就是这样。
要支付交易手续费,就把他捐给系统吧。
然后我们来发一笔转账吧,首先讲一讲怎么发?我们要写一个自己的轻钱包来发,怎么写,用这个
https://github.com/NewEconoLab/neo-thinsdk-cs
这个项目提供了neo所需的那些加密运算,你需要生成转账的交易、签名、然后把他发布到链上。
1.生成转账交易
最终转账交易是个二进制的数据块,怎么生成这个二进制的块,我们通常会写一个数据结构,然后提供序列化方法,把数据结构的内容变成数据块。
这个结构叫做Transaction,thinsdk 当然提供了transaction结构和序列化方法,你也可以阅读neo官方文档,其中有对二进制数据构成的描述。你也可以阅读neo源代码,此处,只要会用就好。
class Transactoion
{
type//交易类型
vertion//交易版本 目前只有 0 1
[type extdata]//每种交易类型独有的数据,比如调用应用合约交易,就得有调用脚本
attributes//交易属性(0到多个),可以附加一些东西,比如注释啦,除输入以外的附加鉴证人啦
inputs//交易的输入(0到多个)(utxo销毁)
outputs//交易的输出(0到多个)(utxo产生)
witnesses;//鉴证人(0到多个)
}
要填充一笔转账交易,输入是一个问题,你要怎么知道一个账户到底有哪些utxo呢?thinsdk不管,api管,看问后例子就好。
2.签名交易
嗯,这里产生了一个概念,NEO把交易分为未签名交易与已签名交易。
不含鉴证人部分的交易称为未签名交易,那自然含有鉴证人部分的交易称为签名交易咯。
签好名的交易序列化为二进制块,广播,这个行为就叫做发交易
而签名交易其实是一个非常不准确的说法,只是签名这个说法大部分人更容易理解
签名过程是给交易填上鉴证人数据,script attribute 和 inputs 会导致需要鉴证人,大部分人理解的就是:因为inputs来自我的账户,所以需要我的签名。因为要用我的钱,所以需要我签名。
签名,就是将该交易的未签名部分序列化为二进制块,然后调用ecc算法签个名,把这个数据加点东西加入签名的witnesses中。
实际上算签名只是鉴证的一个特例。
交易签名只是用了特例的名字命名了整个行为,交易签名并不是一定在签名,只有需要的鉴证人要求签名时才签名。
这就讲到智能合约的鉴证触发器那里了,这个话题有空再讲,让我们接着聊转账
3.发布交易
签好名的交易序列化为二进制块,广播,这个行为就叫做发交易。
怎么广播?thinsdk不管,api管,看例子就好。
这里是例子
在smartcontractDemo中,有叫做转账的例子,这是一个netcore 命令行程序。
1.拼合约
例子中用了下面这个接口来获取一个账户的所有utxo
Helper.HttpGet(api + "?method=getutxo&id=1¶ms=['" + _addr + "']")
记得吗,rpc接口不能直接给你这个,你就没法拼转账交易,所以要爬虫要api。
此处的api由NEL提供,不要问我api是多少,他可能会变,但是smartcontractDemo这个代码里会保证api肯定是测试过的。
看代码
然后你可以看看我们是怎么拼装一个转账交易的,有maketran方法,就是填充inputs outputs那些,不再赘述。
2.签合约
byte[] msg = tran.GetMessage();//tran是交易,getmessage是得到未签名交易的二进制数据块
byte[] signdata = ThinNeo.Helper.Sign(msg, prikey);//签名算法,私钥什么的都是写死的,sdk也支持从neo json格式打开私钥,这个不介绍了。
tran.AddWitness(signdata, pubkey, address);//添加普通账户鉴证人,私钥公钥地址,全都是一个人
string txid = tran.GetHash().ToString();
byte[] data = tran.GetRawData();//得到签名交易的二进制数据块
直接如代码
3.发交易
string rawdata = ThinNeo.Helper.Bytes2HexString(data);
byte[] postdata;
var url = Helper.MakeRpcUrlPost(api, "sendrawtransaction", out postdata, new MyJson.JsonNode_ValueString(rawdata));
var result = await Helper.HttpPost(url, postdata);