你能用Java编写虚函数/方法吗?

yon*_*tan 150 java virtual virtual-functions

是否有可能在Java中编写虚拟方法,就像在C++中一样?

或者,是否有适当的Java方法可以实现,产生类似的行为?我能举一些例子吗?

Kla*_*sen 283

来自维基百科

Java中,默认情况下,所有非静态方法都是" 虚函数".只有标记为关键字final的方法(不能被覆盖)以及 未被继承的私有方法才是非虚方法.

  • [这里](http://stackoverflow.com/a/295248/689991)是[Jon Skeet]之一(http://stackoverflow.com/users/22656/jon-skeet)的回答. (3认同)

Eri*_*ski 92

你能用Java编写虚函数吗?

是.实际上,Java中的所有实例方法都是默认的虚拟方法.只有某些方法不是虚拟的:

  • 类方法(因为通常每个实例都保存信息,如关于其特定方法的vtable指针,但此处没有实例可用).
  • 私有实例方法(因为没有其他类可以访问该方法,调用实例始终是定义类本身的类型,因此在编译时明确知道).

这里有些例子:

"正常"虚拟功能

以下示例来自另一个答案中提到的维基百科页面的旧版本.

import java.util.*;

public class Animal 
{
   public void eat() 
   { 
      System.out.println("I eat like a generic Animal."); 
   }

   public static void main(String[] args) 
   {
      List<Animal> animals = new LinkedList<Animal>();

      animals.add(new Animal());
      animals.add(new Fish());
      animals.add(new Goldfish());
      animals.add(new OtherAnimal());

      for (Animal currentAnimal : animals) 
      {
         currentAnimal.eat();
      }
   }
}

class Fish extends Animal 
{
   @Override
   public void eat() 
   { 
      System.out.println("I eat like a fish!"); 
   }
}

class Goldfish extends Fish 
{
   @Override
   public void eat() 
   { 
      System.out.println("I eat like a goldfish!"); 
   }
}

class OtherAnimal extends Animal {}
Run Code Online (Sandbox Code Playgroud)

输出:

I eat like a generic Animal.
I eat like a fish!
I eat like a goldfish!
I eat like a generic Animal.

带接口的示例

Java 接口方法都是虚拟的.它们必须是虚拟的,因为它们依赖于实现类来提供方法实现.要执行的代码只能在运行时选择.

例如:

interface Bicycle {         //the function applyBrakes() is virtual because
    void applyBrakes();     //functions in interfaces are designed to be 
}                           //overridden.

class ACMEBicycle implements Bicycle {
    public void applyBrakes(){               //Here we implement applyBrakes()
       System.out.println("Brakes applied"); //function
    }
}
Run Code Online (Sandbox Code Playgroud)

带有抽象类的虚函数示例.

与接口类似抽象类 必须包含虚方法,因为它们依赖于扩展类的实现.例如:

abstract class Dog {                   
    final void bark() {               //bark() is not virtual because it is 
        System.out.println("woof");   //final and if you tried to override it
    }                                 //you would get a compile time error.

    abstract void jump();             //jump() is a "pure" virtual function 
}                                     
class MyDog extends Dog{
    void jump(){
        System.out.println("boing");    //here jump() is being overridden
    }                                  
}
public class Runner {
    public static void main(String[] args) {
        Dog dog = new MyDog();       // Create a MyDog and assign to plain Dog variable
        dog.jump();                  // calling the virtual function.
                                     // MyDog.jump() will be executed 
                                     // although the variable is just a plain Dog.
    }
}
Run Code Online (Sandbox Code Playgroud)

  • 这应该是最完整的答案了。由于java没有关键字,因此它提供了两种实现虚函数的方法。谢谢。 (2认同)

duf*_*ymo 54

Java中的所有函数都是默认的虚拟.

您必须通过添加"final"关键字来编写非虚函数.

这与C++/C#默认相反.默认情况下,类函数是非虚拟的; 你可以通过添加"虚拟"修饰符来实现它们.

  • Klaus的答案中所述的私人功能也是非虚拟的. (4认同)

And*_*mas 9

默认情况下,所有非私有实例方法在Java中都是虚拟的.

在C++中,私有方法可以是虚拟的.这可以用于非虚拟接口(NVI)习语.在Java中,您需要保护NVI可覆盖方法.

从Java语言规范,v3:

8.4.8.1重写(通过实例方法)在类C中声明的实例方法m1将覆盖在类A中声明的另一个实例方法m2,如果以下所有条件都为真:

  1. C是A的子类.
  2. m1的签名是m2签名的子签名(§8.4.2).
  3. *m2是公共的,受保护的或在与C相同的包中使用默认访问权限声明,或*m1覆盖方法m3,m3与m1不同,m3与m2不同,因此m3覆盖m2.