【微软技术栈】C#.NET 正则表达式源生成器

本文内容

  1. 已编译的正则表达式
  2. 源生成
  3. 在源生成的文件中
  4. 何时使用

正则表达式 (regex) 是一个字符串,它使开发人员能够表达要搜索的模式,使其成为搜索文本和提取结果作为已搜索字符串子集的一种很常见的方法。 在 .NET 中,System.Text.RegularExpressions 命名空间用于定义 Regex 实例和静态方法,并匹配用户定义的模式。 本文介绍如何使用源生成来生成 Regex 实例以优化性能。

 备注

请尽可能地使用源生成的正则表达式,而不是使用 RegexOptions.Compiled 选项编译正则表达式。 源生成可帮助应用更快地启动、更快地运行且更易剪裁。

1、已编译的正则表达式

编写 new Regex("somepattern") 时,会发生一些事情。 将分析指定的模式,既要确保模式的有效性,又要将其转换为表示已分析正则表达式的内部树。 然后,以各种方式优化树,将模式转换为功能上等效的变体,可以更高效地执行。 该树被写入一种形式,可以解释为一系列操作码和操作数,它们提供有关如何匹配的正则表达式解释器引擎的说明。 执行匹配时,解释器只需遍历这些指令,针对输入文本处理它们。 实例化新的 Regex 实例或调用 Regex 上的一个静态方法时,解释器是采用的默认引擎。

指定 RegexOptions.Compiled 时,将执行所有相同的构造时间工作。 生成的指令将由基于反射发出的编译器进一步转换为 IL 指令,这些指令将写入几个 DynamicMethod。 执行匹配时,将调用这些 DynamicMethod。 此 IL 实质上将完全执行解释器将执行的操作,但专用于所处理的确切模式除外。 例如,如果模式包含 [ac],解释器将看到一个操作码,表示“将当前位置的输入字符与在此集说明中指定的集匹配”,而编译的 IL 将包含有效表示“将目前位置的输入字母与 'a' 或 'c' 相匹配”的代码。 这种特殊的大小写和基于模式知识执行优化的能力是指定 RegexOptions.Compiled 比解释器产生更快的匹配吞吐量的主要原因。

RegexOptions.Compiled 有几个缺点。 最有影响力的是,它产生的构造成本比使用解释器要多得多。 不仅要为解释器支付相同的成本,而且还需要将生成的 RegexNode 树和生成的操作码/操作数编译到 IL 中,这会增加不小的开销。 生成的 IL 还需要在首次使用时进行 JIT 编译,这会导致启动时产生更多费用。 RegexOptions.Compiled 表示首次使用开销与每次后续使用开销之间的基本权衡。 System.Reflection.Emit 的使用也会抑制某些环境中 RegexOptions.Compiled 的使用;某些操作系统不允许动态生成的代码执行,并且在此类系统上,Compiled 将变为无操作。

2、源生成

.NET 7 引入了新的 RegexGenerator 源生成器。 当 C# 编译器重写为“Roslyn”C# 编译器时,它将公开整个编译管道的对象模型以及分析器。 最近,Roslyn 已启用源生成器。 就像分析器一样,源生成器是插入编译器的组件,并且可以提供与分析器相同的所有信息,但除了能够发出诊断信息外,它还可以使用其他源代码来增强编译单元。 .NET 7 SDK 包含一个新的源生成器,用于识别返回 Regex 的分部方法上的新 GeneratedRegexAttribute。 源生成器提供该方法的实现,该方法用于实现 Regex 的所有逻辑。 例如,可能已编写如下所示的代码:

private static readonly Regex s_abcOrDefGeneratedRegex =new(pattern: "abc|def",options: RegexOptions.Compiled | RegexOptions.IgnoreCase);private static void EvaluateText(string text)
{if (s_abcOrDefGeneratedRegex.IsMatch(text)){// Take action with matching text}
}

现在可以按如下所示重写前面的代码:

[GeneratedRegex("abc|def", RegexOptions.IgnoreCase, "en-US")]
private static partial Regex AbcOrDefGeneratedRegex();private static void EvaluateText(string text)
{if (AbcOrDefGeneratedRegex().IsMatch(text)){// Take action with matching text}
}

生成的 AbcOrDefGeneratedRegex() 实现同样缓存 Regex 单一实例,因此不需要额外的缓存来使用代码。

 提示

源生成器会忽略标志 RegexOptions.Compiled,因此在源生成的版本中不再需要它。

cached-instance.png

但可以看到,它不仅仅是在执行 new Regex(...)。 相反,源生成器以 C# 代码的形式发出自定义 Regex 派生实现,其逻辑类似于 IL 中 RegexOptions.Compiled 发出的内容。 可以获得 RegexOptions.Compiled 的所有吞吐量性能优势(实际上更多)和 Regex.CompileToAssembly 的启动优势,但没有 CompileToAssembly 的复杂性。 发出的源是项目的一部分,这意味着它也可以轻松查看和调试。

debuggable-source.png

 提示

在 Visual Studio 中,右键单击分部方法声明,然后选择“转到定义”。 或者,也可以在“解决方案资源管理器”中选择项目节点,然后展开“依赖项”>“分析器”>“System.Text.RegularExpressions.Generator”>“System.Text.RegularExpressions.Generator.RegexGenerator”>“RegexGenerator.gcs”,查看从此正则表达式生成器生成的 C# 代码。

可以在其中设置断点,可以逐步执行,并可以使用它作为学习工具来准确了解正则表达式引擎使用输入处理模式的方式。 生成器甚至会生成三斜杠 (XML) 注释,以帮助使表达式一目了然和了解使用的位置。

xml-comments.png

3、在源生成的文件中

使用 .NET 7 时,源生成器和 RegexCompiler 几乎完全重写,从根本上改变了生成的代码的结构。 此方法已扩展,以处理所有构造(有一个警告),RegexCompiler 和源生成器仍遵循新方法以 1:1 的方式彼此映射。 请考虑 (a|bc)d 表达式中某个主要函数的源生成器输出:

private bool TryMatchAtCurrentPosition(ReadOnlySpan<char> inputSpan)
{int pos = base.runtextpos;int matchStart = pos;int capture_starting_pos = 0;ReadOnlySpan<char> slice = inputSpan.Slice(pos);// 1st capture group.//{capture_starting_pos = pos;// Match with 2 alternative expressions.//{if (slice.IsEmpty){UncaptureUntil(0);return false; // The input didn't match.}switch (slice[0]){case 'a':pos++;slice = inputSpan.Slice(pos);break;case 'b':// Match 'c'.if ((uint)slice.Length < 2 || slice[1] != 'c'){UncaptureUntil(0);return false; // The input didn't match.}pos += 2;slice = inputSpan.Slice(pos);break;default:UncaptureUntil(0);return false; // The input didn't match.}//}base.Capture(1, capture_starting_pos, pos);//}// Match 'd'.if (slice.IsEmpty || slice[0] != 'd'){UncaptureUntil(0);return false; // The input didn't match.}// The input matched.pos++;base.runtextpos = pos;base.Capture(0, matchStart, pos);return true;// <summary>Undo captures until it reaches the specified capture position.</summary>[MethodImpl(MethodImplOptions.AggressiveInlining)]void UncaptureUntil(int capturePosition){while (base.Crawlpos() > capturePosition){base.Uncapture();}}
}

源生成的代码的目标易于理解,具有易于遵循的结构,具有解释每个步骤正在执行的操作的注释,并且通常根据指导原则发出代码,即生成器应发出代码,就像人类编写代码一样。 即使涉及回溯,回溯的结构也会成为代码结构的一部分,而不是依赖堆栈来指示下一步的跳转位置。 例如,以下是表达式为 [ab]*[bc] 时生成的相同匹配函数的代码:

private bool TryMatchAtCurrentPosition(ReadOnlySpan<char> inputSpan)
{int pos = base.runtextpos;int matchStart = pos;int charloop_starting_pos = 0, charloop_ending_pos = 0;ReadOnlySpan<char> slice = inputSpan.Slice(pos);// Match a character in the set [ab] greedily any number of times.//{charloop_starting_pos = pos;int iteration = slice.IndexOfAnyExcept('a', 'b');if (iteration < 0){iteration = slice.Length;}slice = slice.Slice(iteration);pos += iteration;charloop_ending_pos = pos;goto CharLoopEnd;CharLoopBacktrack:if (Utilities.s_hasTimeout){base.CheckTimeout();}if (charloop_starting_pos >= charloop_ending_pos ||(charloop_ending_pos = inputSpan.Slice(charloop_starting_pos, charloop_ending_pos - charloop_starting_pos).LastIndexOfAny('b', 'c')) < 0){return false; // The input didn't match.}charloop_ending_pos += charloop_starting_pos;pos = charloop_ending_pos;slice = inputSpan.Slice(pos);CharLoopEnd://}// Advance the next matching position.if (base.runtextpos < pos){base.runtextpos = pos;}// Match a character in the set [bc].if (slice.IsEmpty || !char.IsBetween(slice[0], 'b', 'c')){goto CharLoopBacktrack;}// The input matched.pos++;base.runtextpos = pos;base.Capture(0, matchStart, pos);return true;
}

可以在代码中看到回溯的结构,其中发出了一个 CharLoopBacktrack 标签(用于回溯到的位置)和一个 goto(用于在正则表达式后续部分失败时跳转到该位置)。

如果查看实现 RegexCompiler 的代码和源生成器,它们看起来非常相似:类似命名的方法、类似的调用结构,甚至整个实现中的类似注释。 在大多数情况下,它们会生成相同的代码,尽管一个在 IL 中,一个在 C# 中。 当然,C# 编译器负责将 C# 转换为 IL,因此这两种情况下生成的 IL 可能不相同。 源生成器依赖于在各种情况下,利用 C# 编译器进一步优化各种 C# 构造的事实。 因此,与 RegexCompiler 相比,源生成器将生成一些更优化的匹配代码。 例如,在前面的一个示例中,可以看到发出 switch 语句的源生成器,其中一个分支用于 'a',另一个分支用于 'b'。 由于 C# 编译器非常擅长优化 switch 语句,可以使用多种策略来有效地优化语句,因此源生成器具有一种 RegexCompiler 无法进行的特殊优化。 对于交替,源生成器将查看所有分支,如果可以证明每个分支都以不同的起始字符开头,它将在第一个字符上发出 switch 语句,并避免为该交替输出任何回溯代码。

下面是一个稍微复杂的相关示例。 在 .NET 7 中,将更深入地分析交替,以确定是否可以采用某种方式重构它们,从而使它们更容易被回溯引擎优化,这将导致更简单的源生成的代码。 其中一种优化支持从分支中提取常见前缀,如果交替是原子性的,则排序并不重要,重新排序分支以允许更多此类提取。 可以看到这对以下工作日模式 Monday|Tuesday|Wednesday|Thursday|Friday|Saturday|Sunday 的影响,该模式生成如下匹配函数:

private bool TryMatchAtCurrentPosition(ReadOnlySpan<char> inputSpan)
{int pos = base.runtextpos;int matchStart = pos;ReadOnlySpan<char> slice = inputSpan.Slice(pos);// Match with 5 alternative expressions, atomically.{if (slice.IsEmpty){return false; // The input didn't match.}switch (slice[0]){case 'M':// Match the string "onday".if (!slice.Slice(1).StartsWith("onday")){return false; // The input didn't match.}pos += 6;slice = inputSpan.Slice(pos);break;case 'T':// Match with 2 alternative expressions, atomically.{if ((uint)slice.Length < 2){return false; // The input didn't match.}switch (slice[1]){case 'u':// Match the string "esday".if (!slice.Slice(2).StartsWith("esday")){return false; // The input didn't match.}pos += 7;slice = inputSpan.Slice(pos);break;case 'h':// Match the string "ursday".if (!slice.Slice(2).StartsWith("ursday")){return false; // The input didn't match.}pos += 8;slice = inputSpan.Slice(pos);break;default:return false; // The input didn't match.}}break;case 'W':// Match the string "ednesday".if (!slice.Slice(1).StartsWith("ednesday")){return false; // The input didn't match.}pos += 9;slice = inputSpan.Slice(pos);break;case 'F':// Match the string "riday".if (!slice.Slice(1).StartsWith("riday")){return false; // The input didn't match.}pos += 6;slice = inputSpan.Slice(pos);break;case 'S':// Match with 2 alternative expressions, atomically.{if ((uint)slice.Length < 2){return false; // The input didn't match.}switch (slice[1]){case 'a':// Match the string "turday".if (!slice.Slice(2).StartsWith("turday")){return false; // The input didn't match.}pos += 8;slice = inputSpan.Slice(pos);break;case 'u':// Match the string "nday".if (!slice.Slice(2).StartsWith("nday")){return false; // The input didn't match.}pos += 6;slice = inputSpan.Slice(pos);break;default:return false; // The input didn't match.}}break;default:return false; // The input didn't match.}}// The input matched.base.runtextpos = pos;base.Capture(0, matchStart, pos);return true;
}

请注意 Thursday 是如何被重新排序为紧跟在 Tuesday 之后的,以及 Tuesday/Thursday 对和 Saturday/Sunday 对如何以多级开关结束。 在极端情况下,如果你要创建许多不同单词的长交替,源生成器最终会发出 trie^1 的逻辑等效项,从而读取每个字符并 switch(切换)到用于处理单词的其余部分的分支。 这是匹配字词的一种非常高效的方法,也是源生成器在此处执行的操作。

同时,源生成器还有其他问题需要解决,直接输出到 IL 时根本不存在这些问题。 如果你回过头来看看几个代码示例,可以看到一些大括号有些奇怪地注释掉了。这不是一个错误。 源生成器认识到,如果未注释掉这些大括号,则回溯的结构依赖于从范围外跳到该范围内定义的标签;此类标签对于此类 goto 不可见,代码将无法编译。 因此,源生成器需要避免在方式上存在范围。 在某些情况下,只需按照此处所述注释掉范围。 在其他不可能的情况下,如果这样做会有问题,有时可能会避免需要范围的构造(例如多语句 if 块)。

源生成器处理所有 RegexCompiler 句柄,但有一个例外。 与处理 RegexOptions.IgnoreCase 一样,实现现在使用大小写表在构造时生成集,以及 IgnoreCase 向后引用匹配需要查阅该大小写表的方式。 该表是 System.Text.RegularExpressions.dll 的内部表,至少目前,该程序集的外部代码(包括源生成器发出的代码)没有访问该表的权限。 这使得处理 IgnoreCase 向后引用在源生成器中成为一个挑战,并且它们不受支持。 这是 RegexCompiler 支持的源生成器不支持的一个构造。 如果尝试使用具有其中之一(这种情况很少见)的模式,源生成器不会发出自定义实现,而是回退到缓存常规 Regex 实例:

cached-instance-fallback.png

此外,RegexCompiler 和源生成器都不支持新的 RegexOptions.NonBacktracking。 如果指定 RegexOptions.Compiled | RegexOptions.NonBacktracking,则只会忽略 Compiled 标志,如果将 NonBacktracking 指定给源生成器,它将同样回退到缓存常规 Regex 实例。

4、何时使用

一般指导是,如果可以使用源生成器,请使用它。 如果当前在 C# 中使用 Regex,并且在编译时有已知的参数,特别是如果你已经在使用 RegexOptions.Compiled(因为正则表达式已被确定为一个热点,可以从更快的吞吐量中受益),则应更倾向于使用源生成器。 源生成器将为正则表达式提供以下优势:

  • RegexOptions.Compiled 的所有吞吐量优势。
  • 在运行时不必执行所有正则表达式解析、分析和编译的启动优势。
  • 将提前编译与为正则表达式生成的代码一起使用的选项。
  • 更好的可调试性和对正则表达式的理解。
  • 通过剪裁与 RegexCompiler 关联的大量代码(甚至可能是反射发出自身),可以减小剪裁后的应用程序的大小。

当与源生成器无法为其生成自定义实现的 RegexOptions.NonBacktracking 等选项一起使用时,它仍将发出描述实现的缓存和 XML 注释,使其很有价值。 源生成器的主要缺点是它会向程序集发出其他代码,因此可能会增加大小。 应用程序中的正则表达式越多,它们就越大,将为它们发出的代码就越多。 在某些情况下,同样 RegexOptions.Compiled 可能不是必要的,源生成器也可能不是必要的。 例如,如果你有一个仅需要很少且吞吐量无关紧要的正则表达式,那么仅依赖解释器进行零星使用可能更有益。

 重要

.NET 7 包含一个分析器,用于标识可转换为源生成器的 Regex 的使用,还包含一个为你执行转换的修复程序:

convert-to-regexgenerator.png

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/136907.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

支持C#的开源免费、新手友好的数据结构与算法入门教程 - Hello算法

前言 前段时间完成了C#经典十大排序算法&#xff08;完结&#xff09;然后有很多小伙伴问想要系统化的学习数据结构和算法&#xff0c;不知道该怎么入门&#xff0c;有无好的教程推荐的。今天给大家推荐一个支持C#的开源免费、新手友好的数据结构与算法入门教程&#xff1a;He…

C# wpf 实现任意控件(包括窗口)更多拖动功能

系列文章目录 第一章 Grid内控件拖动 第二章 Canvas内控件拖动 第三章 任意控件拖动 第四章 窗口拖动 第五章 附加属性实现任意拖动 第六章 拓展更多拖动功能&#xff08;本章&#xff09; 文章目录 系列文章目录前言一、添加的功能1、任意控件MoveTo2、任意控件DragMove3、边…

【STM32】STM32的Cube和HAL生态

1.单片机软件开发的时代变化 1.单片机的演进过程 (1)第1代&#xff1a;4004、8008、Zilog那个年代&#xff08;大约1980年代之前&#xff09; (2)第2代&#xff1a;51、PIC8/16、AVR那个年代&#xff08;大约2005年前&#xff09; (3)第3代&#xff1a;51、PIC32、Cortex-M0、…

解决IDEA使用卡顿的问题

*问题&#xff1a;使用IDEA的时候卡顿 原因&#xff1a;IDEA默认分配的内存有上限 **可以查看内存分配情况及使用情况__ 解决&#xff1a; 设置JVM的启动参数&#xff1a; 进入idea的安装目录的bin文件夹 -Xms1024m -Xmx2048m -XX:ReservedCodeCacheSize1024m -XX:UseG1G…

IP-guard WebServer RCE漏洞复现

0x01 产品简介 IP-guard是由溢信科技股份有限公司开发的一款终端安全管理软件&#xff0c;旨在帮助企业保护终端设备安全、数据安全、管理网络使用和简化IT系统管理。 0x02 漏洞概述 漏洞成因 在Web应用程序的实现中&#xff0c;参数的处理和验证是确保应用安全的关键环节…

springboot中定时任务cron不生效,fixedRate指定间隔失效,只执行一次的问题

在调试计算任务的时候&#xff0c;手动重置任务为初始状态&#xff0c;但是并没有重新开始计算&#xff0c;检查定时任务代码&#xff1a; 从Scheduled(fixedRate 120000)可以看到&#xff0c;应该是间隔120秒执行一次该定时任务&#xff0c;查看后台日志&#xff0c;并没有重…

Uniapp实现多语言切换

前言 之前做项目过程中&#xff0c;也做过一次多语言切换&#xff0c;大致思想都是一样的&#xff0c;想了解的可以看下之前的文章C#WinForm实现多语言切换 使用i18n插件 安装插件 npm install vue-i18n --saveMain.js配置 // 引入 多语言包 import VueI18n from vue-i18n…

OpenGL_Learn08(坐标系统与3D空间)

目录 1. 概述 2. 局部空间 3. 世界空间 4. 观察空间 5. 剪裁空间 6. 初入3D 7. 3D旋转 8. 多个正方体 9. 观察视角 1. 概述 OpenGL希望在每次顶点着色器运行后&#xff0c;我们可见的所有顶点都为标准化设备坐标(Normalized Device Coordinate, NDC)。也就是说&#x…

关于 国产系统UOS系统Qt开发Tcp服务器外部连接无法连接上USO系统 的解决方法

若该文为原创文章&#xff0c;转载请注明原文出处 本文章博客地址&#xff1a;https://hpzwl.blog.csdn.net/article/details/134254817 红胖子(红模仿)的博文大全&#xff1a;开发技术集合&#xff08;包含Qt实用技术、树莓派、三维、OpenCV、OpenGL、ffmpeg、OSG、单片机、软…

Termius for Mac:掌控您的云端世界,安全高效的SSH客户端

你是否曾经在Mac上苦苦寻找一个好用的SSH客户端&#xff0c;让你能够远程连接到Linux服务器&#xff0c;轻松管理你的云端世界&#xff1f;现在&#xff0c;我们向你介绍一款强大而高效的SSH客户端——Termius。 Termius是一款专为Mac用户设计的SSH客户端&#xff0c;它提供了…

this.$message提示内容添加换行

0 效果 1 代码 let msgArr [只允许上传doc/docx/xls/xlsx/pdf/png/jpg/bmp/ppt/pptx/rar/zip格式文件,且单个文件大小不能超过20MB,已过滤无效的文件] let msg msgArr.join(<br/>) this.$message({dangerouslyUseHTMLString: true,message: msg,type: warning })

安卓 车轮视图 WheelView kotlin

安卓 车轮视图 WheelView kotlin 前言一、代码解析1.初始化2.初始化数据3.onMeasure4.onDraw5.onTouchEvent6.其他 6.ItemObject二、完整代码总结 前言 有个需求涉及到类似这个视图&#xff0c;于是在网上找了个轮子&#xff0c;自己改吧改吧用&#xff0c;拿来主义当然后&…

物联网中的毫米波雷达:连接未来的智能设备

随着物联网&#xff08;IoT&#xff09;技术的飞速发展&#xff0c;连接设备的方式和效能变得越来越重要。毫米波雷达技术作为一种先进的感知技术&#xff0c;正在为物联网设备的连接和智能化提供全新的可能性。本文将深入探讨毫米波雷达在物联网中的应用&#xff0c;以及它是如…

高防CDN与高防服务器:为什么高防服务器不能完全代替高防CDN

在当今的数字化时代&#xff0c;网络安全已经成为企业不容忽视的关键问题。面对不断增长的网络威胁和攻击&#xff0c;许多企业采取了高防措施以保护其网络和在线资产。然而&#xff0c;高防服务器和高防CDN是两种不同的安全解决方案&#xff0c;各自有其优势和局限性。在本文中…

Apache APISIX Dashboard 未经认证访问导致 RCE(CVE-2021-45232)漏洞复现

漏洞描述 Apache APISIX 是一个动态、实时、高性能的 API 网关&#xff0c;而 Apache APISIX Dashboard 是一个简单易用的前端界面&#xff0c;用于管理 Apache APISIX。 在 2.10.1 之前的 Apache APISIX Dashboard 中&#xff0c;Manager API 使用了两个框架&#xff0c;并在…

Mysql数据库 12.SQL语言 触发器

一、触发器&#xff08;操作日志表&#xff09; 1.介绍 不需要主动调用的一种储存过程&#xff0c;是一个能够完成特定过程&#xff0c;存储在数据库服务器上的SQL片段。 对当前表中数据增删改查的一种记录<日志表>&#xff0c;根据触发器自动执行&#xff0c;记录当前…

图解三傻排序 选择排序、冒泡排序、插入排序

&#xff08;1&#xff09;选择排序 // 交换 void swap(int arr[], int i, int j) {int tmp arr[i];arr[i] arr[j];arr[j] tmp; }// 选择排序 void selectionSort(int arr[],int len) {if (len < 2) return;for (int minIndex, i 0; i < len - 1; i) {minIndex i;f…

【11】使用透视投影建立一个3D空间的测试

核心操作&#xff1a; 1.proj view model 这三个矩阵 glm::mat4 mvp m_Proj * m_View * model; m_Shader->Bind(); m_Shader->SetUniformMat4f("u_MVP", mvp);着色器里面就&#xff1a; proj:投影矩阵&#xff0c;可以选择正交投影&#xff0c;或者透视投影…

实操创建属于自己的亚马逊云科技VPS服务:Amazon Lightsail

本文主要讲述如何独立创建自己的亚马逊云科技VPS服务&#xff0c;希望此文能帮助你对亚马逊云科技VPS服务也就是Amazon Lightsail&#xff0c;有个新的认识&#xff0c;对所使用的VPS有所帮助。 Amazon Lightsail是一款无论云计算的新手还是专家&#xff0c;都可通过其快速启动…

将 Ordinals 与比特币智能合约集成:第 4 部分

控制 BSV-20 代币的分配 在上一篇文章中&#xff0c;我们展示了智能合约可以在铸造后控制 BSV-20 代币的转移。 今天&#xff0c;我们演示如何控制此类代币的分发/发行。 无Tick模式 BSV-20 在 V2 中引入了无Tick模式&#xff0c;并采用了与 V1 不同的方法。 部署 (Deploy) …