神奇的decimal,也许面试会问到哦~

简介:

继续扯,为什么说decimal神奇呢,大家都知道decimal是基元类型,但是

这个decimal类型在IL中居然没有相应的IL指令,也就是说CLR根本不认识decimal,全是编译器这一层在糊弄我们。

     话不多说,看下最simple的例子,(加了点注释方便理解)

static void Main(string[] args)
        {
            //居然调用了有参构造函数
            decimal d = 1;

            //直接将常量10推送到计算堆栈,然后将10放入局部变量索引为1的位置,也就是i
            int i = 10;

            //居然调用了隐式转换操作符,IL中就是调用相应的方法
            d = i;

            //居然调用了显式转换操作符,IL中就是调用相应的方法
            i = (int)d;
        }

 

从IL中可以看到,对decimal的所有操作最后玩的都是方法,对编译器上层的我们而言却一无所知,那么下一个问题来了,这些

都是怎么做到的呢?

 

一:decimal源代码

当我们对decimal的实现充满好奇心的时候,最满足的方式的就是看源代码了,大家应该都有对新鲜事物的好奇心,不管看不看

得懂都得装X看。

1:implicit/explicit 操作符

    从下面的IL中我们看到了这些乱七八槽的操作符,可能我们用的比较少或者有些人都没看过,不过终有它的用武之地。

 

结合上面的IL代码,我们发现了implicit和explicit关键字,这两个就是所谓的转换操作符,顾名思义,implicit就是所谓的隐式转换

操作符,explicit是显式转换了,再结合上面的IL代码,我们会发现给我们最终生成的是op_Implicit 和 op_Explicit方法。

可能有些人看不明白了,那我就举个例子吧。

public class Program
    {
        static void Main(string[] args)
        {
            //这里就是语法糖,c=10 最终调用的就是:隐式转换调用
            Complex c = 10;

            //语法糖,(int)最终调用的是:显式转换调用
            int j = (int)c;
        }
    }

    public struct Complex
    {
        public Complex(int num) { }

        /// <summary>
        /// 隐式转换调用的方法
        /// </summary>
        /// <param name="value"></param>
        /// <returns></returns>
        public static implicit operator Complex(int value)
        {
            return new Complex(value);
        }

        /// <summary>
        /// 强制转换调用的方法
        /// </summary>
        /// <param name="value"></param>
        /// <returns></returns>
        public static explicit operator int(Complex value)
        {
            return Convert.ToInt32(value);
        }
    }

从我的sample和IL中看,我想你应该清楚了,为了方便我们编码效率以及更好的让人理解,C#提供了这么个好玩的语法糖,清晰明了。

 

2:op_*** 重载操作符

    既然是基元类型就避免不了大量的算术运算和比较元算,那这些decimal又是如何做到的?还是继续在源代码里面找找。

 

   从源代码里面可以看到,原来C#用operator重载运算符对我们这个++,--,!=,<= 进行了重载,和转换运算符一样,最终

在IL层也是转换为各种方法。

   还是看个例子:

static void Main(string[] args)
        {
            decimal i = 10;

            decimal j = 12;

            var r1 = i > j;

            var r2 = i == j;
        }

 

好了,我想你一切都清楚了,当我们在愉快的写着++,--的时候,殊不知编译器给我们做的太多太多,最后得要感谢一下编译器。

相关文章
|
4月前
|
存储 Java
【IO面试题 四】、介绍一下Java的序列化与反序列化
Java的序列化与反序列化允许对象通过实现Serializable接口转换成字节序列并存储或传输,之后可以通过ObjectInputStream和ObjectOutputStream的方法将这些字节序列恢复成对象。
|
24天前
|
存储 缓存 算法
面试官:单核 CPU 支持 Java 多线程吗?为什么?被问懵了!
本文介绍了多线程环境下的几个关键概念,包括时间片、超线程、上下文切换及其影响因素,以及线程调度的两种方式——抢占式调度和协同式调度。文章还讨论了减少上下文切换次数以提高多线程程序效率的方法,如无锁并发编程、使用CAS算法等,并提出了合理的线程数量配置策略,以平衡CPU利用率和线程切换开销。
面试官:单核 CPU 支持 Java 多线程吗?为什么?被问懵了!
|
1月前
|
存储 算法 Java
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
本文详解自旋锁的概念、优缺点、使用场景及Java实现。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
|
1月前
|
存储 缓存 Java
大厂面试必看!Java基本数据类型和包装类的那些坑
本文介绍了Java中的基本数据类型和包装类,包括整数类型、浮点数类型、字符类型和布尔类型。详细讲解了每种类型的特性和应用场景,并探讨了包装类的引入原因、装箱与拆箱机制以及缓存机制。最后总结了面试中常见的相关考点,帮助读者更好地理解和应对面试中的问题。
52 4
|
2月前
|
算法 Java 数据中心
探讨面试常见问题雪花算法、时钟回拨问题,java中优雅的实现方式
【10月更文挑战第2天】在大数据量系统中,分布式ID生成是一个关键问题。为了保证在分布式环境下生成的ID唯一、有序且高效,业界提出了多种解决方案,其中雪花算法(Snowflake Algorithm)是一种广泛应用的分布式ID生成算法。本文将详细介绍雪花算法的原理、实现及其处理时钟回拨问题的方法,并提供Java代码示例。
85 2

热门文章

最新文章

相关实验场景

更多