(教学思路 C#之类三)方法参数类型(ref、out、parmas)

简介:
   这节课我们来学习方法中的参数传递,在面向对象二中,我曾说过,参数也属于变量的一种,在c语言的学习时,同学们都学习过参数这个概念和用法,方法使用参数列表来传递变量的基本语法如下:
returnType  FunctionName(paraType1  paraName1,paraType2  paraName2,……)
{
   Function body;
}
其中的returnType是指方法的返回值,FunctionName指方法的标识符,paraType1指参数1的数据类型,paraName1是参数1的标识符,调用这种类型的方法的时候,必须传入与类型paraType1相符的参数值,并且传入的参数个数也必须相符,方法内部以参数名称paraName1来识别。
     在C#中,参数一共四种形式:分别是
参数类型 值参数
引用参数--ref
输出参数--out
动态参数--parmas

       他们都有什么区别呢?具体怎么用呢?我们现在就开始具体的讲解,
值参数
 
         值参数就是参数的是值类型的,如 int   add(int i,int j){},i和j就是值参数,值参数很好理解,我们也经常用到,下面的代码实例主要是为了对比引用参数和输出参数的区别的实现两个数值交换的例子,请同学们体会它们的不同。
值参数传参实例
InBlock.gif 1 //定义一个交换两个数的值的一个方法Exchange 
InBlock.gif 2                 //此处请注意,本方法使用了static关键字,原因是:根据我们类一所提到的,如果是实例方法(不加static关键字), 
InBlock.gif 3                 //必须通过类的对象来调用,即使在同一个类中,加入static关键字后,静态成员就属于类的引用范围了, 
InBlock.gif 4                 //因为在一个类中,所以不必类名.,在其他方法中直接就可以调用这个方法。对比下面的实例方法 
InBlock.gif 5                 static  void Exchange( int a,  int b) 
InBlock.gif 6                { 
InBlock.gif 7                        Console .WriteLine (); 
InBlock.gif 8                        Console.WriteLine( "值参数方法进来时,a={0}    b={1}", a, b); 
InBlock.gif 9                         int c = 0; 
InBlock.gif10                        c = a; 
InBlock.gif11                        a = b; 
InBlock.gif12                        b = c; 
InBlock.gif13                        Console.WriteLine(); 
InBlock.gif14                        Console.WriteLine( "经过值参数交换方法交换后,a={0}    b={1}", a, b); 
InBlock.gif15                        Console.WriteLine(); 
InBlock.gif16 
InBlock.gif17                } 
InBlock.gif18                 //看这个实例方法,对照上一个方法,看在main方法中的调用是不同的, 
InBlock.gif19                 //此方法必须通过Class2的对象s.来调用,而上一个方法是直接调用(因为在一个类中,省略了类名.来调用) 
InBlock.gif20                 void Exchange1() 
InBlock.gif21                {    
InBlock.gif22                } 
InBlock.gif23 
InBlock.gif24                 static  void Main() 
InBlock.gif25                { 
InBlock.gif26                         //输入初值 
InBlock.gif27                        Console.Write( "请输入整数a="); 
InBlock.gif28                         int a =  int.Parse(Console .ReadLine()); 
InBlock.gif29                         //输入初值 
InBlock.gif30                        Console.Write( "请输入整数b="); 
InBlock.gif31                         int b =  int.Parse(Console.ReadLine()); 
InBlock.gif32                         
InBlock.gif33                         //观察1:经过交换后a和b的结果的变化,(方法结束时交换了a和b的值) 
InBlock.gif34                         //2:对比下面的调用,记住静态方法调用的方式 
InBlock.gif35                         //3:在main方法结束前,我们再次打印main中定义的a和b的值的变化(和初值相同) 
InBlock.gif36                        Exchange(a,b); 
InBlock.gif37 
InBlock.gif38 
InBlock.gif39                         //对比上面的调用,记住实例方法调用的方式 
InBlock.gif40                        Class2 s =  new Class2(); 
InBlock.gif41                        s.Exchange1(); 
InBlock.gif42 
InBlock.gif43                         //因为是值参数的传递方法,不会改变Main方法中的a的值,结果应该和初值相同 
InBlock.gif44                        Console.WriteLine( "采用值参数交换方法后,Main方法中现在为a="+a); 
InBlock.gif45                        Console.WriteLine(); 
InBlock.gif46                         //因为是值参数的传递方法,不会改变Main方法中的b的值,结果应该和初值相同 
InBlock.gif47                        Console.WriteLine( "经过值参数交换方法后,Main方法中现在为b=" + b); 
InBlock.gif48                } 
InBlock.gif49 
InBlock.gif50                
InBlock.gif51        }
 
运行结果
请输入整数a=2
请输入整数b=4
值参数方法进来时,a=2  b=4
经过值参数交换方法交换后,a=4  b=2
采用值参数交换方法后,Main方法中现在为a=2
经过值参数交换方法后,Main方法中现在为b=4
请按任意键继续. . .
 
可以看出值参数经过方法传递后,初值是不改变的,下面我们来看一下引用参数。
引用参数

 
      如果方法的参数为数值类型时,则传递的方式主要分为两种:传值和传址。这两种方式最大的差异在于:当一个参数以传值的方式传递时,变量的值即使在方法中被改变,它本身还是维持一开始传入的值,我们刚才学习的值参数就是传值方式;而以传址方式传入的变量,当方法将其改变的时候,此变量的值便永远被更改,C#默认以传值方式传递参数,如果要使用传址方式传递,就需要用到下面我要说的理论。数值类型一律是以传值方式进行传递的,如果要改变这种行为,用传址方式传递参数,必须使用ref和out关键字进行修饰,当然如果参数本身就是引用类型的(string类型除外),不需要加ref和out关键字,他们的传递也是采用传址方式的.
      使用ref关键字修饰的引用参数传递的方法书写格式如:static  int add(ref int i,ref int j){}  .在参数数值类型的前面加入ref修饰,在调用这个方法时,传入的参数前也必须加ref关键字,如:add(ref a,ref b);
     下面我们上面的例子中Exchange方法中参数b变成ref形式,同学们来体会一下a作为值参数传值传递和b作为引用参数传址传递的区别:
值参数与引用参数的区别
InBlock.gif 1class Class2 
InBlock.gif 2        { 
InBlock.gif 3                 //定义一个交换两个数的值的一个方法Exchange 
InBlock.gif 4                 //此处请注意,本方法使用了static关键字,原因是:根据我们类一所提到的,如果是实例方法(不加static关键字), 
InBlock.gif 5                 //必须通过类的对象来调用,即使在同一个类中,加入static关键字后,静态成员就属于类的引用范围了, 
InBlock.gif 6                 //因为在一个类中,所以不必类名.,在其他方法中直接就可以调用这个方法。对比下面的实例方法 
InBlock.gif 7                 static  void Exchange( int a,  ref  int b) 
InBlock.gif 8                { 
InBlock.gif 9                        Console .WriteLine (); 
InBlock.gif10                        Console.WriteLine( "参数进来时,a={0}    b={1}", a, b); 
InBlock.gif11                         int c = 0; 
InBlock.gif12                        c = a; 
InBlock.gif13                        a = b; 
InBlock.gif14                        b = c; 
InBlock.gif15                        Console.WriteLine(); 
InBlock.gif16                        Console.WriteLine( "方法交换后,a={0}    b={1}", a, b); 
InBlock.gif17                        Console.WriteLine(); 
InBlock.gif18 
InBlock.gif19                } 
InBlock.gif20                 //看这个实例方法,对照上一个方法,看在main方法中的调用是不同的, 
InBlock.gif21                 //此方法必须通过Class2的对象s.来调用,而上一个方法是直接调用(因为在一个类中,省略了类名.来调用) 
InBlock.gif22                 void Exchange1() 
InBlock.gif23                {    
InBlock.gif24                } 
InBlock.gif25 
InBlock.gif26                 static  void Main() 
InBlock.gif27                { 
InBlock.gif28                         //输入初值 
InBlock.gif29                        Console.Write( "请输入整数a="); 
InBlock.gif30                         int a =  int.Parse(Console .ReadLine()); 
InBlock.gif31                         //输入初值 
InBlock.gif32                        Console.Write( "请输入整数b="); 
InBlock.gif33                         int b =  int.Parse(Console.ReadLine()); 
InBlock.gif34                         
InBlock.gif35                         //观察1:经过交换后a和b的结果的变化,(方法结束时交换了a和b的值) 
InBlock.gif36                         //2:对比下面的调用,记住静态方法调用的方式 
InBlock.gif37                         //3:main方法结束前,我们再次打印main中定义的a和b的值的变化:a和初值相同,b与出方法值相同 
InBlock.gif38                        Exchange(a, ref b); 
InBlock.gif39 
InBlock.gif40 
InBlock.gif41                         //对比上面的调用,记住实例方法调用的方式 
InBlock.gif42                        Class2 s =  new Class2(); 
InBlock.gif43                        s.Exchange1(); 
InBlock.gif44 
InBlock.gif45                         //因为是值参数的传递方法,不会改变Main方法中的a的值,结果应该和初值相同 
InBlock.gif46                        Console.WriteLine( "采用值参数传递交换方法后,结果应该和初值相同,Main方法中现在为a="+ a); 
InBlock.gif47                        Console.WriteLine(); 
InBlock.gif48                         //因为是引用ref参数的传递方法,会改变Main方法中的b的值,结果应该和出方法时的值相同 
InBlock.gif49                        Console.WriteLine( "经过引用参数传递交换方法后,结果应该和出方法时的值相同,Main方法中现在为b=" + b); 
InBlock.gif50                }        
InBlock.gif51        }
运行结果
请输入整数a=2
请输入整数b=4
参数进来时,a=2  b=4
方法交换后,a=4  b=2
采用值参数传递交换方法后,结果应该和初值相同,Main方法中现在为a=2
经过引用参数传递交换方法后,结果应该和出方法时的值相同,Main方法中现在为b=2
请按任意键继续. . .
    很明显看出,Main方法中b的值变化了,和出方法时的值相同,而a因为是传值传递,仍然是初值。参数是数值型的传址传递还有一种就是使用out关键字,那么它与ref有什么不同呢?
输出参数
      输出参数传址时,要使用out关键值,out关键字允许我们以一个未初始化的变量作为参数,直接传入方法当中,在方法结束的时候,指定所要返回的值给它,而使用ref关键字传给方法之前,必须将一个数值传入到方法中,out关键字在某种程度上来说,有点像return的功能,就是把方法的结果返回到调用这个方法的主方法中,使用out关键字要注意到:主方法中传入调用的方法的out参数值应该为未初始化的,即使你初始化了,初始化的值也相当于不存在。我们来看这个例子,体会一下out的用法。
比较out和return的用法
InBlock.gif 1class Class2 
InBlock.gif 2        { 
InBlock.gif 3                 //定义一个求正方形面积的方法,传入正方形的边长,返回正方形的面积 
InBlock.gif 4                 static  void QMJ( double bc, out     double mj) 
InBlock.gif 5                    { 
InBlock.gif 6                            mj = bc * bc;                
InBlock.gif 7                     } 
InBlock.gif 8 
InBlock.gif 9                 //对比使用return返回面积的方法,达到的效果是一样的 
InBlock.gif10                 static  double returnQMJ( double bc) 
InBlock.gif11                { 
InBlock.gif12                      return bc * bc; 
InBlock.gif13                } 
InBlock.gif14                 static  void Main() 
InBlock.gif15                { 
InBlock.gif16                        Console.Write( "请输入正方形的边长bc="); 
InBlock.gif17                         double bc =  double.Parse(Console.ReadLine()); 
InBlock.gif18                        Console.WriteLine(); 
InBlock.gif19                         //传入QMJ方法的mj是为赋初值的 
InBlock.gif20                         double mj; 
InBlock.gif21 
InBlock.gif22                         //通过方法调用,改变mj的值为出方法的值,打印出来,bc的值不变 
InBlock.gif23                        QMJ(bc,  out mj);                         
InBlock.gif24                        Console.WriteLine( "正方形的面积mj="+mj); 
InBlock.gif25 
InBlock.gif26                        Console.WriteLine( "================="); 
InBlock.gif27                         //使用return方式,返回方法的结果值 
InBlock.gif28                        Console.WriteLine( "使用returnQMJ方法面积mj="+returnQMJ(bc)); 
InBlock.gif29                }        
InBlock.gif30        }
 
运行结果
请输入正方形的边长bc=4
正方形的面积mj=16
=================
使用returnQMJ方法面积mj=16
请按任意键继续. . .
       这个例子我主要是想告诉大家使用out能够达到和return返回值一样的功能效果。
       下面我们来看一下为什么加了ref和out关键字后,主方法中的数值会改变呢?这主要是因为 存放参数的内存机制发生了变化而引起的。上面的3个例子都是数值类型作为参数,但是因为第二、三例子中加入了ref和out关键字,改变了数值类型按值传递的特性,改为按照传递在内存中存放的地址,按地址传递后参数的值会改变的原因就在此,因为按内存地址传递时,调用方法的参数在定义时,不会因为参数是数值类型而在线程堆栈中开辟新的空间来存放新的值,而是会在线程堆栈中存放一个指向主方法定义的变量(初值)的内存地址,当这个地址指向的内存值因为调用方法中对参数的作用(如交换,如平方)后,内存中的值就改成了调用方法作用后的值,这样在打印主方法的变量时,值自然就改成了调用方法后的值了。
       接下来我们来看一个参数为引用类型的实例,因为参数本身就是引用类型(string类型除外),所以这种参数传递属于传址传递:
数组为参数的传址传递
InBlock.gif 1class Class2 
InBlock.gif 2        {    
InBlock.gif 3                 //定义一个静态的方法,参数是引用类型的数组参数a, 
InBlock.gif 4                 //方法内部的功能是,改变数值a中每个元素的值,让改变后的值是原来的2倍 
InBlock.gif 5                 static  void ArrayMethod( int [] a) 
InBlock.gif 6                { 
InBlock.gif 7                        Console.WriteLine( "数值参数传入时,数组中的每个元素值是:"); 
InBlock.gif 8                         foreach ( int i  in a) 
InBlock.gif 9                        { 
InBlock.gif10                                Console.Write(i+ "    "); 
InBlock.gif11                        } 
InBlock.gif12                        Console.WriteLine(); 
InBlock.gif13                         //改变数值a中每个元素的值,让改变后的值是原来的2倍 
InBlock.gif14                         for ( int j = 0; j < a.Length; j++) 
InBlock.gif15                        { 
InBlock.gif16                                a[j] = a[j] * 2; 
InBlock.gif17                        } 
InBlock.gif18                        Console.WriteLine( "改变后,方法内部的数组a中的每个元素值是:"); 
InBlock.gif19                         foreach ( int i  in a) 
InBlock.gif20                        { 
InBlock.gif21                                Console.Write(i +  "    "); 
InBlock.gif22                        } 
InBlock.gif23                        Console.WriteLine(); 
InBlock.gif24                } 
InBlock.gif25             
InBlock.gif26                 static  void Main() 
InBlock.gif27                { 
InBlock.gif28                         //给数值a赋初值 
InBlock.gif29                         int[] a =  new  int[] {1,2,3 }; 
InBlock.gif30 
InBlock.gif31                         //调用方法将引用类型的数组参数a带入方法中 
InBlock.gif32                        ArrayMethod(a); 
InBlock.gif33 
InBlock.gif34                         //经过方法后引用类型的数组参数a的元素值应该发生改变, 
InBlock.gif35                        Console.WriteLine( "经过方法后,Main方法中数组a的每个元素值是:"); 
InBlock.gif36                         foreach ( int i  in a) 
InBlock.gif37                        { 
InBlock.gif38                                Console.Write(i +  "    "); 
InBlock.gif39                        } 
InBlock.gif40                }        
InBlock.gif41        }
运行结果:
数值参数传入时,数组中的每个元素值是:
1  2  3
改变后,方法内部的数组a中的每个元素值是:
2  4  6
经过方法后,Main方法中数组a的每个元素值是:
2  4  6  请按任意键继续. . .
      通过这个实例,同学们应该看出引用类型作为参数时,也会影响主方法中带入到调用方法的值。下面我们来学习最后一直参数类型叫做动态传参。
动态传参

 
      动态传参主要是指参数的个数可以根据需要改变,使用动态传参要用到parmas关键字,它的具体用法我们来看这个例子:
动态传参实例1
InBlock.gif 1  class Class2 
InBlock.gif 2        {    
InBlock.gif 3                 //定义一个静态的方法,参数是引用类型的数组参数a,采用动态传参方式, 
InBlock.gif 4                 //方法内部的功能是,改变数值a中每个元素的值,让改变后的值是原来的2倍 
InBlock.gif 5                 static  void ArrayMethod( params  int [] a) 
InBlock.gif 6                { 
InBlock.gif 7                        Console.WriteLine( "数值参数传入时,数组中的每个元素值是:"); 
InBlock.gif 8                         foreach ( int i  in a) 
InBlock.gif 9                        { 
InBlock.gif10                                Console.Write(i+ "    "); 
InBlock.gif11                        } 
InBlock.gif12                        Console.WriteLine(); 
InBlock.gif13                         //改变数值a中每个元素的值,让改变后的值是原来的2倍 
InBlock.gif14                         for ( int j = 0; j < a.Length; j++) 
InBlock.gif15                        { 
InBlock.gif16                                a[j] = a[j] * 2; 
InBlock.gif17                        } 
InBlock.gif18                        Console.WriteLine( "改变后,方法内部的数组a中的每个元素值是:"); 
InBlock.gif19                         foreach ( int i  in a) 
InBlock.gif20                        { 
InBlock.gif21                                Console.Write(i +  "    "); 
InBlock.gif22                        } 
InBlock.gif23                        Console.WriteLine(); 
InBlock.gif24                } 
InBlock.gif25             
InBlock.gif26                 static  void Main() 
InBlock.gif27                { 
InBlock.gif28                         //使用了动态传参,我们可以不规定数组的大小,按照需要进行传入。 
InBlock.gif29                        ArrayMethod(1,2,3,4); 
InBlock.gif30 
InBlock.gif31                        /** /////给数值a赋初值 
InBlock.gif32                         //int[] a = new int[] { 1, 2, 3 }; 
InBlock.gif33 
InBlock.gif34                        /** /////调用方法将引用类型的数组参数a带入方法中 
InBlock.gif35                         //ArrayMethod(a); 
InBlock.gif36 
InBlock.gif37                         //经过方法后引用类型的数组参数a的元素值应该发生改变, 
InBlock.gif38                         //Console.WriteLine("经过方法后,Main方法中数组a的每个元素值是:"); 
InBlock.gif39                         //foreach (int i in a) 
InBlock.gif40                         //{ 
InBlock.gif41                         //        Console.Write(i + "    "); 
InBlock.gif42                         //} 
InBlock.gif43                }        
InBlock.gif44        }
 对照一下上个例子和本例,我们会发现,动态传参,不必规定数组的个数,而是根据需要使用,本例结果如下:
数值参数传入时,数组中的每个元素值是:
1  2  3  4
改变后,方法内部的数组a中的每个元素值是:
2  4  6  8
请按任意键继续. . .
 
    现在我们来做下一个例子,就是调用方法中有两个参数,一个int型,一个动态的数组类型parmas int[],要告诉大家的是,如果使用parma动态参数,动态参数一定只有一个,必须当有多个参数时,动态参数的位置在最后,我们把上面的例子修改一下,大家体会一下不同。
动态传参实例2
InBlock.gif 1class Class2 
InBlock.gif 2        {    
InBlock.gif 3                 //定义一个静态的方法,参数是引用类型的数组参数a,采用动态传参方式, 
InBlock.gif 4                 //方法内部的功能是,改变数值a中每个元素的值,让改变后的值是原来的2倍 
InBlock.gif 5                 static  void ArrayMethod( int k, params  int [] a) 
InBlock.gif 6                {    
InBlock.gif 7                        
InBlock.gif 8                        Console.WriteLine( "数值参数传入时,数组中的每个元素值是:"); 
InBlock.gif 9                         foreach ( int i  in a) 
InBlock.gif10                        { 
InBlock.gif11                                Console.Write(i+ "    "); 
InBlock.gif12                        } 
InBlock.gif13                        Console.WriteLine( "k="+k); 
InBlock.gif14                        Console.WriteLine(); 
InBlock.gif15                         //改变数值a中每个元素的值,让改变后的值是原来的2倍 
InBlock.gif16                         for ( int j = 0; j < a.Length; j++) 
InBlock.gif17                        { 
InBlock.gif18                                a[j] = a[j] * 2; 
InBlock.gif19                        } 
InBlock.gif20                        Console.WriteLine( "改变后,方法内部的数组a中的每个元素值是:"); 
InBlock.gif21                         foreach ( int i  in a) 
InBlock.gif22                        { 
InBlock.gif23                                Console.Write(i +  "    "); 
InBlock.gif24                        } 
InBlock.gif25                        Console.WriteLine( "k="+ k); 
InBlock.gif26                        Console.WriteLine(); 
InBlock.gif27                } 
InBlock.gif28             
InBlock.gif29                 static  void Main() 
InBlock.gif30                { 
InBlock.gif31                         //使用了动态传参,我们可以不规定数组的大小,按照需要进行传入。 
InBlock.gif32                         //如果调用方法把parma参数写在前面,因为它的个数是不固定的, 
InBlock.gif33                         //她会把最后一个值,也当成是数组中元素进行操作。 
InBlock.gif34                        ArrayMethod(1,2,3,4); 
InBlock.gif35 
InBlock.gif36                     
InBlock.gif37                }        
InBlock.gif38        }
 
运行结果
数值参数传入时,数组中的每个元素值是:
2  3  4  k=1
改变后,方法内部的数组a中的每个元素值是:
4  6  8  k=1
请按任意键继续. . .
       可以看出因为把parmas的动态参数放到了最后,编译器就自动的把第一个值1直接给了k,把剩下的2,3,4给了数值a参数带入到方法中,所以使用parmas一定要注意到这个问题。
       本节课我们就学习到这里,下一节我们要来学习一下类四:构造方法、析构方法
本文转自叶子文文博客51CTO博客,原文链接http://blog.51cto.com/leafwf/185723如需转载请自行联系原作者

叶子文文
相关文章
|
1月前
|
Java 调度 C#
C#学习系列相关之多线程(一)----常用多线程方法总结
C#学习系列相关之多线程(一)----常用多线程方法总结
|
1月前
|
C#
C#学习相关系列之数组---常用方法使用(二)
C#学习相关系列之数组---常用方法使用(二)
|
1月前
|
C#
C#系列之ref和out的区别
C#系列之ref和out的区别
|
1月前
|
存储 C# 数据库
C# 生成唯一ID,有哪些方法?
【2月更文挑战第12天】
162 0
|
3月前
|
存储 编解码 开发工具
Baumer工业相机堡盟工业相机如何通过NEOAPI SDK使用UserSet功能保存和载入相机的各类参数(C#)
Baumer工业相机堡盟工业相机如何通过NEOAPI SDK使用UserSet功能保存和载入相机的各类参数(C#)
37 0
|
30天前
|
开发框架 小程序 .NET
C#动态生成带参数的小程序二维码
C#动态生成带参数的小程序二维码
|
1月前
|
C#
C#学习相关系列之数据类型类----嵌套类和嵌套方法(三)
C#学习相关系列之数据类型类----嵌套类和嵌套方法(三)
|
1月前
|
开发框架 .NET C#
C#学习相关系列之Linq常用方法---排序(一)
C#学习相关系列之Linq常用方法---排序(一)
|
1月前
|
开发框架 .NET 编译器
C#学习相关系列之匿名方法和Lambda表达式
C#学习相关系列之匿名方法和Lambda表达式
|
1月前
|
C#
C#中保留小数点后N位的方法_kaic
C#中保留小数点后N位的方法_kaic