首页 > 编程语言 >c# is 和 as 浅看重制版

c# is 和 as 浅看重制版

时间:2024-11-30 20:22:06浏览次数:4  
标签:obj ConsoleApp4 c# void 看重 制版 Program IL Teacher

前言

当年写的比较差:https://www.cnblogs.com/aoximin/p/12965408.html,所以特来重新写一遍。

正文

首先为什么会出现is 和 as 呢?

因为是为了有需要检验的地方,如果直接使用显示转换的话,那么可能直接报错了。

namespace ConsoleApp4
{
    class Program
    {
        static void Main(string[] args)
        {
            object obj = new Student();
            var c = (Teacher)obj;
        }

        public class Student
        {
            
        }

        public class Teacher
        { 
        }
    }
}

比如这样肯定会报错的, 因为在运行的时候我们来查看一下il语句。

.method private hidebysig static 
	void Main (
		string[] args
	) cil managed 
{
	.custom instance void System.Runtime.CompilerServices.NullableContextAttribute::.ctor(uint8) = (
		01 00 01 00 00
	)
	// Method begins at RVA 0x20a4
	// Header size: 12
	// Code size: 15 (0xf)
	.maxstack 1
	.entrypoint
	.locals init (
		[0] object obj,
		[1] class ConsoleApp4.Program/Teacher c
	)

	IL_0000: nop
	IL_0001: newobj instance void ConsoleApp4.Program/Student::.ctor()
	IL_0006: stloc.0
	IL_0007: ldloc.0
	IL_0008: castclass ConsoleApp4.Program/Teacher
	IL_000d: stloc.1
	IL_000e: ret
} // end of method Program::Main

会调用castclass 进行转换。

因为castclass 无法找到他们两个的转换方法(编写显示转换或者隐式转换的方法),他们也不是继承关系,所以会抛出异常。

但是也不要以为所以的强制转换就一定要编写啥编写显示转换或者隐式转换的方法或者是啥继承关系,还有一种是编译器行为。

比如说:

long a = 1;
int b = (int)a;

编译出来的代码是:

IL_000d: stloc.1
IL_000e: ldc.i4.1
IL_000f: conv.i8
IL_0010: stloc.2
IL_0011: ldloc.2
IL_0012: conv.i4
IL_0013: stloc.3

这就是编译器的行为了,编译器认为自己可以处理就不报错了,直接截断作为处理了。

好吧,不能走的太远了,回到is 上。

那么is就可以避免一些运行时候的报错,而不需要用try catch 这种不太优雅的方式。

但是呢,is 是无法去检查自己编写的显示转换或隐式转换的方法,请看VAR:

class Program
{
	static void Main(string[] args)
	{
		object obj = new Student();
		if (obj is Teacher)
		{
			var c = (Teacher)obj;
		}
	}

	public class Student
	{
		
	}

	public class Teacher
	{ 
	}
}

编译出来呢?是下面这样:

.method private hidebysig static 
	void Main (
		string[] args
	) cil managed 
{
	.custom instance void System.Runtime.CompilerServices.NullableContextAttribute::.ctor(uint8) = (
		01 00 01 00 00
	)
	// Method begins at RVA 0x20a4
	// Header size: 12
	// Code size: 30 (0x1e)
	.maxstack 2
	.entrypoint
	.locals init (
		[0] object obj,
		[1] bool,
		[2] class ConsoleApp4.Program/Teacher c
	)

	IL_0000: nop
	IL_0001: newobj instance void ConsoleApp4.Program/Student::.ctor()
	IL_0006: stloc.0
	IL_0007: ldloc.0
	IL_0008: isinst ConsoleApp4.Program/Teacher
	IL_000d: ldnull
	IL_000e: cgt.un
	IL_0010: stloc.1
	IL_0011: ldloc.1
	IL_0012: brfalse.s IL_001d

	IL_0014: nop
	IL_0015: ldloc.0
	IL_0016: castclass ConsoleApp4.Program/Teacher
	IL_001b: stloc.2
	IL_001c: nop

	IL_001d: ret
} // end of method Program::Main

is 转换为il代码就是isinst

这个只能判断继承关系,所以嘛,这个呢,其实也能理解,如果是自己编写了转换方法,哪里自己不知道还要is呢。

那么为啥会出现as呢?

还是性能问题嘛。

v1:

static void Main(string[] args)
{
	object obj = new Student();
	if (obj is Teacher)
	{
		var c = (Teacher)obj;
	}
}

v2:

static void Main(string[] args)
{
	object obj = new Student();
	var c = obj as Teacher;
}

public class Student
{
	
}

public class Teacher
{ 
}

这两者有啥区别呢?道理上运行结果都一致。

但是呢,v2 更优,因为v2检查了一次,而v1检查来了两次。

v2的il代码:

.method private hidebysig static 
	void Main (
		string[] args
	) cil managed 
{
	.custom instance void System.Runtime.CompilerServices.NullableContextAttribute::.ctor(uint8) = (
		01 00 01 00 00
	)
	// Method begins at RVA 0x20a4
	// Header size: 12
	// Code size: 15 (0xf)
	.maxstack 1
	.entrypoint
	.locals init (
		[0] object obj,
		[1] class ConsoleApp4.Program/Teacher c
	)

	IL_0000: nop
	IL_0001: newobj instance void ConsoleApp4.Program/Student::.ctor()
	IL_0006: stloc.0
	IL_0007: ldloc.0
	IL_0008: isinst ConsoleApp4.Program/Teacher
	IL_000d: stloc.1
	IL_000e: ret
} // end of method Program::Main

这里也是使用了isinst,因为isinst本身就是检测并且赋值,再来看下isinst的功效

as 只是运行了stloc.1,弹出赋值给局部变量,多了一个赋值过程,两者性能都差不多,所以呢,不要觉得as 好像多比 is 多个一个转换啥的,其实调用的是同一个语句。

只是觉得以前写的过于潦草,整理下罢了。操作系统篇马上就要出炉,比较生硬,共128篇,简单整理一下。

标签:obj,ConsoleApp4,c#,void,看重,制版,Program,IL,Teacher
From: https://www.cnblogs.com/aoximin/p/18572890

相关文章

  • C语言 神奇的幻方(洛谷 p2615 )幻方是一种很神奇的N*N矩阵
            题目:神奇的幻方(洛谷p2615)幻方是一种很神奇的N*N矩阵:它由数字1,2,3,…,N*N构成,且每行、每列及两条对角线上的数字之和都相等。当N为奇数时,可以通过以下方法构建一个幻方:首先将1写在第一行的中间;之后,按如下方式从小到大依次填写每个数k(k=2,3,…,N*N)若(k-1)在第一......
  • 【C语言】二维数组的声明
    一、1.在被调用函数的形参数组声明可以省略第一维的大小,第二维不可省略2.实参数组第一维第二维都需要#include<stdio.h>intmain(){ inta[][]; //Error intb[1][]; //Error intc[][1]; //Error intd[1][1]; return0;}voidf(inta[][]) //Error{ }v......
  • 【docker】Dockerfile指令讲解,与企业案例应用
    Dockerfile简介Dockerfile是一个文本文件,包含了构建Docker镜像所需的所有命令。通过执行dockerbuild命令,Docker会按照Dockerfile中的指令一步步构建出镜像。Dockerfile基本结构一个简单的Dockerfile可能包含以下内容:FROMubuntu:20.04LABELmaintainer="y......
  • ORB-SLAM2 ----- LocalMapping::ComputeF12和ORBmatcher::CheckDistEpipolarLine
    文章目录一、函数意义二、LocalMapping::ComputeF12()1.函数讲解2.函数代码三、ORBmatcher::CheckDistEpipolarLine()1.函数讲解2.函数代码四、总结一、函数意义这两个函数在LocalMapping::CreateNewMapPoints()被调用,之所以单独拿出来讲,是因为这两个函都是计算的......
  • LeetCode 283:移动零的普通解法与优化解法(含动图演示)
    LeetCode刷题记录文章目录......
  • [ctf]跟着风二西复现NSSCTF流量题目
    题目参考博客https://blog.csdn.net/zerorzeror/article/details/135737476?spm=1001.2014.3001.550220241130 [GKCTF2021]签到解题过程可以看到流量并不多,看到GET和POST里面有tmpshell 然后追踪HTTP流  可以看到初始的这一段字符,因为字符中字母最大的为f,无其他字......
  • 48. Web前端网页案例——【王者荣耀游戏主题网页( 6页)】 大学生期末大作业 html5+css3+
    目录一、网页概述二、网页文件 三、网页效果四、代码展示1.html2.CSS3.JS五、总结1.简洁实用2.使用方便3.整体性好4.形象突出5.交互式强六、更多推荐♬♬♬​​​​​​​欢迎光临我的CSDN!这里是Web前端网页案例大集汇,有各行各业的前端网页案例,每天会持续更......
  • AI开发平台ModelArts-基本配置-在ECS中创建ma-user和ma-group
    在ModelArts训练平台使用的自定义镜像时,默认用户为ma-user、默认用户组为ma-group。如果在训练时调用ECS中的文件,需要修改文件权限改为ma-user可读,否则会出现Permissiondenied错误,因此需要在ECS中提前创建好ma-user和ma-group。在terminal中执行以下命令:default_user=$(gete......
  • 2024CCPC郑州邀请赛(组队VP)
    B.扫雷1面:T0xel喜欢玩扫雷,但是他玩的扫雷游戏有名为“地雷探测器”的特殊道具。具体来说,T0xel会进行$n$轮扫雷。每轮扫雷开始之前,T0xel会获得1枚扫雷币。扫雷币在每轮扫雷结束后不会回收,可以保留至下一轮扫雷。T0xel知道,在第$i$轮$(1\leqi\leqn)$扫雷中,......
  • 【Unity 任务系统工具】Quests 2 | Game Creator 2 强大的任务系统,用于管理和设计复杂
    Quests2|GameCreator2是由CatsoftWorks开发的Unity插件,是GameCreator2插件套件的一部分。它为开发者提供了一个强大的任务系统,用于管理和设计复杂的游戏任务、支线任务、日常任务等。这个插件能够帮助开发者轻松地创建任务链、条件触发和奖励系统,并与GameCre......