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

PHP教程|ASP.NET教程|JAVA教程|ASP教程|

服务器之家 - 编程语言 - JAVA教程 - 详解Java面向对象编程中方法的使用

详解Java面向对象编程中方法的使用

2020-04-02 13:29志在必得 JAVA教程

这篇文章主要介绍了详解Java面向对象编程中方法的使用,包括方法的重载和参数以及泛型方法等知识点,需要的朋友可以参考下

一个 Java 方法是为了执行某个操作的一些语句的组合。举个例子来说,当你调用 System.out.println 方法时,系统实际上会执行很多语句才能在控制台上输出信息。

现在你将学习怎么创建你自己的方法,他们可以有返回值也可以没有返回值,可以有参数,也可以没有参数,重载方法要使用相同的方法名称,并在程序设计中利用抽象的方法。

创建方法
我们用下面的例子来解释方法的语法:

?
1
2
3
public static int funcName(int a, int b) {
 // body
}

在这里

  • public static:修饰符
  • int:返回值类型
  • funcName:函数名称
  • a,b:形式参数
  • int a,int b:参数列

方法也包含过程或函数。

  • 过程:他们不返回值
  • 函数:他们返回值

方法的定义包含方法头和方法体。如下所示:

?
1
2
3
modifier returnType nameOfMethod (Parameter List) {
 // method body
}

以上的语法包括

  • modifier:他定义了方法的访问类型,它是可选的。
  • returnType:方法是可能返回一个值的。
  • nameOfMethod:这是方法的名称。方法签名包括方法名称和参数列表。
  • Parameter List:参数列表,它是参数的次序,类型,以及参数个数的集合。这些都是可选的,当然方法也可以没有参数。
  • 方法体:方法体定义了这个方法是用来做什么的。

示例

这是上面定义的方法max(),该方法接受两个参数num1和num2返回两者之间的最大值。

?
1
2
3
4
5
6
7
8
9
10
/** the snippet returns the minimum between two numbers */
public static int minFunction(int n1, int n2) {
  int min;
  if (n1 > n2)
   min = n2;
  else
   min = n1;
 
  return min;
}

方法调用
要想使用一个方法,该方法必须要被调用。方法调用有两种方式,一种是有返回值的,一种是没有返回值的。

调用方法很简单,当程序需要调用一个方法时,控制程序转移到被调用的方法,方法将会返回两个条件给调用者:

  • 返回一条执行语句
  • 执行到方法结束

将返回void的方法作为一个调用语句,让我看下面的例子:

?
1
System.out.println("wiki.jikexueyuan.com!");

该方法的返回值可以通过下面的例子被理解:

?
1
int result = sum(6, 9);

示例

下面的例子表明了怎么定义方法和怎么调用它:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
public class ExampleMinNumber{
 
  public static void main(String[] args) {
   int a = 11;
   int b = 6;
   int c = minFunction(a, b);
   System.out.println("Minimum Value = " + c);
  }
 
  /** returns the minimum of two numbers */
  public static int minFunction(int n1, int n2) {
   int min;
   if (n1 > n2)
     min = n2;
   else
     min = n1;
 
   return min;
  }
}

将会产生如下的结果

?
1
Minimum value = 6

关键字 void
关键字 void 允许我们创建一个没有返回值的方法。这里我们在下一个例子中创建一个 void 方法 methodRankPoints。这个方法是没有返回值类型的。调用 void 方法必须声明 methodRankPoints(255.7); Java 语句以分号结束,如下所示:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
public class ExampleVoid {
 
  public static void main(String[] args) {
   methodRankPoints(255.7);
  }
 
  public static void methodRankPoints(double points) {
   if (points >= 202.5) {
     System.out.println("Rank:A1");
   }
   else if (points >= 122.4) {
     System.out.println("Rank:A2");
   }
   else {
     System.out.println("Rank:A3");
   }
  }
}

这将产生如下的结果:

?
1
Rank:A1

通过值来传递参数
在调用函数时参数是必须被传递的。并且他们的次序必须和他们创建时的参数次序是一样的。参数可以通过值或引用来传递。

通过值传递参数意味着调用方法的参数,通过参数值来传递给参数。

示例

下面的程序给出了一个例子来显示通过值来传递参数。在调用方法后参数值是不会发生变化的。

?
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
public class swappingExample {
 
  public static void main(String[] args) {
   int a = 30;
   int b = 45;
 
   System.out.println("Before swapping, a = " +
             a + " and b = " + b);
 
   // Invoke the swap method
   swapFunction(a, b);
   System.out.println("\n**Now, Before and After swapping values will be same here**:");
   System.out.println("After swapping, a = " +
             a + " and b is " + b);
  }
 
  public static void swapFunction(int a, int b) {
 
   System.out.println("Before swapping(Inside), a = " + a
              + " b = " + b);
   // Swap n1 with n2
   int c = a;
   a = b;
   b = c;
 
   System.out.println("After swapping(Inside), a = " + a
              + " b = " + b);
  }
}

这将产生如下的结果:

?
1
2
3
4
5
6
Before swapping, a = 30 and b = 45
Before swapping(Inside), a = 30 b = 45
After swapping(Inside), a = 45 b = 30
 
**Now, Before and After swapping values will be same here**:
After swapping, a = 30 and b is 45

方法的重载
当一个方法有两个或者更多的方法,他们的名字一样但是参数不同时,就叫做方法的重载。它与覆盖是不同的。覆盖是指方法具有相同的名字,类型以及参数的个数。

让我们来考虑之前的找最小整型数的例子。如果我们要求寻找浮点型中最小的数时,我们就需要利用方法的重载来去创建函数名相同,但参数不一样的两个或更多的方法。

下面的例子给予解释:

?
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
public class ExampleOverloading{
 
  public static void main(String[] args) {
   int a = 11;
   int b = 6;
   double c = 7.3;
   double d = 9.4;
   int result1 = minFunction(a, b);
   // same function name with different parameters
   double result2 = minFunction(c, d);
   System.out.println("Minimum Value = " + result1);
   System.out.println("Minimum Value = " + result2);
  }
 
 // for integer
  public static int minFunction(int n1, int n2) {
   int min;
   if (n1 > n2)
     min = n2;
   else
     min = n1;
 
   return min;
  }
  // for double
  public static double minFunction(double n1, double n2) {
   double min;
   if (n1 > n2)
     min = n2;
   else
     min = n1;
 
   return min;
  }
}

这将产生如下结果:

?
1
2
Minimum Value = 6
Minimum Value = 7.3

重载方法使程序易读。在这里,两种方法名称相同但参数不同。产生整型和浮点类型的最小数作为程序运行结果。

使用命令行参数
有时你想要在程序运行之前传递参数。这可以通过给 main 函数传递命令行参数来实现。

在命令行中,当要执行程序文件时,一个命令行参数是紧接着文件名字后面的出现的。要接受命令行参数在 Java 程序中是十分容易的。它们传递到 main 函数字符数组内。

示例

下面的例子展示了将所有命令行参数输出的程序:

?
1
2
3
4
5
6
7
8
9
public class CommandLine {
 
  public static void main(String args[]){
   for(int i=0; i<args.length; i++){
     System.out.println("args[" + i + "]: " +
                      args[i]);
   }
  }
}

通过以下方法来执行该程序:

?
1
java CommandLine this is a command line 200 -100

这将产生如下的结果:

?
1
2
3
4
5
6
7
args[0]: this
args[1]: is
args[2]: a
args[3]: command
args[4]: line
args[5]: 200
args[6]: -100

构造函数
这是一个简单的使用构造函数的例子:

?
1
2
3
4
5
6
7
8
9
// A simple constructor.
class MyClass {
  int x;
 
  // Following is the constructor
  MyClass() {
   x = 10;
  }
}

你可以通过以下方法来调用构造函数来实例化一个对象:

?
1
2
3
4
5
6
7
8
public class ConsDemo {
 
  public static void main(String args[]) {
   MyClass t1 = new MyClass();
   MyClass t2 = new MyClass();
   System.out.println(t1.x + " " + t2.x);
  }
}

通常,你将需要用构造函数来接受一个或多个参数。参数的传递和以上介绍的普通方法的参数传递是一样的,就是在构造函数的名字后面列出参数列表。

示例

这是一个简单的使用构造函数的例子:

?
1
2
3
4
5
6
7
8
9
// A simple constructor.
class MyClass {
  int x;
 
  // Following is the constructor
  MyClass(int i ) {
   x = i;
  }
}

你可以通过以下方法来调用构造函数来实例化一个对象:

?
1
2
3
4
5
6
7
8
public class ConsDemo {
 
  public static void main(String args[]) {
   MyClass t1 = new MyClass( 10 );
   MyClass t2 = new MyClass( 20 );
   System.out.println(t1.x + " " + t2.x);
  }
}

这将产生如下的结果:

?
1
10 20

可变长参数
JDK1.5 能够允许你传递可变长的同一类型的参数。用如下方法进行声明:

?
1
typeName... parameterName

方法声明时,你要在省略号前明确参数类型,并且只能有一个可变长参数,并且可变长参数必须是所有参数的最后一个。

示例

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
public class VarargsDemo {
 
  public static void main(String args[]) {
   // Call method with variable args
   printMax(34, 3, 3, 2, 56.5);
   printMax(new double[]{1, 2, 3});
  }
 
  public static void printMax( double... numbers) {
  if (numbers.length == 0) {
   System.out.println("No argument passed");
   return;
  }
 
  double result = numbers[0];
 
  for (int i = 1; i < numbers.length; i++)
   if (numbers[i] > result)
   result = numbers[i];
   System.out.println("The max value is " + result);
  }
}

这将产生如下的结果:

?
1
2
The max value is 56.5
The max value is 3.0

finalize() 方法
你可以定义一个方法,仅在被垃圾收集器销毁之前才会被调用。这个方法叫做 finalize() 方法,它也可以用来确保一个对象被干净清除了。

举个例子,你也许用 finalize() 来确保被一个对象打开的文件已经关闭了。

为了给类添加一个终结器,你只需定义 finalize() 方法。Java要回收该类的一个对象时,会调用该方法。

在 finalize() 方法中,你将指定一些必须在对象销毁之前要做的行为。

finalize()方法一般是如下形似:

?
1
2
3
4
protected void finalize( )
{
  // finalization code here
}

这里,关键字 protected 是为了保证在类外的代码不能访问 finalize() 方法。

这意味着你不能知道 finalize() 什么时候执行。举个例子,如果你的程序在垃圾收集器发生之前就结束了,finalize() 方法将不会被执行。

泛型方法:
java泛型方法在方法返回值是容器类对象时广泛使用。

?
1
2
3
public static List<T> find(Class<T> clazz,String userId){
   ....
}

一般来说编写java泛型方法时,返回值类型和至少一个参数类型应该是泛型,而且类型应该是一致的,如果只有返回值类型或参数类型之一使用了泛型,这个泛型方法的使用就大大的限制了,基本限制到跟不用泛型一样的程度。

下面主要介绍两种十分相似的java泛型方法的使用以及它们之间的区别。
第一种:

?
1
2
3
4
5
6
7
8
public static <T extends CommonService> T getService(Class<T> clazz) {
  T service = (T) serviceMap.get(clazz.getName());
  if (service == null) {
    service = (T) ServiceLocator.getService(clazz.getName());
    serviceMap.put(clazz.getName(), service);
  }
  return service;
}

第二种:

?
1
2
3
4
5
6
7
8
public static <T> T getService(Class<? extends CommonService> clazz) {
  T service = (T) serviceMap.get(clazz.getName());
  if (service == null) {
    service = (T) ServiceLocator.getService(clazz.getName());
    serviceMap.put(clazz.getName(), service);
  }
  return service;
}

下面是泛型方法所在的类:

?
1
2
3
4
5
6
7
public abstract class CommonService {
  private static HashMap<String, CommonService> serviceMap = new HashMap<String, CommonService>();
  //这里是泛型方法定义
  .
  .
  .
}

这两个泛型方法只有方法的签名不一样,方法体完全相同,那它们有什么不一样呢?
我们来使用一下它们,就知道它们的区别了。
对第一种泛型方法使用:

?
1
2
3
4
5
6
7
public class Main {
  public static void main(String[] args) {
    NoticeService noticeService=CommonService.getService(NoticeService.class);//正确的使用第一种泛型方法,不会出现编译错误。
    NoticeService noticeService=CommonService.getService(UserService.class);//不正确的使用第一种泛型方法,会出现编译错误。
    
  }
}

对第二种泛型方法使用:

?
1
2
3
4
5
6
7
public class Main {
  public static void main(String[] args) {
    NoticeService noticeService=CommonService.getService(NoticeService.class);//正确的使用第二种泛型方法,不会出现编译错误,逻辑也正确,运行时不会出现异常。
    NoticeService noticeService=CommonService.getService(UserService.class);//不正确的使用第二种泛型方法,不会出现编译错误,但逻辑不正确,运行时会出现异常,危险!
 
  }
}

现在知道了这两种极其相似的泛型方法的区别了吧?

  • 第一种泛型方法:返回值和参数值的类型是一致,推荐使用;
  • 第二种泛型方法:返回值和参数值的类型不是一致,请谨慎或避免使用。

延伸 · 阅读

精彩推荐