服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - C# - C# 循环判断会进来几次的实现代码

C# 循环判断会进来几次的实现代码

2022-02-24 14:19lindexi C#

这篇文章主要介绍了C# 循环判断会进来几次的实现代码,代码中就一个循环,循环的判断是从一个函数获取值,需要的朋友可以参考下

最近有小伙伴告诉我,在循环的判断条件只会计算一次,本金鱼不相信,于是就做了测试,本文记录我做的测试。

先来写一个简单的代码, 就一个循环,循环的判断是从一个函数获取值

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
class program
  {
    static void main(string[] args)
    {
      var meepemorcear = new meepemorcear();
      meepemorcear.birmerulerrayjairbay();
    }
  }
  class meepemorcear
  {
    public void birmerulerrayjairbay()
    {
      for (int i = 0; i < daydrearnenawerlai(); i++)
      {
        console.writeline("第" +i.tostring()+"个逗比");
      }
    }
    public int daydrearnenawerlai()
    {
      console.writeline("进入");
      return 10;
    }
  }

通过 main 调用 birmerulerrayjairbay ,这个函数里面的 for 判断是 daydrearnenawerlai 拿到一个值,我尝试在 release 运行,结果每个判断都需要进入 daydrearnenawerlai 函数,请看输出

进入
第0个逗比
进入
第1个逗比
进入
第2个逗比
进入
第3个逗比
进入
第4个逗比
进入
第5个逗比
进入
第6个逗比
进入
第7个逗比
进入
第8个逗比
进入
第9个逗比

也就是在 debug 或 release 下,for 里面的判断都是需要执行,所以在 for 里的判断如果写了很长的计算,那么就会在每次循环都需要重新计算。即使每次计算出来的值都是一样,也需要重新计算。

所以这样看起来性能不如这样写,使用一个临时的变量获取判断的值

?
1
2
3
4
5
6
7
8
public void birmerulerrayjairbay()
    {
      var mowraitepalor = daydrearnenawerlai();
      for (int i = 0; i < mowraitepalor; i++)
      {
        console.writeline("第" +i.tostring()+"个逗比");
      }
    }

但是很快,另一个小伙伴告诉我,你把输出去掉,然后使用断点,你再看看

C# 循环判断会进来几次的实现代码

我添加了断点,在断点输出 123 然后运行

C# 循环判断会进来几次的实现代码

这时我发现运行没有输出 123 也就是函数没有进来,我再次添加断点,跟着函数也没有访问

所以这时的 daydrearnenawerlai 函数就被优化掉了

我和一个小伙伴说了这个问题,他说是被 il 优化了,我一点不相信,所以我就去看 il 代码

从下面的代码

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
public void birmerulerrayjairbay()
    {
      for (int i = 0; i < daydrearnenawerlai(); i++)
      {
        console.writeline("第" +i.tostring()+"个逗比");
      }
    }
    /// <summary>
    /// 进入lindexi.github.io可以看到更多博客
    /// </summary>
    /// <returns></returns>
    public static int daydrearnenawerlai()
    {
      return 10;
    }

转 il 可以看到下面代码,我会在 il 添加很多注释,所以很容易看懂。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
.method public hidebysig instance void
  birmerulerrayjairbay() cil managed
 {
  .maxstack 3
  .locals init (
   [0] int32 i
  )
  // 第 23 行 18 个字符到 27 个字符
  // [23 18 - 23 27]
  il_0000: ldc.i4.0
  // 定义 i ,代码的 int i = 0; 
  il_0001: stloc.0   // i
  il_0002: br.s     il_0023
  // 这里就是进入循环 for ,在 il 不管 for 还是 while 都是差不多
  // start of loop, entry point: il_0023
   // [25 17 - 25 60]
   // 下面这个代码就是 console.writeline("第" +i.tostring()+"个逗比"); 从代码可以看到
   // 需要先申请"第"
   il_0004: ldstr    "第"
   // 然后把 i 放入栈
   il_0009: ldloca.s   i
   // 调用 int.tostring ,使用的是实例的方法
   il_000b: call     instance string [mscorlib]system.int32::tostring()
   // 把"个逗比"放入栈
   il_0010: ldstr    "个逗比"
   // 调用字符串组合方法,组合三个字符串,返回一个字符串。
   // 把刚才入栈三个字符串出栈,返回的字符串入栈
   il_0015: call     string [mscorlib]system.string::concat(string, string, string)
   // 调用 console.writeline ,从栈拿到一个字符串输出
   il_001a: call     void [mscorlib]system.console::writeline(string)
   // 下面是 i++ 代码
   // [23 55 - 23 58]
   // 将指定索引处的局部变量加载到计算堆栈上,这里的索引是 0 ,在代码的变量是 i 所以把 i 加载到计算堆栈
   il_001f: ldloc.0   // i
   // 将整数值 1 作为 int32 推送到计算堆栈上
   il_0020: ldc.i4.1  
   // 从堆栈出栈两个数值进行相加,返回的值放在栈
   il_0021: add    
   // 从计算堆栈的顶部弹出当前值并将其存储到指定索引处的局部变量列表中,这里索引是 0 ,在代码的变量是 i ,所以 i = i + 1 的代码就是这样做
   il_0022: stloc.0   // i
   // 从堆栈加载 i ,把 i 入栈
   // [23 29 - 23 53]
   il_0023: ldloc.0   // i
   // 调用方法 daydrearnenawerlai 拿到返回值
   il_0024: call     int32 muhoubearwhedoofi.meepemorcear::daydrearnenawerlai()
   // 如果第一个值小于第二个值,则将控制转移到目标指令,这里的第一个值就是 i ,第二个值就是 daydrearnenawerlai 的返回值。跳转到标签 il_0004 ,如果没有小于,就继续代码。
   il_0029: blt.s    il_0004
  // end of loop
  // [27 9 - 27 10]
  il_002b: ret    
 } // end of method meepemorcear::birmerulerrayjairbay
 .method public hidebysig static int32
  daydrearnenawerlai() cil managed
 {
  .maxstack 8
  // 把一个值 放入堆栈,放入的是 10 ,然后从栈拿到值返回
  // [36 13 - 36 23]
  il_0000: ldc.i4.s   10 // 0x0a
  il_0002: ret    
 } // end of method meepemorcear::daydrearnenawerlai

从上面代码可以发现,实际 daydrearnenawerlai 没有被优化掉,还是有这个方法。

总结

以上所述是小编给大家介绍的c# 循环判断会进来几次的实现代码,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对服务器之家网站的支持!

原文链接:https://lindexi.gitee.io/lindexi/post/C-循环的判断会进来几次.html

延伸 · 阅读

精彩推荐
  • C#利用C#实现网络爬虫

    利用C#实现网络爬虫

    这篇文章主要介绍了利用C#实现网络爬虫,完整的介绍了C#实现网络爬虫详细过程,感兴趣的小伙伴们可以参考一下...

    C#教程网11852021-11-16
  • C#如何使用C#将Tensorflow训练的.pb文件用在生产环境详解

    如何使用C#将Tensorflow训练的.pb文件用在生产环境详解

    这篇文章主要给大家介绍了关于如何使用C#将Tensorflow训练的.pb文件用在生产环境的相关资料,文中通过示例代码介绍的非常详细,需要的朋友可以参考借鉴...

    bbird201811792022-03-05
  • C#C#微信公众号与订阅号接口开发示例代码

    C#微信公众号与订阅号接口开发示例代码

    这篇文章主要介绍了C#微信公众号与订阅号接口开发示例代码,结合实例形式简单分析了C#针对微信接口的调用与处理技巧,需要的朋友可以参考下...

    smartsmile20127762021-11-25
  • C#VS2012 程序打包部署图文详解

    VS2012 程序打包部署图文详解

    VS2012虽然没有集成打包工具,但它为我们提供了下载的端口,需要我们手动安装一个插件InstallShield。网上有很多第三方的打包工具,但为什么偏要使用微软...

    张信秀7712021-12-15
  • C#C#设计模式之Strategy策略模式解决007大破密码危机问题示例

    C#设计模式之Strategy策略模式解决007大破密码危机问题示例

    这篇文章主要介绍了C#设计模式之Strategy策略模式解决007大破密码危机问题,简单描述了策略模式的定义并结合加密解密算法实例分析了C#策略模式的具体使用...

    GhostRider10972022-01-21
  • C#SQLite在C#中的安装与操作技巧

    SQLite在C#中的安装与操作技巧

    SQLite,是一款轻型的数据库,用于本地的数据储存。其优点有很多,下面通过本文给大家介绍SQLite在C#中的安装与操作技巧,感兴趣的的朋友参考下吧...

    蓝曈魅11162022-01-20
  • C#深入理解C#的数组

    深入理解C#的数组

    本篇文章主要介绍了C#的数组,数组是一种数据结构,详细的介绍了数组的声明和访问等,有兴趣的可以了解一下。...

    佳园9492021-12-10
  • C#三十分钟快速掌握C# 6.0知识点

    三十分钟快速掌握C# 6.0知识点

    这篇文章主要介绍了C# 6.0的相关知识点,文中介绍的非常详细,通过这篇文字可以让大家在三十分钟内快速的掌握C# 6.0,需要的朋友可以参考借鉴,下面来...

    雨夜潇湘8272021-12-28