Java面向对象(二)

打印 上一主题 下一主题

主题 842|帖子 842|积分 2526

Java面向对象(二)


目录

五、方法

5.1 方法的重载(overload)


  • 定义:在同一个类中,允许定义多个相同名字的方法,只要参数列表(参数类型或者参数个数)是不同的
  • 判断是否为方法重载:
    (1)同一个类,同样的方法名,不同的参数列表
    (2)与方法修饰符、方法返回值、形参名、方法体无关
    1. public class OverloadTest{
    2.        
    3.     public void getSum(int a,int b){}
    4.    
    5.     public void getSum(int a,int b,int c){}
    6.     public void getSum(double a,double b){}
    7.    
    8.     public void getSum(int a,double b){}
    9.    
    10.     public void getSum(double a,int b){}
    11.    
    12. /*  不属于方法重载:与方法返回值类型无关  
    13.         public int getSum(int a,int b){      
    14.         return 0;
    15.     }
    16. */
    17. /*  不属于方法重载:与形参名称无关
    18.         public void getSum(int m,int n){}
    19. */
    20.    
    21. /*  不属于方法重载:与方法修饰符无关   
    22.         private void getSum(int a,int b){}
    23. */
    24.    
    25. }
    复制代码
  • 方法重载中一些特殊情况:
情况一:
  1. public class OverloadTest {
  2.     public static void main(String[] args){
  3.         OverloadTest o = new OverloadTest();
  4.         
  5.         System.out.println(o.getSum(1,2));                // 输出3
  6.         System.out.println(o.getSum(1.0,2.0));        // 输出3.0
  7.         System.out.println(o.getSum(1.0,2));        // 输出3.0
  8.     }
  9.     public int getSum(int a,int b){
  10.         return a + b;
  11.     }
  12.     public double getSum(double a,double b){
  13.         return a + b;
  14.     }
  15. }
  16. /*         可以看到,我们并没有定义 getSum(double,int)类型的方法,但是 getSum(1.0,2)还是正常输出,没有报错。
  17.         这里getSum(1.0,2)是调用了getSum(double a,double b)的方法,原因是实参2被自动转换为double类型并传递给这个方法。
  18.        
  19.         那为什么getSum(1,2)调用时不会使用getSum(double a,double b)方法?
  20.         原因:
  21.         其实 getSum(int a,int b) 和 getSum(double a,double b) 都是 getSum(1.0,2) 可能的匹配。调用方法时,Java编译器会查找最精确匹配的方法,因为方法 getSum(int a,int b) 比 getSum(double a,double b) 更精确,所以调用 getSum(1,2) 时使用的是getSum(int a,int b)。
  22.        
  23. */
复制代码
情况二:
  1. public class OverloadTest {
  2.     public static void main(String[] args){
  3.         OverloadTest o = new OverloadTest();
  4.         System.out.println(o.getSum(1,2));
  5.     }
  6.     public double getSum(int a,double b){
  7.         return 1.0;
  8.     }
  9.     public double getSum(double a,double b){
  10.         return 2.0;
  11.     }
  12. }
  13. /*         输出 1.0
  14.         根据调用方法时,Java编译器会查找最精确匹配的方法。
  15.         所以 getSum(1,2)调用时使用的是 public double getSum(int a,double b)方法
  16. */
复制代码
情况三:
有时调用一个方法时会有两个或多个可能的匹配,但编译器无法判断哪个是最精确的匹配。这就称为“歧义调用”(ambiguous invocation),会产生编译错误。
  1. public class OverloadTest {
  2.     public static void main(String[] args){
  3.         OverloadTest o = new OverloadTest();
  4.         System.out.println(o.getSum(1,2));
  5.     }
  6.     public double getSum(int a,double b){
  7.         return 1.0;
  8.     }
  9.     public double getSum(double a,int b){
  10.         return 2.0;
  11.     }
  12. }
  13. /*        编译报错:
  14.         Error:(9, 29) java: 对getSum的引用不明确
  15.         javase.OverloadTest 中的方法 getSum(int,double) 和 javase.OverloadTest 中的方法 getSum(double,int) 都匹配       
  16. */
复制代码
5.2 可变参数的形参


  • JavaSE 5.0 中提供了 Varargs (variable number of arguments)机制,允许直接定义能和多个实参相匹配的形参。从而,可以用一种更简单的方式,来传递个数可变的实参。
  1. //JDK 5.0以前:采用数组形参来定义方法,传入多个同一类型变量
  2. public class Test {
  3.     public static void main(String[] args){
  4.         Test o = new Test();
  5.         o.test(new String[]{"aa","bb","cc"});        //调用时必须new String[]数组
  6.     }
  7.     public static void test(String[] books){
  8.     for (int i = 0; i < books.length; i++) {
  9.             System.out.println(books[i]);
  10.         }
  11.     }
  12. }
  13. //JDK5.0:采用可变个数形参来定义方法,传入多个同一类型变量
  14. public class Test {
  15.     public static void main(String[] args){
  16.         Test o = new Test();
  17.         o.test(new String[]{"aa","bb","cc"});        //同样可以用
  18.         o.test("aa","bb","cc");                                        //更加简单
  19.     }
  20.     public static void test(String ... books){
  21.         for (int i = 0; i < books.length; i++) {
  22.             System.out.println(books[i]);
  23.         }
  24.     }
  25. }
  26. /*        输出:
  27.         aa
  28.         bb
  29.         cc
  30. */
复制代码

  • 说明:

    • 声明格式:方法名(参数的类型名 ... 参数名) 。
    • 可变参数:方法参数部分指定类型的参数个数是可变多个:0个,1个或多个
    • 可变个数形参的方法与同名的方法之间,彼此构成重载
    • 可变参数方法的使用与方法参数部分使用数组是一致的故两个不能构成方法重载
    • 方法的参数部分有可变形参,需要放在形参声明的最后
      1. public static void test(int a ,String ... books);
      2. /*        报错,Java编译器不知道可变参数什么时候结束
      3.         public static void test(String ... books,int a);
      4. */
      复制代码
    • 在一个方法的形参位置,最多只能声明一个可变个数形参

5.3 方法参数的值传递机制


  • Java 里方法的参数传递方式只有一种:值传递。 即将实际参数值的副本(复制品)传入方法内,而参数本身不受影响

    • 形参是基本数据类型:将实参基本数据类型变量的“数据值”传递给形参
    • 形参是引用数据类型:将实参引用数据类型变量的“地址值”传递给形参

  • 例子:
  1. //        针对基本数据类型:
  2. public class Test {
  3.     public static void main(String[] args) {
  4.         int num1 = 1;
  5.         int num2 = 2;
  6.         System.out.println("在调用 swap 方法前,变量 num1 的值为:" + num1 + ",变量 num2 的值为:" + num2);
  7.         swap(num1, num2);
  8.         System.out.println("在调用 swap 方法后,变量 num1 的值为:" + num1 + ",变量 num2 的值为:" + num2);
  9.     }
  10.     public static void swap(int n1, int n2) {
  11.         System.out.println("在进行交换前,变量 n1 的值为:" + n1 + ",变量 n2 的值为:" + n2);
  12.         int temp = n1;
  13.         n1 = n2;
  14.         n2 = temp;
  15.         System.out.println("在进行交换后,变量 n1 的值为:" + n1 + ",变量 n2 的值为:" + n2);
  16.     }
  17.    
  18. }
  19. /*        输出结果:
  20.         在调用 swap 方法前,变量 num1 的值为:1,变量 num2 的值为:2
  21.         在进行交换前,变量 n1 的值为:1,变量 n2 的值为:2
  22.         在进行交换后,变量 n1 的值为:2,变量 n2 的值为:1
  23.         在调用 swap 方法后,变量 num1 的值为:1,变量 num2 的值为:2
  24.         可以看到,num1 和 num2 的值并没有因为调用 swap 方法而进行交换。
  25. */
复制代码
Java 内存分析:
  1. //        针对引用数据类型:
  2. public class Test {
  3.     public static void main(String[] args) {
  4.         Data data = new Data();
  5.         data.m = 10;
  6.         data.n = 20;
  7.         System.out.println("m = " + data.m + ", n = " + data.n);
  8.         Test test = new Test();
  9.         test.swap(data);
  10.         System.out.println("m = " + data.m + ", n = " + data.n);
  11.     }
  12.     public void swap(Data d){
  13.         int temp = d.m;
  14.         d.m = d.n;
  15.         d.n = temp;
  16.     }
  17. }
  18. class Data{
  19.     int m;
  20.     int n;
  21. }
复制代码
Java内存分析:

所以,在写交换数组元素位置方法时:
  1. //错误的:交换数组中指定两个位置元素的值
  2. //        public void swap(int i,int j){
  3. //                int temp = i;
  4. //                i = j;
  5. //                j = temp;
  6. //        }
  7.         //正确的:交换数组中指定两个位置元素的值
  8.         public void swap(int[] arr,int i,int j){
  9.                 int temp = arr[i];
  10.                 arr[i] = arr[j];
  11.                 arr[j] = temp;
  12.         }
复制代码

  • 特殊的,String类型
  1. public class Test {
  2.         public static void main(String[] args) {
  3.                
  4.                 String s1 = "hello";
  5.                
  6.                 Test test = new Test();
  7.                 test.change(s1);
  8.                
  9.                 System.out.println(s1);       
  10.                
  11.         }
  12.        
  13.         public void change(String s){
  14.                 s = "hi";
  15.         }
  16. }
  17. //        输出:hello
复制代码
Java内存分析:

结论:String 属于引用类型变量,形参仍然接收的是实参变量的地址值,只是字符串在常量池是一个 char[] 型的一个字符数组,新赋值 hi 时会在常量池新创建一个字符数组,再将 hi 字符串的地址赋值给 s 变量。
5.4 递归方法


  • 递归方法:一个方法体内调用它自身。

    • 方法递归包含了一种隐式的循环,它会重复执行某段代码,但这种重复执行无须循环控制。
    • 递归一定要向已知方向递归,否则这种递归就变成了无穷递归,类似于死循环。

  • 典型使用场景:

    • 求 n 的阶乘
      1. //        求 n!方法
      2. public int f(int n) {
      3.         if (n == 1) {
      4.                 return 1;
      5.         } else {
      6.                 return n * f(n - 1);
      7.         }
      8. }
      复制代码
    • 斐波那契数列
      1. public int f(int n) {
      2.         if (n == 1 || n == 2) {
      3.                 return 1;
      4.         }
      5.     if (n > 2) {
      6.         return f(n-1) + f(n-2);
      7.     }
      8.     return -1;
      9. }
      复制代码
    • 快速排序


免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

冬雨财经

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表