.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 ...
随机推荐
- c# Lambda扩展
扩展类 public static class LinqExtensions { /// <summary> /// 创建lambda表达式:p=>true /// </sum ...
- Java实现Qt的SIGNAL-SLOT机制(保存到Map中,从而将它们关联起来,收到信号进行解析,最后反射调用)
SIGNAL-SLOT是Qt的一大特色,使用起来十分方便.在传统的AWT和Swing编程中,我们都是为要在 监听的对象上添加Listener监听器.被监听对象中保存有Listener的列表,当相关事件 ...
- 细谈unity资源加载和卸载
转载请标明出处:http://www.cnblogs.com/zblade/ 一.概要 在了解unity的资源管理方式之后,接下来细谈一下Unity的资源是如何从磁盘中加载到运行时的内存中,以及又是如 ...
- abp(net core)+easyui+efcore仓储系统——创建应用服务(五)
abp(net core)+easyui+efcore仓储系统目录 abp(net core)+easyui+efcore仓储系统——ABP总体介绍(一) abp(net core)+easyui+e ...
- CSS元素的基本应用(附加京东面试题)
ONE! 列表~ 列表分为有序列表和无序列表还有定义列表(ul和ol,dl) ul 无序列表 ul它天生自带内边距 还有一个 p 标签也是天生就自带内边距的(内边距 padding) list-st ...
- 44 | 测试先行:测试驱动开发(TDD)
- Django中间件加载原理
假设我们有如下中间件: setting.py文件 MIDDLEWARE = [ 'django.middleware.A', 'django.middleware.B', 'django.middle ...
- mac-VBox-Centos6.6安装增强功能
1. 更新内核 yum update kernel 2.安装依赖包 yum install kernel-headers yum install kernel-devel yum install gc ...
- java-NIO-FileChannel(文件IO)
Java NIO中的FileChannel是一个连接到文件的通道.可以通过文件通道读写文件. FileChannel无法设置为非阻塞模式,它总是运行在阻塞模式下. 对于文件的复制,平时我们都是使用输入 ...
- Linux中修改远程地址
首先跳转到本地用户root,如果不是的话可能没有权限 第一步:安装ssh服务 执行命令:yum install openssh-server (因为我已经安装过了,所以显示的是已安装) 第二步:修改S ...