Solana 代币合约入口程序学习

2024-02-03 10:44

本文主要是介绍Solana 代币合约入口程序学习,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本文是学习Solana 程序库代币合约系列,需要有一定的Rust基础

我们今天学习spl/token/program/src/lib.rsentrypoint.rs文件,也就是Solana 统一代币合约的入口文件。

我们首先学习lib.rs文件,其代码只有93行,也比较简单,我们来快速学习一下。

一 内部属性

内部属性应用于定义它的元素整体,因为它定义在作用的元素内部,所以在内部属性。相应的,定义在元素之外的叫外部属性。关于属性,这里有一篇文章,看完就基本明白了。

【Rust每周一知】 Attribute 属性

我们的lib.rs的前三行代码正好是定义了三个内部属性:

#![allow(clippy::arithmetic_side_effects)]
#![deny(missing_docs)]
#![cfg_attr(not(test), forbid(unsafe_code))]

第一行是允许做什么(允许工具属性),第二行是拒绝什么,第三行是配置属性。具体含义大家可以参考上面那篇文章,我也并没有仔细研究。

二 定义的module

接下来定义了5个公共的module和一个内部的module。

pub mod error;
pub mod instruction;
pub mod native_mint;
pub mod processor;
pub mod state;#[cfg(not(feature = "no-entrypoint"))]
mod entrypoint;

我们知道,公共模块是暴露给外部的,内部模块是隐藏的。但是这里有一个问题,其实整个程序的入口是entrypoint模块中的process_instruction函数,并且该函数也为内部的,那么该入口函数是怎样被调用的?需要后面进一步查看entrypoint!宏的用法。

注意:

entrypoint模块(入口模块)定义上方有一个配置属性,是非no-entrypoint特性。为什么这么配置呢?Solana官方文档上讲的很清楚,一个程序是可以作为库被引入到另一个程序中的,如果这两个程序都会有入口,就会起冲突。因此,我们定义了一个叫no-entrypoint的features,只有在非该特性的条件下才会定义entrypoint模块。当我们的程序作为第三方库引入到其它程序中时,其它程序只要在依赖定义里指定features = [ "no-entrypoint" ]就行了。这样两个程序加起来也会只有一个入口,就不会起冲突了。

三 导入其它库

接下来18-19行是导入了solana_program库及其特定的结构体,注意Rust中的一般原则,结构,枚举等定义直接导入全部路径,而函数一般只导入相应的包,使用时采用包名::函数名的语法,这样就为了方便的区分该函数是外部包定义的还是本包定义的。

pub use solana_program;
use solana_program::{entrypoint::ProgramResult, program_error::ProgramError, pubkey::Pubkey};

注意这里pub use是导入的同时并重新导出整个solana_program包,这样做的原因是方便其它包引入我们的程序时,如果想访问solana_program包中元素,直接使用spl-token::solana_program就行,而不用重新在依赖库里定义solana_program.

四 数值转换

接下来五个函数是用来进行数值转换的,这里是模仿ERC20的概念,代币是有精度的,假如美元,1美元等于100美分,把它当作一个代币的话,它的精度就是2(100 = 10.pow(2))。它的基本单位就是美元,最小单位是美分。然而区块链上一般是整数操作,因此操作的单位是美分(这样就不会有小数了),但我们平常使用习惯是美元。因此1.5美元等于多少美分,或者234美分等于多少美元?接下来这五个函数就是作这种转换的,其实就是乘上/除于相应的精度(10.pow(精度))。

这五个函数分别为(这里的ui_amount就是人们的习惯单位,例如美元,amount就是最小的不可分隔的单位,例如美分):

注: 这里的乘上/除于精度其实是指乘上/除于 10.pow(精度)。

  • ui_amount_to_amount 从美元到美分,乘上精度即可。
  • amount_to_ui_amount 从美分到美元,除以精度即可。因为我们可能会得到小数,所以最终结果是f64类型。
  • amount_to_ui_amount_string 从美分到美元的字符串形式,这里有些奇怪,为什么不采用amount_to_ui_amount结果的字符串形式呢?这里经过实际测试,amount_to_ui_amount_string 这种形式会在最后补上多余的0(小数位不够精度时),这样可以看出精度是多少。例如11.000000,后者可以看出精度是多少。
  • amount_to_ui_amount_string_trimmed 这里trimed应该是截断了后面的0,所以应该和amount_to_ui_amount结果的字符串形式相同,简单测试了几下也是如此,但为什么中间实现要采用amount_to_ui_amount_string呢,不得而知。
  • try_ui_amount_into_amount 函数,是将ui_amount的字符串形式转化为amount,因为字符串转数字有可能失败(例如字符串不合法),因此返回结果为一个Result,中间的实现过程有些复杂,我们就不管了。因为它可能失败返回Result,所以函数以try开头表明该含义。

注意,这里几个函数未考虑到溢出的情况,例如超过了u64的最大值(精度为18时很容易),但Token合约创建的代币精度为9,Supply也是u64,所以正常情况下是不会溢出的,毕竟数量不能超过supply.

declare_id!

该宏定义了本程序的账号地址,这个地址在编译合约时就可以得到,然后进行替换就行。这里并未研究程序地址的计算方法,只是知道就是这么做的。

自己写合约时,随便复制一个id写上,编译完成后再换上正确的ID,然后一定要重新编译(切记),否则部署的程序地址和ID对不上。

六 check_program_account 函数

这个函数用于检测程序ID(程序地址)是否为本合约的ID,用于作为包引入到其它程序时进行相关判断。注意,它返回的不是bool,而是一个ProgramResult。

这里的 id() 函数应该是上面的declare_id宏产生的,它估计返回一个静态的Pubkey,注意这里比较的中两个引用,在Rust中,比较引用其实是比较指向的值(当然引用类型得相同),比较引用地址是否相同有专门的函数。

显然,Pubkey 实现了 PartialEq特型,否则无法比较是否不相等。实质上,Pubkey的定义是这样子的:

#[wasm_bindgen]
#[repr(transparent)]
#[derive(AbiExample,BorshDeserialize,BorshSchema,BorshSerialize,Clone,Copy,Default,Deserialize,Eq,Hash,Ord,PartialEq,PartialOrd,Pod,Serialize,Zeroable,
)]
pub struct Pubkey(pub(crate) [u8; 32]);

我们可以看到,它的内部(底层)其实是一个32元素的u8数组(所以其大小为256位),除了 PartialEq特型,它还实现了很多常用特型,例如CloneCopy等。因为它的底层是u8数组,所以直接使用derive派生宏来实现部分特型就行。

七 entrypoint模块

entrypoint模块 代码很少,除了必要的引入外,就只有一个宏调用和程序入口函数定义。我们来看这个函数process_instruction

顾名思义,用来处理指令 ,它的参数列表是固定的。

fn process_instruction(program_id: &Pubkey,accounts: &[AccountInfo],instruction_data: &[u8],
) -> ProgramResult {if let Err(error) = Processor::process(program_id, accounts, instruction_data) {// catch the error so we can print iterror.print::<TokenError>();return Err(error);}Ok(())
}

第一个参数,program_id,很好理解,就是调用的程序的id(地址),其实这里并没有什么实际作用,如果不知道program_id,是无法调用程序的,所以这里的program_id必定是本程序的id。系统自动传过来意义不大,也许有其它用处。

第二个参数,accounts 这个参数是指令调用时所有涉及到的账号信息,这个账号是在客户端输入的,因此用户可以输入任意账号信息,所以必须对其合法性和有效性作验证,特别是只读账号,因为如果是写账号,会有写操作权限判定,会好一些。

第三个参数 指令数据,其实就是一串16进制数据,它也是用户输入的,需要对其进行有效性判断和解码,从而进行下一步操作。

函数内部直接调用了Processor结构体的处理函数进行处理,这里以后再学。

entrypoint!

该宏的定义是这样的

#[macro_export]
macro_rules! entrypoint {($process_instruction:ident) => {/// # Safety#[no_mangle]pub unsafe extern "C" fn entrypoint(input: *mut u8) -> u64 {let (program_id, accounts, instruction_data) =unsafe { $crate::entrypoint::deserialize(input) };match $process_instruction(&program_id, &accounts, &instruction_data) {Ok(()) => $crate::entrypoint::SUCCESS,Err(error) => error.into(),}}$crate::custom_heap_default!();$crate::custom_panic_default!();};
}

这里我们可以看到,它其实定义了一个 pub 函数 entrypoint,用来解析用户输入并将它作为参数传递给我们的process_instruction函数。但是为什么可以调用entrypoint包(非外部包)的这个entrypoint函数,还需要仔细看相关文档,

这里 extern 关键字是用创建允许其它语言调用Rust的接口

还有一点是#[macro_export]宏导出,

默认情况下,宏没有基于路径的作用域。但是如果该宏带有 #[macro_export] 属性,则相当于它在当前 crate 的根作用域的顶部被声明。标有 #[macro_export] 的宏始终是 pub 的.

属性no_mangle,用来关闭 Rust 的名称修改(name mangling)功能。Mangling 是编译器在解析名称时,修改我们定义的函数名称,增加一些用于其编译过程的额外信息。

所以为了使 Rust 函数能在其它语言中被调用,必须禁用 Rust 编译器的名称修改功能。通过在1.1的示例代码中增加属性 #[no_mangle] ,告诉 Rust 编译器不要修改此函数的名称。

这个宏定义上也写了,这是全局的,因此只能only once,所以引入其它定义了entrypoint!宏的包时,需要启用 no-entrypoint特性,当然这个特性名称其实是可以自取的。

我们知道这里是大致怎么一回事就行了,有时间时再详细研究。

因为只是个人的学习记录,因此肯定有理解错误的地方,欢迎大家指正,共同学习提高!

这篇关于Solana 代币合约入口程序学习的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/673903

相关文章

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

Ilya-AI分享的他在OpenAI学习到的15个提示工程技巧

Ilya(不是本人,claude AI)在社交媒体上分享了他在OpenAI学习到的15个Prompt撰写技巧。 以下是详细的内容: 提示精确化:在编写提示时,力求表达清晰准确。清楚地阐述任务需求和概念定义至关重要。例:不用"分析文本",而用"判断这段话的情感倾向:积极、消极还是中性"。 快速迭代:善于快速连续调整提示。熟练的提示工程师能够灵活地进行多轮优化。例:从"总结文章"到"用

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

学习hash总结

2014/1/29/   最近刚开始学hash,名字很陌生,但是hash的思想却很熟悉,以前早就做过此类的题,但是不知道这就是hash思想而已,说白了hash就是一个映射,往往灵活利用数组的下标来实现算法,hash的作用:1、判重;2、统计次数;

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

【机器学习】高斯过程的基本概念和应用领域以及在python中的实例

引言 高斯过程(Gaussian Process,简称GP)是一种概率模型,用于描述一组随机变量的联合概率分布,其中任何一个有限维度的子集都具有高斯分布 文章目录 引言一、高斯过程1.1 基本定义1.1.1 随机过程1.1.2 高斯分布 1.2 高斯过程的特性1.2.1 联合高斯性1.2.2 均值函数1.2.3 协方差函数(或核函数) 1.3 核函数1.4 高斯过程回归(Gauss

【学习笔记】 陈强-机器学习-Python-Ch15 人工神经网络(1)sklearn

系列文章目录 监督学习:参数方法 【学习笔记】 陈强-机器学习-Python-Ch4 线性回归 【学习笔记】 陈强-机器学习-Python-Ch5 逻辑回归 【课后题练习】 陈强-机器学习-Python-Ch5 逻辑回归(SAheart.csv) 【学习笔记】 陈强-机器学习-Python-Ch6 多项逻辑回归 【学习笔记 及 课后题练习】 陈强-机器学习-Python-Ch7 判别分析 【学

系统架构师考试学习笔记第三篇——架构设计高级知识(20)通信系统架构设计理论与实践

本章知识考点:         第20课时主要学习通信系统架构设计的理论和工作中的实践。根据新版考试大纲,本课时知识点会涉及案例分析题(25分),而在历年考试中,案例题对该部分内容的考查并不多,虽在综合知识选择题目中经常考查,但分值也不高。本课时内容侧重于对知识点的记忆和理解,按照以往的出题规律,通信系统架构设计基础知识点多来源于教材内的基础网络设备、网络架构和教材外最新时事热点技术。本课时知识

线性代数|机器学习-P36在图中找聚类

文章目录 1. 常见图结构2. 谱聚类 感觉后面几节课的内容跨越太大,需要补充太多的知识点,教授讲得内容跨越较大,一般一节课的内容是书本上的一章节内容,所以看视频比较吃力,需要先预习课本内容后才能够很好的理解教授讲解的知识点。 1. 常见图结构 假设我们有如下图结构: Adjacency Matrix:行和列表示的是节点的位置,A[i,j]表示的第 i 个节点和第 j 个