脚本是一种简单的脚本语言,也是比特币交易处理的核心。若是你曾经写过汇编代码,你会发现这篇文章很容易理解,并且多是有趣的;不然它多是特别具挑战性的。因此请保持专一!php
脚本是计算机程序,做为程序员,你固然知道程序是什么。程序接受输入,执行一段时间,而后返回输出。编程语言是咱们编写计算机能理解的程序的工具,由于大多数语言都带有编译器,能够将人性化的代码映射到CPU来操做,因此也称为操做码。java
操做码包括内存操做,数学,循环,函数调用以及在程序编程语言(如C)中找到的全部内容。它们构成CPU的口语,即所谓的机器码。因为字节是计算机的首选习惯用法,所以操做码也是字节。结果就是,机器码表示要在CPU上执行的操做的字节串。node
在像C这样的高级编程语言中考虑这段代码:python
x = 0x23; x += 0x4b; x *= 0x1e;
如今假设你要在假设的小尾数的CPU上编译和运行此代码,该CPU具备16位内存(寄存器)的单个单元和如下操做码集:android
opcode | encoding | V |
---|---|---|
SET(V) | ab V | 16-bit |
ADD(V) | ac V | 16-bit |
MUL(V) | ad V | 16-bit |
操做码解释以下:程序员
这种CPU的编译器将生成这9个字节的机器代码:web
ab 23 00 ac 4b 00 ad 1e 00
如下是它的解释方式:mongodb
寄存器保存最终结果,即ce4。编程
大多数状况下,咱们须要使用变量跟踪复杂的程序状态。在C中,根据变量是静态分配仍是使用malloc分配,它存储在不一样排列的内存中。虽然malloc-ed数据像一个很是大的数组中的元素同样被访问,但静态变量被推送到一堆名为stack的项目中并从中弹出。堆栈以LIFO方式运行(后进先出),这意味着你推送的最后一个项目将是第一个弹出的项目。数组
考虑这个虚函数:
int foo() { /* 1 */ /* 2 */ uint8_t a = 0x12; uint16_t b = 0xa4; uint32_t c = 0x2a5e7; /* 3 */ uint32_t d = a + b + c; return d; /* 4 */ }
堆栈最初是空的(1):
[]
而后,推送三个变量(2):
[12] [12, a4 00] [12, a4 00, e7 a5 02 00]
第四个变量被赋予其余变量的总和并被推入堆栈(3):
[12, a4 00, e7 a5 02 00, 9d a6 02 00]
堆栈的尖端是返回值,并经过其余方式发送回函数调用者。每一个临时堆栈变量都会在块(4)的末尾弹出,由于必须平衡推push/弹pop操做,以便堆栈始终返回其初始状态:
[12, a4 00, e7 a5 02 00] [12, a4 00] [12] []
一样,比特币核心有本身的“虚拟处理器”来解释脚本机器码。脚本具备丰富的操做码,但与英特尔等彻底成熟的CPU相比却很是有限。关于脚本的一些关键事实:
实际上,第1点也意味着第2点。第3点意味着在Script中没有像命名变量这样的东西,你只需在堆栈上进行计算。一般,你推送的堆栈项成为后续操做码的操做数。在脚本的末尾,顶部堆栈项是返回值。
在介绍现实世界的脚本以前,让咱们先列举一些操做码。如需全套,请查看比特官方维基页面。
如下操做码将数字0-16推入堆栈:
opcode | encoding |
---|---|
OP_0 | 00 |
OP_1-OP_16 | 51-60 |
按照惯例,OP_0
和OP_1
也表示布尔值OP_FALSE
(零)和OP_TRUE
(非零)。
例:
54 57 00 60
或者:
OP_4 OP_7 OP_0 OP_16
这是堆栈如何发展:
[] [4] [4, 7] [4, 7, 0] [4, 7, 0, 16]
返回值是最高项,所以脚本返回16。我知道,这是毫无心义的,但这是一个开始。
提供了几个操做码来推送自定义数据。它们的操做数大小不一样:
opcode | encoding | L (length) | D (data) |
---|---|---|---|
OP_PUSHDATA1 | 4c L D | 8-bit | L bytes |
OP_PUSHDATA2 | 4d L D | 16-bit | L bytes |
OP_PUSHDATA4 | 4e L D | 32-bit | L bytes |
例如,若是你的数据长度能够存储为8位数字,那么OP_PUSHDATA1
是你的最佳选择。看这个:
4c 14 11 06 03 55 04 8a 0c 70 3e 63 2e 31 26 30 24 06 6c 95 20 30
第一个字节显然是OP_PUSHDATA1
操做码,后面是1字节长度14,即十进制20.所以,接下来会有20个字节的数据。这条指令的做用是将这些数据压入堆栈:
[11 06 03 55 04 8a 0c 70 3e 63 2e 31 26 30 24 06 6c 95 20 30]
实际上,与varints同样,对于很是短的数据有一种特殊的编码。若是操做码位于01和4b(包括)之间,则它是一个推送数据操做,其中操做码自己是以字节为单位的长度:
opcode | encoding | L (length) | D (data) |
---|---|---|---|
L | L D | 01-4b | L bytes |
例如,在字符串中:
07 8f 49 b2 e2 ec 7c 44
操做码07意味着要推送7个字节的数据:
[8f 49 b2 e2 ec 7c 44]
区块链中的下一个块呢?
你学到了一些关于机器代码和操做码的知识。脚本是矿工软件理解的简单低级语言。使用堆栈内存跟踪脚本状态。
在下一篇文章中,我将向你展现操做代码,它不只仅是推送数据。 若是你喜欢它,请分享这篇文章!
======================================================================
分享一些以太坊、EOS、比特币等区块链相关的交互式在线编程实战教程:
- java以太坊开发教程,主要是针对java和android程序员进行区块链以太坊开发的web3j详解。
- python以太坊,主要是针对python工程师使用web3.py进行区块链以太坊开发的详解。
- php以太坊,主要是介绍使用php进行智能合约开发交互,进行帐号建立、交易、转帐、代币开发以及过滤器和交易等内容。
- 以太坊入门教程,主要介绍智能合约与dapp应用开发,适合入门。
- 以太坊开发进阶教程,主要是介绍使用node.js、mongodb、区块链、ipfs实现去中心化电商DApp实战,适合进阶。
- C#以太坊,主要讲解如何使用C#开发基于.Net的以太坊应用,包括帐户管理、状态与交易、智能合约开发与交互、过滤器和交易等。
- EOS教程,本课程帮助你快速入门EOS区块链去中心化应用的开发,内容涵盖EOS工具链、帐户与钱包、发行代币、智能合约开发与部署、使用代码与智能合约交互等核心知识点,最后综合运用各知识点完成一个便签DApp的开发。
- java比特币开发教程,本课程面向初学者,内容即涵盖比特币的核心概念,例如区块链存储、去中心化共识机制、密钥与脚本、交易与UTXO等,同时也详细讲解如何在Java代码中集成比特币支持功能,例如建立地址、管理钱包、构造裸交易等,是Java工程师不可多得的比特币开发学习课程。
- php比特币开发教程,本课程面向初学者,内容即涵盖比特币的核心概念,例如区块链存储、去中心化共识机制、密钥与脚本、交易与UTXO等,同时也详细讲解如何在Php代码中集成比特币支持功能,例如建立地址、管理钱包、构造裸交易等,是Php工程师不可多得的比特币开发学习课程。
- tendermint区块链开发详解,本课程适合但愿使用tendermint进行区块链开发的工程师,课程内容即包括tendermint应用开发模型中的核心概念,例如ABCI接口、默克尔树、多版本状态库等,也包括代币发行等丰富的实操代码,是go语言工程师快速入门区块链开发的最佳选择。
汇智网原创翻译,转载请标明出处。这里是原文比特币脚本语言