准备工作:先确保 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 欢乐编程术之类型超级转换之术👍👍的更多相关文章

  1. C#高级编程笔记 2016年10月8日运算符和类型强制转换

    1.checked和unchecked 运算符 C#提供了checked 和uncheckde 运算符.如果把一个代码块标记为checked, CLR就会执行溢出检查,如果发生溢出,就抛出overfl ...

  2. C#高级编程9-第7章 运算符和类型强制转换

    运算符和类型强制转换 1.运算符 运算符的简化操作 条件运算符: if-else的简化操作,也称三元运算符.如果条件为真,返回一个值,为假返回另外一个值. condition?true_value:f ...

  3. C#高级编程 (第六版) 学习 第六章:运算符和类型强制转换

    第六章 运算符和类型强制转换 1,运算符 类别 运算符 算术运算符 + - * / % 逻辑运算符 & | ^ ~ && || ! 字符串连接运算符 + 增量和减量运算符 ++ ...

  4. 【读书笔记】C#高级编程 第七章 运算符和类型强制转换

    (一)运算符 类别 运算符 算术运算符 + - * / % 逻辑运算符 & | ^ ~ && || ! 字符串连接运算符 + 增量和减量运算符 ++ -- 移位运算符 < ...

  5. 【转】java中byte数组与int类型的转换(两种方式)----不错

    原文网址:http://blog.csdn.net/piaojun_pj/article/details/5903009 java中byte数组与int类型的转换,在网络编程中这个算法是最基本的算法, ...

  6. C语言指针类型 强制转换

    关于C语言指针类型 强制转换  引用一篇文章: C语言中,任何一个变量都必须占有一个地址,而这个地址空间内的0-1代码就是这个变量的值.不同的数据类型占有的空间大小不一,但是他们都必须有个地址,而这个 ...

  7. C++四种类型的转换

    在C/C++使用的语言 (type) value(您还可以使用type(value))对于显式类型转换,经常提到投.转换程序猿的精度等完全掌握手,一个传统投往往是过度使用.成为C++要根源. 为了降低 ...

  8. 《Python核心编程》数字类型

    1.数字类型简单介绍 Python中数字类型包含:整型.长整型.布尔型.双精度浮点型.十进制浮点型.复数.这些数字类型都是不可变类型.也就是说,改变了数字的值会生成新的对象. 在Python中删除数字 ...

  9. FILETIME类型到LARGE_INTEGER类型的转换

    核心编程第5版 245页到247页的讲到SetWaitableTimer函数的使用 其中提到 FILETIME类型到LARGE_INTEGER类型的转换问题,如下代码 //我们声明的局部变量 HAND ...

随机推荐

  1. 屏蔽按CapsLock键切换到大写时,编辑框自动弹出的提示(UnregisterClass(TOOLTIPS_CLASS)后,重新设置WndProc并注意返回值)

    WNDPROC OldProc; LPCTSTR lpStr = TEXT("保持大写锁定打开可能会使您错误输入密码"); LRESULT CALLBACK WindowProc( ...

  2. [2017.02.21-22] 《Haskell趣学指南 —— Learning You a Haskell for Great Good!》

    {- 2017.02.21-22 <Haskell趣学指南 -- Learning You a Haskell for Great Good!> 学习了Haskell的基本语法,并实现了一 ...

  3. [2017.02.06] 阅读《Effective Morden C++》

  4. 数组Array

    数组Array是最基本的数据结构,在内存中为一段定长连续内存,很多编程语言都有实现. 一.一维数组 下面代码实现了一维数组和它的遍历. clear并非清空数组,而是采用具体值对数组进行初始化. imp ...

  5. Android native进程间通信实例-binder篇之——HAL层访问JAVA层的服务

    有一天在群里聊天的时候,有人提出一个问题,怎样才能做到HAL层访问JAVA层的接口?刚好我不会,所以做了一点研究. 之前的文章末尾部分说过了service call 可以用来调试系统的binder服务 ...

  6. 10 table标签

    <!DOCTYPE html> <html lang="zh-CN"> <head> <meta charset="UTF-8& ...

  7. Azkaban —— 编译及部署

    一.Azkaban 源码编译 1.1 下载并解压 Azkaban 在3.0版本之后就不提供对应的安装包,需要自己下载源码进行编译. 下载所需版本的源码,Azkaban的源码托管在GitHub上,地址为 ...

  8. outerHTML、innerHTML以及innerText三者的区别

  9. python爬虫调用谷歌翻译接口

    2019年7月4日15:53:17 (¦3[▓▓] 晚安 谷歌翻译环境 Python 3.6 第三方库 Execjs (pip install PyExecJS ) 文件列表 同目录下的四个文件: - ...

  10. python读取excel文件中所有sheet表格

    sales: store: """(1)用load_workbook函数打开excel文件,返回一个工作簿对象 (2)用工作簿对象获取所有的sheet (3)第一个for ...