阿里二面:挂在main方法继承上?

开发 后端
我们一开始学习Java程序的时候,最先跑的一段代码肯定是main方法,那么main方法有什么特殊的地方呢?我们来简单看一下。

[[442552]]

问题一:包装类的缓存还记得不?

我们来看一下包装类相关的比较,看下下面的代码,最终将打印什么呢?

 

  1. public static void main(String[] args) { 
  2.     Boolean bool1 = true, bool2 = true
  3.     System.out.println("bool1==bool2 ? " + (bool1 == bool2) ); 
  4.  
  5.     Character c1 = 127, c2 = 127; 
  6.     Character c3 = 128, c4 = 128; 
  7.     System.out.println("c1==c2 ? " + (c1 == c2)); 
  8.     System.out.println("c3==c4 ? " + (c3 == c4) ); 
  9.  
  10.     Integer i1 = 10, i2 = 10; 
  11.     Integer i3 = 300, i4 = 300; 
  12.     System.out.println("i1==i2 ? " + (i1 == i2)); 
  13.     System.out.println("i3==i4 ? " + (i3 == i4) ); 
  14.  
  15.     Long long1 = 10L, long2 = 10L; 
  16.     Long long3 = 300L, long4 = 300L; 
  17.     System.out.println("long1==long2 ? " + (long1 == long2)); 
  18.     System.out.println("long3==long4 ? " + (long3 == long4)); 
  19.  
  20.     Float float1 = 10f, float2 = 10f; 
  21.     Float float3 = 300f, float4 = 300f; 
  22.     System.out.println("float1==float2 ? " + (float1 == float2)); 
  23.     System.out.println("float3==float4 ? " + (float3 == float4)); 

代码很简单,就是对各个包装类的几个值进行比较,可以猜测下这段代码的打印结果。这里我们直接将打印结果贴出来:

 

  1. public static void main(String[] args) { 
  2.     Boolean bool1 = true, bool2 = true
  3.     System.out.println("bool1==bool2 ? " + (bool1 == bool2) ); 
  4.  
  5.     Character c1 = 127, c2 = 127; 
  6.     Character c3 = 128, c4 = 128; 
  7.     System.out.println("c1==c2 ? " + (c1 == c2)); 
  8.     System.out.println("c3==c4 ? " + (c3 == c4) ); 
  9.  
  10.     Integer i1 = 10, i2 = 10; 
  11.     Integer i3 = 300, i4 = 300; 
  12.     System.out.println("i1==i2 ? " + (i1 == i2)); 
  13.     System.out.println("i3==i4 ? " + (i3 == i4) ); 
  14.  
  15.     Long long1 = 10L, long2 = 10L; 
  16.     Long long3 = 300L, long4 = 300L; 
  17.     System.out.println("long1==long2 ? " + (long1 == long2)); 
  18.     System.out.println("long3==long4 ? " + (long3 == long4)); 
  19.  
  20.     Float float1 = 10f, float2 = 10f; 
  21.     Float float3 = 300f, float4 = 300f; 
  22.     System.out.println("float1==float2 ? " + (float1 == float2)); 
  23.     System.out.println("float3==float4 ? " + (float3 == float4)); 

如果和你的预期结果是一致的,那么说明你这里掌握的很好,而如果和你的预期结果有稍稍不同的话,那么或许你可以再接着往下看。

解惑1:

首先,我们学习String的时候,都知道String类是不可变的,因此编译阶段会将String常量放入字符串常量池中,当下次使用时就可以直接从字符串常量池中提取。

而对于包装类来说,其对象同样也是不可变的。所以对于某些频繁使用的值,系统也提供了包装类的缓存,当需要时直接从缓存中取值,而不是再创建一个新的包装类对象。

这些包装类缓存的范围如下:

  • boolean的所有值(true和false);
  • char值的0~127;
  • byte,short,int,long的-128~127;

以Long为例,我们来简单看一下源码:

  1. private static class LongCache { 
  2.     private LongCache(){} 
  3.  
  4.     static final Long cache[] = new Long[-(-128) + 127 + 1]; 
  5.  
  6.     static { 
  7.         for(int i = 0; i < cache.length; i++) 
  8.             cache[i] = new Long(i - 128); 
  9.     } 

这个LongCache就是Long中缓存的实现,其他的也是类似,如IntegerCache,CharacterCache等,这里比较有意思的是:

 

  1. static final Long cache[] = new Long[-(-128) + 127 + 1]; 

这里数组的容量,使用了-(-128) + 127 + 1,这个我觉得写的挺有意思的,相当于是间接标识出了数组元素对应的范围:

-(-128) 表示负数的元素是128个;127 则表示正数的元素是127个,1 则表示元素0的个数;

这里我们在写代码的时候可以参考下,而其他的Cache的实现则是类似的,大家有兴趣的可以扒下代码看看。

对于浮点类型Float和Double,包装类没有缓存。

问题二:main方法有什么特殊的呢

我们一开始学习Java程序的时候,最先跑的一段代码肯定是main方法,main方法的格式如下:

 

  1. public static void main(String[] args)  

那么main方法有什么特殊的地方呢?我们来简单看一下。

解惑2:

首先针对main方法的格式定义:

  • public:main方法是启动的时候由JVM进行加载的,public的可访问权限是最高的,所以需要声明为public;
  • static:方法的调用要么是通过对象,要么是通过类,而main方法的话因为是由虚拟机调用的,所以无需生成对象,那么声明为static即可;
  • main:至于为什么方法名称叫main,我想应该是参考的是C语言的方法名吧;
  • void:main方法退出时,并没有需要有相关返回值需要返回,所以是void;
  • String[]:此字符串数组用来运行时接受用户输入的参数;因为字符串在Java中是具有通用普遍性的,所以使用字符串是最优选择;数组的话,因为我们的参数不止一个,所以数组肯定是合适的;

不过自动JDK1.5引入动态参数后,String[]数组也可以使用String... args来实现:

 

  1. public static void main(String... args) 

除了上面JVM规定的这个main方法比较特殊外,其他的main方法与普通的静态方法是没有什么不同的。

1. main方法能重载么?

这个是可以的,比如说我们给它重载一个方法:

 

  1. public class Main { 
  2.     public static void main(String args) { 
  3.         System.out.println("hello world:" + args); 
  4.     } 
  5.  
  6.     public static void main(String[] args) { 
  7.         main("test"); 
  8.     } 

编译运行,很显然没啥问题,除了JVM规定的作为应用程序入口的main方法之外,其他的main方法都是比较普通的方法。

2. main方法能被其他方法调用么?

 

  1. public class Main { 
  2.     private static int times = 3; 
  3.  
  4.     public static void main2(String[] args) { 
  5.         times--; 
  6.         main(args); 
  7.     } 
  8.  
  9.     public static void main(String[] args) { 
  10.         System.out.println("main方法执行:" + times); 
  11.         if (times <= 0) { 
  12.             System.exit(0); 
  13.         } 
  14.         main2(args); 
  15.     } 

运行一下代码,可以发现代码能正常执行:

 

  1. main方法执行:3 
  2. main方法执行:2 
  3. main方法执行:1 
  4. main方法执行:0 

所以说即使是作为应用程序入口的main方法,也是可以被其他方法调用的,但要注意程序的关闭方式,别陷入死循环了。

3. main方法可以继承么?

我们以前了解过,当类继承时,子类可以继承父类的方法和变量,那么当父类定义了main方法,而子类没有main方法时,能继承父类的main方法,从而正常的运行程序么?

 

  1. public class Main { 
  2.     public static void main(String[] args) { 
  3.         System.out.println("hello world"); 
  4.     } 

定义子类:

 

  1. public class Main2 extends Main { 

这时候我们运行子类Main2,可以发现,同样打印了hello world,这说明main方法也是可以继承的。那么还有一种隐藏的情况也很显然了,子类定义自己的main方法,隐藏掉父类中的实现,那么这也是可以的。

 

  1. public class Main2 extends Main { 
  2.     public static void main(String[] args) { 
  3.         System.out.println("hello world Main2"); 
  4.     } 

这时候就会打印子类自己的内容了:hello world Main2。

这么来看,除了main方法作为应用程序的入口比较特殊外,其他情况下与正常的静态方法是没什么区别的。 

责任编辑:庞桂玉 来源: java版web项目
相关推荐

2021-04-25 09:58:48

mmapJava面试

2021-03-17 15:54:32

IO零拷贝方式

2024-03-22 13:31:00

线程策略线程池

2021-10-27 20:54:24

分库分表高并发

2022-06-02 10:54:16

BrokerRocketMQ

2022-04-15 11:26:14

缓存功能

2022-10-18 08:38:16

内存泄漏线程

2023-10-30 01:02:56

Java类类加载器双亲委派

2022-09-05 16:55:23

RocketMQBroker

2021-09-18 08:54:19

zookeeper一致性算法CAP

2019-05-25 14:17:51

面试阿里社招

2019-12-03 14:09:49

阿里面试HR

2023-11-03 08:10:49

ThreadLoca内存泄露

2023-07-18 08:28:58

注册中心下线Nacos

2023-06-13 08:25:14

注册中心Nacos上线

2024-05-24 10:15:36

2022-03-14 11:05:01

RocketMQRedis缓存

2024-01-16 08:24:59

消息队列KafkaRocketMQ

2023-10-24 08:25:20

TCC模式事务

2021-06-17 09:16:34

MySQL数据库隔离级别
点赞
收藏

51CTO技术栈公众号