.NET 欢乐编程术之类型超级转换之术👍👍
准备工作:先确保 VS 版本大于 2017,且支持C# 7.0 语言版本。然后新建 .Net Core 项目,在 Nuget 包管理上引入微软霸霸官方包 System.Runtime.CompilerServices.Unsafe。此包提供了非常底层又符合 .Net CLR 的 API,包括操作指针,引用,内存的方法。
接下来我们就可以利用这个包,去获取一个字符串的内存信息,然后更改这个字符串的内容。众所周知,.Net 中的字符串是不可变的,C# 和 .Net 都极大的限制程序员不可修改字符串的内容,因为一旦修改了字符串的内容,将破环 CLR 的规则,使得程序变得不稳定。
首先我们定义一个与 String 类型字段结构完全一样的类型:
public sealed class MyString
{
/// <summary>
/// 字符串的长度。
/// </summary>
public int _stringLength; /// <summary>
/// 字符串第一个字符,它与后续的字符的内存是连续的。
/// </summary>
public char _firstChar;
}
然后我们定义一个字符串:
var str = "Dogwei 牛B!";
然后我们将这个字符串超级转换为 MyString 类型:
var myStr = System.Runtime.CompilerServices.Unsafe.As<MyString>(str);
现在我们可以修改字符串的内容了:
var str = "Dogwei 牛B!";
var myStr = System.Runtime.CompilerServices.Unsafe.As<MyString>(str);
Unsafe.Add(ref myStr._firstChar, str.IndexOf('牛')) = 'S';
Console.WriteLine(str); // Output : Dogwei SB!
怎么样,是不是很有意思?我们再来试试修改字符串长度:
myStr._stringLength = ; Console.WriteLine(str); // Output : Dogwei myStr._stringLength = ; Console.WriteLine(str); // Dogwei SB! ??翽 鄈淭翽...
长度超过字符串本来的长度会输出一串乱码。
同样转换之后的方法也是可以执行的:
public class Demo
{
public static void Main()
{
var str = "Dogwei 牛B!"; var myStr = Unsafe.As<MyString>(str); myStr.SayHello();
}
}
public sealed class MyString
{
/// <summary>
/// 字符串的长度。
/// </summary>
public int _stringLength; /// <summary>
/// 字符串第一个字符,它与后续的字符的内存是连续的。
/// </summary>
public char _firstChar; public void SayHello()
{
var str = Unsafe.As<string>(this); var splits = str.Split(' '); var name = splits[]; var say = splits[]; Console.WriteLine($"Hello! my name is {name}, I am {say}.");
}
}
但是执行方法有一个必须要注意的地方,就是执行的方法必须是最终方法!(何为最终方法请查阅微软官方文档 System.Reflection.MethodInfo.IsFinal)。如果不是最终方法会怎么样呢?我们来试试:
同上例,Main 方法保持不变,修改 MyString 为如下:
public class MyString
{
/// <summary>
/// 字符串的长度。
/// </summary>
public int _stringLength; /// <summary>
/// 字符串第一个字符,它与后续的字符的内存是连续的。
/// </summary>
public char _firstChar; public virtual void SayHello()
{
var str = Unsafe.As<string>(this); var splits = str.Split(' '); var name = splits[]; var say = splits[]; Console.WriteLine($"Hello! my name is {name}, I am {say}.");
}
}
执行程序后什么也没发生,既没执行,也没报错:

到这里相信大家也对类型强转超级之术有一点理解,但是这个“巫术”有一些限制:
1:不能转换为值类型!
2:转换之后必须显式定义类型,否则将无意义。
下一章我们将讲超级转换之术二代!可以转换任何对象,且是实际意义转换。
.NET 欢乐编程术之类型超级转换之术👍👍的更多相关文章
- C#高级编程笔记 2016年10月8日运算符和类型强制转换
1.checked和unchecked 运算符 C#提供了checked 和uncheckde 运算符.如果把一个代码块标记为checked, CLR就会执行溢出检查,如果发生溢出,就抛出overfl ...
- C#高级编程9-第7章 运算符和类型强制转换
运算符和类型强制转换 1.运算符 运算符的简化操作 条件运算符: if-else的简化操作,也称三元运算符.如果条件为真,返回一个值,为假返回另外一个值. condition?true_value:f ...
- C#高级编程 (第六版) 学习 第六章:运算符和类型强制转换
第六章 运算符和类型强制转换 1,运算符 类别 运算符 算术运算符 + - * / % 逻辑运算符 & | ^ ~ && || ! 字符串连接运算符 + 增量和减量运算符 ++ ...
- 【读书笔记】C#高级编程 第七章 运算符和类型强制转换
(一)运算符 类别 运算符 算术运算符 + - * / % 逻辑运算符 & | ^ ~ && || ! 字符串连接运算符 + 增量和减量运算符 ++ -- 移位运算符 < ...
- 【转】java中byte数组与int类型的转换(两种方式)----不错
原文网址:http://blog.csdn.net/piaojun_pj/article/details/5903009 java中byte数组与int类型的转换,在网络编程中这个算法是最基本的算法, ...
- C语言指针类型 强制转换
关于C语言指针类型 强制转换 引用一篇文章: C语言中,任何一个变量都必须占有一个地址,而这个地址空间内的0-1代码就是这个变量的值.不同的数据类型占有的空间大小不一,但是他们都必须有个地址,而这个 ...
- C++四种类型的转换
在C/C++使用的语言 (type) value(您还可以使用type(value))对于显式类型转换,经常提到投.转换程序猿的精度等完全掌握手,一个传统投往往是过度使用.成为C++要根源. 为了降低 ...
- 《Python核心编程》数字类型
1.数字类型简单介绍 Python中数字类型包含:整型.长整型.布尔型.双精度浮点型.十进制浮点型.复数.这些数字类型都是不可变类型.也就是说,改变了数字的值会生成新的对象. 在Python中删除数字 ...
- FILETIME类型到LARGE_INTEGER类型的转换
核心编程第5版 245页到247页的讲到SetWaitableTimer函数的使用 其中提到 FILETIME类型到LARGE_INTEGER类型的转换问题,如下代码 //我们声明的局部变量 HAND ...
随机推荐
- Cookie概述与应用
一.概述 Cookie是Web服务器保存在客户端的一系列文本信息 典型应用一:判断注册用户是否已经登录网站. 典型应用二:"购物车"的处理. Cookie的作用: 对特定对 ...
- C#中比较两个对象的地址是否相同(也是引用计数的问题,和Java一样)
private void button1_Click(object sender, EventArgs e) { char[] ch = { 'z', 's', 'w', 'a', 'n', ...
- MSB3268 .Net 4.0工程 引用BCL错误
Severity Code Description Project File Line Suppression StateWarning MSB3268 The primary reference & ...
- SYN2102型 NTP网络时间服务器
SYN2102型 NTP网络时间服务器 ntp主时钟服务器ntp时钟服务器厂商使用说明视频链接: http://www.syn029.com/h-pd-57-0_310_1_-1.html 请将 ...
- js数字转成金额格式
本文有以下三个段落 1.方法展示 2.方法说明 3.方法实例 1.方法展示 //将数字转换成金额显示 function toMoney(num){ num = num.toFixed(2); num ...
- 修改系统的shell
一:修改系统的shell (选用zsh解释器,相对于bash,对它有很好的兼容性,而且功能上更加强大) 1.查看系统中安装的shell cat /etc/shells ...
- NSCache缓存怎么来的
什么是NSCache NSCache主要用来存储临时数据(键值对),当内存资源不够时,系统会自动释放部分数据.它有三个特点: • NSCache为了保持不占用过多的系统内存,它有多种自动回收内存策略: ...
- 自定义vue全局组件use使用、vuex的使用
自定义vue全局组件use使用(解释vue.use()的原理)我们在前面学习到是用别人的组件:Vue.use(VueRouter).Vue.use(Mint)等等.其实使用的这些都是全剧组件,这里我们 ...
- Hadoop 学习之路(二)—— 集群资源管理器 YARN
一.hadoop yarn 简介 Apache YARN (Yet Another Resource Negotiator) 是hadoop 2.0 引入的集群资源管理系统.用户可以将各种服务框架部署 ...
- Ubuntu 配置docker镜像加速器
1. 安装/升级Docker客户端 推荐安装1.10.0以上版本的Docker客户端,参考文档 docker-ce 2. 配置镜像加速器 针对Docker客户端版本大于 1.10.0 的用户 您可以通 ...