31道Java核心面试题,一次性打包送给你(3)

15、为什么 Java 不是纯粹的面向对象编程语言?


之所以不能说 Java 是纯粹的面向对象编程语言,是因为 Java 支持基本数据类型,比如说 int、short、long、double 等,尽管它们有自己的包装器类型,但它们的确不能算是对象。


16、path 和 classpath 之间有什么区别?


path 是操作系统用来查找可执行文件的环境变量,我的电脑上就定义了下图这些 path 变量,比如 Java 和 Maven 的。


31道Java核心面试题,一次性打包送给你(3)


classpath 是针对 Java 而言的,用于指定 Java 虚拟机载入的字节码文件路径。


17、Java 中 main() 方法的重要性是什么?


每个程序都需要一个入口,对于 Java 程序来说,入口就是 main 方法。


public static void main(String[] args) {}

1

public 关键字是另外一个访问修饰符,除了可以声明方法和变量(所有类可见),还可以声明类。main() 方法必须声明为 public。


static 关键字表示该变量或方法是静态变量或静态方法,可以直接通过类访问,不需要实例化对象来访问。


void 关键字用于指定方法没有返回值。


另外,main 关键字为方法的名字,Java 虚拟机在执行程序时会寻找这个标识符;args 为 main() 方法的参数名,它的类型为一个 String 数组,也就是说,在使用 java 命令执行程序的时候,可以给 main() 方法传递字符串数组作为参数。


java HelloWorld 沉默王二 沉默王三

1

javac 命令用来编译程序,java 命令用来执行程序,HelloWorld 为这段程序的类名,沉默王二和沉默王三为字符串数组,中间通过空格隔开,然后就可以在 main() 方法中通过 args[0] 和 args[1] 获取传递的参数值了。


public class HelloWorld {
    public static void main(String[] args) {
        if ("沉默王二".equals(args[0])) {
           
        }
       
        if ("沉默王三".equals(args[1])) {
           
        }
    }
}



main() 方法的写法并不是唯一的,还有其他几种变体,尽管它们可能并不常见,可以简单来了解一下。


第二种,把方括号 [] 往 args 靠近而不是 String 靠近:


public static void main(String []args) { }


第三种,把方括号 [] 放在 args 的右侧:


public static void main(String args[]) { }


第四种,还可以把数组形式换成可变参数的形式:


public static void main(String...args) { }


第五种,在 main() 方法上添加另外一个修饰符 strictfp,用于强调在处理浮点数时的兼容性:


public strictfp static void main(String[] args) { }


也可以在 main() 方法上添加 final 关键字或者 synchronized 关键字。


第六种,还可以为 args 参数添加 final 关键字:


public static void main(final String[] args) { }


第七种,最复杂的一种,所有可以添加的关键字统统添加上:


final static synchronized strictfp void main(final String[] args) { }


当然了,并不需要为了装逼特意把 main() 方法写成上面提到的这些形式,使用 IDE 提供的默认形式就可以了。


18、Java 的重写(Override)和重载(Overload)有什么区别?


先来看一段重写的代码吧。


class LaoWang{
    public void write() {
        System.out.println("老王写了一本《基督山伯爵》");
    }
}
public class XiaoWang extends LaoWang {
    @Override
    public void write() {
        System.out.println("小王写了一本《茶花女》");
    }
}



重写的两个方法名相同,方法参数的个数也相同;不过一个方法在父类中,另外一个在子类中。就好像父类 LaoWang 有一个 write() 方法(无参),方法体是写一本《基督山伯爵》;子类 XiaoWang 重写了父类的 write() 方法(无参),但方法体是写一本《茶花女》。


来写一段测试代码。


public class OverridingTest {

   public static void main(String[] args) {

       LaoWang wang = new XiaoWang();

       wang.write();

   }

}



大家猜结果是什么?


小王写了一本《茶花女》

1

在上面的代码中,们声明了一个类型为 LaoWang 的变量 wang。在编译期间,编译器会检查 LaoWang 类是否包含了 write() 方法,发现 LaoWang 类有,于是编译通过。在运行期间,new 了一个 XiaoWang 对象,并将其赋值给 wang,此时 Java 虚拟机知道 wang 引用的是 XiaoWang 对象,所以调用的是子类 XiaoWang 中的 write() 方法而不是父类 LaoWang 中的 write() 方法,因此输出结果为“小王写了一本《茶花女》”。


再来看一段重载的代码吧。


class LaoWang{

   public void read() {

       System.out.println("老王读了一本《Web全栈开发进阶之路》");

   }

   

   public void read(String bookname) {

       System.out.println("老王读了一本《" + bookname + "》");

   }

}


重载的两个方法名相同,但方法参数的个数不同,另外也不涉及到继承,两个方法在同一个类中。就好像类 LaoWang 有两个方法,名字都是 read(),但一个有参数(书名),另外一个没有(只能读写死的一本书)。


来写一段测试代码。


public class OverloadingTest {

   public static void main(String[] args) {

       LaoWang wang = new LaoWang();

       wang.read();

       wang.read("金瓶梅");

   }

}


这结果就不用猜了。变量 wang 的类型为 LaoWang,wang.read() 调用的是无参的 read() 方法,因此先输出“老王读了一本《Web全栈开发进阶之路》”;wang.read("金瓶") 调用的是有参的 read(bookname) 方法,因此后输出“老王读了一本《金瓶》”。在编译期间,编译器就知道这两个 read() 方法时不同的,因为它们的方法签名(=方法名称+方法参数)不同。


简单的来总结一下:


1)编译器无法决定调用哪个重写的方法,因为只从变量的类型上是无法做出判断的,要在运行时才能决定;但编译器可以明确地知道该调用哪个重载的方法,因为引用类型是确定的,参数个数决定了该调用哪个方法。


2)多态针对的是重写,而不是重载。


31道Java核心面试题,一次性打包送给你(3)


如果在一个类中有多个相同名字的方法,但参数不同,则称为方法重载。


父类中有一个方法,子类中有另外一个和它有相同签名(方法名相同,参数相同、修饰符相同)的方法时,则称为方法重写。子类在重写父类方法的时候可以加一个 @Override 注解。


上一篇:C++函数中return语句的使用方法


下一篇:rabbitmq系列(一)初识rabbitmq