ref與out的區別、冒泡排序、普通排序,以及二分法查詢

来源:http://www.cnblogs.com/shmf/archive/2017/09/08/7492937.html
-Advertisement-
Play Games

一、首先我們先講一下ref與out的區別和使用方法; 1、ref與out的區別: out:需要在使用前聲明變數,分配地址但不能賦值,但是需要在使用中的時候需要初始化(進入方法體中的時候需要先賦值在使用),至於為什麼要在方法體中使用,我個人認為是為了區別ref;(即只出不進) ref:需要在使用前聲明 ...


一、首先我們先講一下ref與out的區別和使用方法;

    1、ref與out的區別:

    out:需要在使用前聲明變數,分配地址但不能賦值,但是需要在使用中的時候需要初始化(進入方法體中的時候需要先賦值在使用),至於為什麼要在方法體中使用,我個人認為是為了區別ref;(即只出不進)

    ref:需要在使用前聲明且初始化,分配地址並且賦值,這樣做可以根據初始化的值帶入,可以根據傳入的值進行一些邏輯判斷;(即有進有出,有頭有尾)

    共同點:都需要先聲明變數,且都有回傳值。

 2、使用方法:

    首先我們先看看兩者使用方法:首先我們先創建兩個方法一個用out一個用ref,回傳一個int值

    

private static int[] bubbleSort(int[] sources,out int count)
            {
                int temp; 
          count = 0; for (int i = 0; i < sources.Length; i++) { for (int j = i + 1; j < sources.Length; j++) { if (sources[j] < sources[i]) { temp = sources[j]; sources[j] = sources[i]; sources[i] = temp; } count++; } } return sources; }

 private static int[] bubbleSort2(int[] sources, ref int count)
            {
                int i, j, temp;
                for (j = 0; j < sources.Length; j++)
                {
                    for (i = 0; i < sources.Length - 1; i++)
                    {
                        if (sources[i] > sources[i + 1])
                        {
                            temp = sources[i];
                            sources[i] = sources[i + 1];
                            sources[i + 1] = temp;
                        }

                        count++;
                    }
                }
                return sources;
            }

 

標黃的就是我們用到的out,ref,在兩個方法體中就可以發現不一樣之處,out方法里的count這個參數進入後就有初始化值,然後在後面才能使用,而下麵用ref的方法體中我們沒有發現count的初始化,就可以直接使用,那麼兩者在調用的區別就在於調用的時候了,下麵就是方法調用的時候:

 static void Main(string[] args)
            {
                int[] array = new[] { 1223, 918, 234, 765, 974, 867, 86786, 145432, 867633, 9999999 }; // 目標數組
                int findValue = 145432; // 被查找數
                //二分法結果
                Console.WriteLine(BinarySearch(array, findValue, 0, array.Length - 1) ? "被查找數存在數組array中" : "被查找數不存在數組array中");
                //冒泡排序
                int count;
                int[] intlist = bubbleSort(array, out count);
               
                for (int i = 0; i < intlist.Length; i++)
                {
                    Console.Write(intlist[i]+"   ");

                } Console.WriteLine("\t"+"迴圈次數為:"+count);

                //類似冒泡排序,單迴圈次數較多
                int count2 = 0;
                int[] intlist2 = bubbleSort2(array, ref count2);
                for (int i = 0; i < intlist2.Length; i++)
                {
                    Console.Write(intlist2[i] + "   ");

                } Console.WriteLine("\t" + "迴圈次數為:" + count2);
                Console.ReadKey();
            }

先只看有深色背景顏色的地方,因為在之前我悶在方法體中看見out在方法體中已經有初始化動作,而ref沒有,那麼再調用之前ref就需要先初始化,out就不需要初始化!

這裡我們在擴展一下,我們知道return也是可以返回值,那麼return的返回值和上述兩者有什麼區別呢?

    首先我們之前說了,out和ref,只要在參數值前面註明out或者ref,那麼咱們的返回值可以實現多個,但是用return的話只能是返回一個唯一值,這是最大的區別!

並且return的返回值是直接不可修改的,但是out和ref是可以修改的!

 

二、接下來我們就看看最常見的排序演算法

    1、(普通排序)首先,我們還是借用上述代碼,

      首先我們先創建一個數組,同樣的 調用我們寫好的方法,

       /// <summary>
            /// 非冒泡排序
            /// </summary>
            /// <param name="sources">目標數組</param>
            /// <param name="count">迴圈次數</param>
            /// <returns>升序排列結果</returns>
            private static int[] bubbleSort2(int[] sources, ref int count)
            {
                int i, j, temp;
                for (j = 0; j < sources.Length; j++)
                {
                    for (i = 0; i < sources.Length - 1; i++)
                    {
                        if (sources[i] < sources[i + 1])
                        {
                            temp = sources[i];
                            sources[i] = sources[i + 1];
                            sources[i + 1] = temp;
                        }

                        count++;
                    }
                }
                return sources;
            }

 此代碼有內外迴圈,外迴圈是增加迴圈次數,內迴圈是則是主迴圈,若當前值sources[i]與下一個值對比,如果滿足條件那麼就將此值記錄下來,這裡面他會將所有的都迴圈一遍,才結束!

2、(冒泡排序)

 1  /// <summary>
 2             /// 冒泡排序
 3             /// </summary>
 4             /// <param name="sources">目標數組</param>
 5             /// <param name="count">迴圈次數</param>
 6             /// <returns>升序排列結果</returns>
 7             private static int[] bubbleSort(int[] sources,out int count)
 8             {
 9                 int temp; count = 0;
10                 for (int i = 0; i < sources.Length; i++)
11                  {
12                      for (int j = i + 1; j < sources.Length; j++)
13                      {
14                          if (sources[j] > sources[i])
15                          {
16                              temp = sources[j];
17                              sources[j] = sources[i];
18                              sources[i] = temp;
19                          }
20                          count++;
21                      }
22                  }
23                 return sources;
24             }

上述代碼我們看出同樣是兩個迴圈,但是不一樣就在於內迴圈中,內迴圈的迴圈次數我們看到了,他是根據外迴圈的次數來相對的,如果外迴圈顯示第一個數據,那麼內迴圈則是顯示第二個數字,所以兩個在結果上是一樣的,但是在迴圈次數上這個(冒泡排序)就比那個快一倍,接下來我們就執行以下這兩個方法,在此同時我們同時在看上述中調用的out和ref的用法也用上了

 1 static void Main(string[] args)
 2             {
 3                 int[] array = new[] { 1223, 918, 234, 765, 974, 867, 86786, 145432, 867633, 9999999 }; // 目標數組
 4              
 5                 //冒泡排序
 6                 int count;
 7                 int[] intlist = bubbleSort(array, out count);
 8                
 9                 for (int i = 0; i < intlist.Length; i++)
10                 {
11                     Console.Write(intlist[i]+"   ");
12 
13                 } Console.WriteLine("\t"+"迴圈次數為:"+count);
14 
15                 //類似冒泡排序,單迴圈次數較多
16                 int count2 = 0;
17                 int[] intlist2 = bubbleSort2(array, ref count2);
18                 for (int i = 0; i < intlist2.Length; i++)
19                 {
20                     Console.Write(intlist2[i] + "   ");
21 
22                 } Console.WriteLine("\t" + "迴圈次數為:" + count2);
23                 Console.ReadKey();
24             }

其運行結果如下

 

結果我們看到了此方法運行,也通過out與ref傳出了我們需要的迴圈次數的值!

三,接下來我們看看二分法

    二分法:通俗理解為在一個大數據中查找需要的值,如果一個人查找的話想對費力,兩個人的話就相對快得多,

      我們看下代碼

 1 static void Main(string[] args)
 2             {
 3                 int[] array = new[] { 1223, 918, 234, 765, 974, 867, 86786, 145432, 867633, 9999999 }; // 目標數組
 4                 int findValue = 145432; // 被查找數
 5                 //二分法結果
 6                 Console.WriteLine(BinarySearch(array, findValue, 0, array.Length - 1) ? "被查找數存在數組array中" : "被查找數不存在數組array中");
 7             }
 8 
 9             /// <summary>
10             /// 二分查找/折半查找(分治思想、遞歸,目標數組必須是有序序列),演算法複雜度為o(log(n),n代表目標數組長度)
11             /// </summary>
12             /// <param name="sources">目標數組</param>
13             /// <param name="findValue">目標查找數</param>
14             /// <param name="low">區間最小索引</param>
15             /// <param name="high">區間最大索引</param>
16             /// <returns>true:存在,false,不存在</returns>
17             private static bool BinarySearch(int[] sources, int findValue, int low, int high)
18             {
19                 // 未找到,終止遞歸
20                 if (low > high) return false;
21 
22                 // 折半查找中間值 索引:(a + b) / 2表示算數平均數,即中點
23                 int middleIndex = (low + high) % 2 == 0 ? (low + high) / 2 : (low + high) / 2 + 1;
24 
25                 if (findValue > sources[middleIndex])
26                 {
27                     // 大於中間值,在區間[middleIndex + 1, high]遞歸繼續查找
28                     return BinarySearch(sources, findValue, middleIndex + 1, high);
29                 }
30                 if (findValue < sources[middleIndex])
31                 {
32                     // 小於中間值,在區間[low, middleIndex - 1]遞歸繼續查找
33                     return BinarySearch(sources, findValue, low, middleIndex - 1);
34                 }
35 
36                 // findValue 等於 sources[middleIndex],找到,終止遞歸
37                 return true;
38             }

 

這就是通過二分法實現查找方式。

 

以上僅為隨筆,若有錯誤,或有所指點之地,希望大神們不要吝嗇。謝謝!

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • Linux對於記憶體的管理涉及到非常多的方面,這篇文章首先從對進程虛擬地址空間的管理說起。(所依據的代碼是2.6.32.60) 無論是內核線程還是用戶進程,對於內核來說,無非都是task_struct這個數據結構的一個實例而已,task_struct被稱為進程描述符(process descripto ...
  • 首先找出所有可選的佈局(layout)方案: 可以看到 us 下有很多常見的佈局方案(以下為節選): 但文檔可能不全,比如早在 13 年加入的 norman 方案就未在 man 手冊列出。 可以通過搜索 symbol 文件找到: 更改佈局方案: 加上 / 選項會顯示輸出信息: 切換回主流的 QWER ...
  • 轉:http://blog.csdn.net/hongchangfirst/article/details/7075026 大家都知道進程,可是知道linux是怎麼管理其進程的嗎?每一個進程都有一個進程描述符,具體是task_struct結構體存儲相關的信息,在linux/sched.h文件里定義, ...
  • 一、分佈位置上的區別: kmalloc()和__get_free_pages()函數申請的記憶體位於物理記憶體的映射區域,而且在物理上也是連續的,它們與真實的物理地址只有一個固定的偏移,因此存在簡單的線性關係;(3G+896M)(低端記憶體); vmalloc函數申請的虛擬記憶體與物理記憶體之間也沒有簡單的換 ...
  • 添加樣式: 在html中,需要創建2層div來實現。一個div包含另一個div: 效果: ...
  • 表格控制項 Spread Studio 發佈了全新的 V11 CTP 版本。在此版本中,Spread For WinForms 引入了 Spread Common,也帶來了 Spread 性能的巨大提升和記憶體消耗的急劇下降。 ...
  • 1. 前言 做了WPF開發多年,一直未曾自己實現一個自定義Window Style,無論是《WPF編程寶典》或是各種博客都建議使用WindowStyle="None" 和 AllowsTransparency="True",於是想當然以為這樣就可以了。最近來了興緻想自己實現一個,才知道WindowS ...
  • Time Protocol(RFC-868)是一種非常簡單的應用層協議:它返回一個32位的二進位數字,這個數字描述了從1900年1月1日0時0分0秒到現在的秒數,伺服器在TCP的37號埠監聽時間協議請求。本函數將伺服器返回值轉化成本地時間。 先前不知道有現成的IPAddress.NetworkTo ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...