move语言的特点

Libra的目标是打造一个全球话的金融和货币的平台,从而赋能地球上的几十亿人。那么作为Libra的move语言就需要在安全性和可编程性上面提供大力的支持。

为了实现这样的目标,move遵从如下四大设计原则:

资源优先

在区块链中我们需要通过程序来存取数字资产,这个数字资产和普通程序中的变量(如booleans,integers,strings)有着明显的区别。我们需要一个特别的方式来表示区块链中的数字资产。

Move语言中的resource就是特别为数字资产定义的,它有这样的特点:

resource是不能被拷贝和隐式销毁的,它只能在不同的用户之间移动。这个特性是在Move的类型定义中已经声明了的。除了这个特殊的限制,resource和其他普通的资源一样,可以存储在数据结构中,可以作为参数传给过程等。

资源优先的概念为程序员写出安全和有效的代码提供了非常大的帮助。

Libra coin就是一种resource,因为它将和现实世界的货币相对应,所以它可以被创建,修改,或者销毁。我们需要做的就是通过modules来控制好操作coin的权限。

move中的modules有点像以太坊中的智能合约,module声明了resource类型和过程(业务逻辑:如何创建,移动,销毁coin)。module中的定义的type和过程可以被其他的module进行调用。

灵活性

move的灵活性体现在可以通过transaction脚本来自由组合各种transaction来实现不同的功能,一个脚本可以调用多个transaction,

在move中modules/resources/procedures的关系有点像面向对象程序语言中classes/objects/methods的关系。

安全性

move定义了资源的安全性,类型的安全性和内存的安全性,任何违背这些安全性的操作都会被拒绝。

一般来说,有两种方式来实现这个功能:1.在高级别的语言定义上,通过编译器来检查这些异常。 2. 在低级别的虚拟机汇编语言上检查这些异常。

move是这两种方式中间的一种方式:Move执行的字节码比汇编语言要高级一点,但是又比编程语言又低级一点。move的字节码提交到链上后,会被字节码验证器校验,然后经由字节码解释器执行。

可验证性

最好的验证方式就是将字节码提交到链上进行真实的验证,但这样很明显会加重链的负担,影响交易的速度,所以在move中我们尽可能多的在链上做轻量级的验证,而在语言级别做线下的静态验证。

Move语句初探

本节我们会通过一个例子来详细的讲解Move语言的具体特性。本节的例子是使用move IR来编写的,这是一个实验性的版本,正式的move语言还在编写中。

点对点支付交易脚本

public main(payee: address, amount: u64) {
let coin: 0x0.Currency.Coin = 0x0.Currency.withdraw_from_sender(copy(amount));
0x0.Currency.deposit(copy(payee), move(coin));
}

上面是一段简单的交易脚本,main方法有两个输入,一个是地址,一个是要支付的数目。

逻辑其实很简单,从发送者的账户里面减去amount, 然后将它转给接收者。

0x0是module的存储地址,Currency是module的名字,0x0.Currency.Coin 代表的就是资源了。

在上面的章节中我们讲到了,coin是resource,只能move不能copy,尝试将move(coin)替换成copy(coin)会报错。

其他的非resource的资源像是payee和amout可以被move也可以被copy.

如果我们添加这样一行:

0x0.Currency.deposit(copy(some_other_payee), move(coin))

则coin将会被使用两次,这在逻辑上是有问题的,在Move语言中,

move(coin)方法在使用一次过后会变得不可用,从而导致第二次move失败,从而有效的保障了应用逻辑。

coin只能也必须移动一次,如果把move(coin)删除,同样会的到一个错误。这样做的目的是有效的避免程序员的疏忽导致的应用逻辑错误。

Currency Module

上面我们定义了一个Currency的module。这里我们讲一下Currency是怎么实现的。

Libra有两种程序,一种是transaction脚本,一种是module,脚本会去调用module里面的过程来更新全局状态。

transaction脚本是一种只能执行一次的脚本,执行完之后就不能再用了,而module是发布在全局状态里面的长期存在的代码。

全局状态是一个账户地址到账户的映射,如下图所示,0x0,0x1,0x2都是账户的地址。

每个地址里面可以包含任意个module和资源。例如0x0地址的账户包含了一个module:0x0.Currency和一个resource:0x0.Currency.Coin.

知道了账户的结构之后,我们看下module是怎么定义资源的:

module Currency {
resource Coin { value: u64 }
// ...
}

上面的代码定义了一个Coin的resource,它有一个value字段,类型是u64。

接下来我们看下存储这个操作是怎么定义的:

public deposit(payee: address, to_deposit: Coin) {
let to_deposit_value: u64 = Unpack<Coin>(move(to_deposit));
let coin_ref: &mut Coin = BorrowGlobal<Coin>(move(payee));
let coin_value_ref: &mut u64 = &mut move(coin_ref).value;
let coin_value: u64 = *move(coin_value_ref);
*move(coin_value_ref) = move(coin_value) + move(to_deposit_value);
}

这个过程主要做了这样几件事情:

  1. 销毁输入的Coin,并且记录下它的值。
  2. 获取存在payee中的Coin的reference。
  3. 把输入的Coin加到payee上。

Unpack<T> 是唯一的销毁T的方式,unpack会销毁T,然后返回T对应的值。

BorrowGlobal<T> 接收一个地址作为参数,然后返回一个指向该地址里的T实例的引用。

同样的,我们看下withdraw_from_sender 是怎么实现的:

public withdraw_from_sender(amount: u64): Coin {
let transaction_sender_address: address = GetTxnSenderAddress();
let coin_ref: &mut Coin = BorrowGlobal<Coin>(move(transaction_sender_address));
let coin_value_ref: &mut u64 = &mut move(coin_ref).value;
let coin_value: u64 = *move(coin_value_ref);
RejectUnless(copy(coin_value) >= copy(amount));
*move(coin_value_ref) = move(coin_value) - copy(amount);
let new_coin: Coin = Pack<Coin>(move(amount));
return move(new_coin);
}

这个过程做了这样3件事情:

  1. 获得发送者地址里面coin的唯一引用。
  2. 减去相应的数量。
  3. 创建并返回一个新的coin。

其中Pack<T>是Unpack<T>的反向操作。用来创建T资源。

更多教程请参考 flydean的博客

Libra教程之:move语言的特点和例子的更多相关文章

  1. Libra教程之:Libra protocol的逻辑数据模型

    文章目录 Libra protocol简介 逻辑数据模型 账本状态 交易 账本历史 Libra protocol简介 Libra区块链本质上是一个加密数据库,这个数据库是通过Libra protoco ...

  2. Libra教程之:来了,你最爱的Move语言

    文章目录 Move语言 Move的核心概念 Move交易脚本 Move modules Move resources 写一个Move程序 编写交易脚本 编写自己的Modules Move语言 Move ...

  3. Libra教程之:执行Transactions

    文章目录 Transactions是什么 Transactions运行的基础条件 Transactions的结构 执行Transactions Transactions是什么 我们讲到了Libra是一 ...

  4. Libra教程之:Libra协议的关键概念

    文章目录 Libra协议 交易和状态 交易详解 账本状态详解 版本数据库 账户 账户地址 Proof 验证节点 存储 Libra协议 Libra协议是Libra区块链的基础,本文主要讲解Libra协议 ...

  5. Libra教程之:运行自定义move modules

    文章目录 简介 创建Move modules 启动本地网络 创建账号并送测试币 编译Move Module 发布编译好的Module 创建交易脚本 编译编译脚本 执行脚本 简介 因为Libra和Mov ...

  6. Libra教程之:Transaction的生命周期

    文章目录 Transaction的生命周期 提交一个Transaction 交易入链的详细过程 接收Transaction 和其他Validators共享这个Transaction 区块Proposi ...

  7. Go安装配置和《菜鸟教程之Go语言教程》学习笔记

    Go 语言是一种让代码分享更容易的编程语言 菜鸟教程-Go语言教程(这个教程过于基础,体现不了Go的特性和强大.) 下载/安装Go语言 https://golang.org/dl/ Mac OS X ...

  8. Libra教程之:Libra testnet使用指南

    文章目录 Libra testnet网络 下载和安装Libra 编译Libra client并连接到Testnet网络 创建两个A和B的两个账号 检查libra cli Client是否运行 创建A的 ...

  9. Libra教程之:数据结构和存储

    文章目录 存储的数据结构 账本历史 账本状态 账户 事件 前面的文章我们知道,libra会把所有的数据都存储在账本中.为了方便业务逻辑和数据的校验,这个存储是以特定的数据结构来实现的,这里我们叫做验证 ...

随机推荐

  1. A 修公路

    时间限制 : - MS   空间限制 : 65536 KB  评测说明 : 时限1000ms 问题描述 某岛国有n个小岛构成(编号1到n),该国政府想要通过n-1条双向公路将这些小岛连接起来,使得任意 ...

  2. JVM类加载过程详细分析

    双亲委派加载模型 为什么需要双亲委派加载模型 主要是为了安全,避免用户恶意加载破坏JVM正常运行的字节码文件,比如说加载一个自己写的java.util.HashMap.class.这样就有可能造成包冲 ...

  3. php--php设计模式留存

    装饰者模式 <?php interface Decorator { public function display(); } class XiaoFang implements Decorato ...

  4. Linux基础篇,磁盘及文件使用管理

    在windows系统下,我们可以使用图形化界面很明了的看出当前硬盘使用量与某个文件的占用空间大小和文件数量.但是在linux系统中,我们应该如何得到这些信息呢? 当然是功能强大的df与du了. 一.d ...

  5. search(4)- elastic4s-ElasticDsl

    上次分析了一下elastic4s的运算框架.本来计划接着开始实质的函数调用示范,不过看过了Elastic4s的所有使用说明文档后感觉还是走的快了一点.主要原因是elasticsearch在7.0后有了 ...

  6. Django 表单处理流程

    Django 的表单处理:视图获取请求,执行所需的任何操作,包括从模型中读取数据,然后生成并返回HTML页面(从模板中),我们传递一个包含要显示的数据的上下文.使事情变得更复杂的是,服务器还需要能够处 ...

  7. 搭建WEB、NFS共享、sersync实时同步以及全网定时备份服务流程

    本次实验的主要目的: 1.搭建web服务,使用nfs服务共享的/data目录挂载到web站点目录上. 2.nfs服务器与backup服务器使用sersync实时同步/data目录中的文件. 3.bac ...

  8. Java学习成长第一集

    由于最近所在项目组的项目临近结尾,所以有时间对自己近来的学习做个总结.不得不说,程序员不学习就退步这句话是真的很让人信服!自己入行将近一年的时间,所学的就是Java开发的专业,很羞愧的是现在的自己能力 ...

  9. AJ整理问题之:copy,对象自定义copy 什么是property

    AJ分享,必须精品 copy copy的正目的 copy 目的:建立一个副本,彼此修改,各不干扰 Copy(不可变)和MutableCopy(可变)针对Foundation框架的数据类型. 对于自定义 ...

  10. Html5移动端弹幕动画实现示例代码

    已知20条内容要有弹幕效果,分成三层,速度随机. 先来看看效果: 所以这里不考虑填写生成的.只是一个展现的效果. 如果要看填写生成的,请不要浪费Time 思路 把单个内容编辑好,计算自身宽度,确定初始 ...