* [一、关键字](#一关键字) * [final](#final) * [static](#static) * [二、Object 通用方法](#二object-通用方法) * [概览](#概览) * [clone()](#clone) * [equals()](#equals) * [四、继承](#四继承) * [访问权限](#访问权限) * [抽象类与接口](#抽象类与接口) * [super](#super) * [重载与重写](#重载与重写) * [五、String](#五string) * [String, StringBuffer and StringBuilder](#string,-stringbuffer-and-stringbuilder) * [String 不可变的原因](#string-不可变的原因) * [String.intern()](#stringintern) * [六、基本类型与运算](#六基本类型与运算) * [包装类型](#包装类型) * [switch](#switch) * [七、反射](#七反射) * [八、异常](#八异常) * [九、泛型](#九泛型) * [十、注解](#十注解) * [十一、特性](#十一特性) * [面向对象三大特性](#面向对象三大特性) * [Java 各版本的新特性](#java-各版本的新特性) * [Java 与 C++ 的区别](#java-与-c++-的区别) * [JRE or JDK](#jre-or-jdk) * [参考资料](#参考资料) # 一、关键字 ## final **1. 数据** 声明数据为常量,可以是编译时常量,也可以是在运行时被初始化后不能被改变的常量。 - 对于基本类型,final 使数值不变; - 对于引用类型,final 使引用不变,也就不能引用其它对象,但是被引用的对象本身是可以修改的。 ```java final int x = 1; x = 2; // cannot assign value to final variable 'x' final A y = new A(); y.a = 1; ``` **2. 方法**
声明方法不能被子类覆盖。 private 方法隐式地被指定为 final,如果在子类中定义的方法和基类中的一个 private 方法签名相同,此时子类的方法不是覆盖基类方法,而是重载了。 **3. 类** 声明类不允许被继承。 ## static **1. 静态变量** 静态变量在内存中只存在一份,只在类第一次实例化时初始化一次。 - 静态变量:类所有的实例都共享静态变量,可以直接通过类名来访问它; - 实例变量:每创建一个实例就会产生一个实例变量,它与该实例同生共死。 ```java public class A { private int x; // 实例变量 public static int y; // 静态变量 } ``` **2. 静态方法** 静态方法在类加载的时候就存在了,它不依赖于任何实例,所以 static 方法必须实现,也就是说它不能是抽象方法(abstract)。 **3. 静态语句块** 静态语句块和静态变量一样在类第一次实例化时运行一次。 **4. 初始化顺序** 静态数据优先于其它数据的初始化,静态变量和静态语句块哪个先运行取决于它们在代码中的顺序。 ```java public static String staticField = "静态变量"; ``` ```java static { System.out.println("静态语句块"); } ``` 实例变量和普通语句块的初始化在静态变量和静态语句块初始化结束之后。 ```java public String field = "实例变量"; ``` ```java { System.out.println("普通语句块"); } ``` 最后才是构造函数中的数据进行初始化 ```java public InitialOrderTest() { System.out.println("构造函数"); } ``` 存在继承的情况下,初始化顺序为: 1. 父类(静态变量、静态语句块块) 2. 子类(静态变量、静态语句块) 3. 父类(实例变量、普通语句块) 4. 父类(构造函数) 5. 子类(实例变量、普通语句块) 6. 子类(构造函数) # 二、Object 通用方法 ## 概览 ```java public final native Class getClass() public native int hashCode() public boolean equals(Object obj) protected native Object clone() throws CloneNotSupportedException public String toString() public final native void notify() public final native void notifyAll() public final native void wait(long timeout) throws InterruptedException public final void wait(long timeout, int nanos) throws InterruptedException public final void wait() throws InterruptedException protected void finalize() throws Throwable {} ``` ## clone() **1. cloneable** clone() 是 Object 的受保护方法,这意味着,如果一个类不显式去重载 clone() 就没有这个方法。 ```java public class CloneTest { private int a; private int b; } ``` ```java CloneTest x = new CloneTest(); CloneTest y = x.clone(); // 'clone()' has protected access in 'java.lang.Object' ``` 接下来重载 Object 的 clone() 得到以下实现: ```java public class CloneTest{ private int a; private int b; @Override protected Object clone() throws CloneNotSupportedException { return super.clone(); } } ``` ```java CloneTest x = new CloneTest(); try { CloneTest y = (CloneTest) x.clone(); } catch (CloneNotSupportedException e) { e.printStackTrace(); } ``` ```html java.lang.CloneNotSupportedException: CloneTest ``` 以上跑出了 CloneNotSupportedException,这是因为 CloneTest 没有实现 Cloneable 接口。应该注意的是,clone() 方法并不是 Cloneable 接口的方法,而是 Object 的一个 protect 方法。Cloneable 接口只是规定,如果一个类没有实现 Cloneable 接口又调用了 clone() 方法,就会抛出 CloneNotSupportedException。 **2. 深拷贝与浅拷贝** 可以使用序列化实现。

> [How do I copy an object in Java?](https://stackoverflow.com/questions/869033/how-do-i-copy-an-object-in-java) - 浅拷贝:引用类型引用同一个对象。 - 深拷贝:引用类型引用新的对象。

## equals() **1. == 与 equals() 区别**
- 对于基本类型,== 判断两个值是否相等; - 对于引用类型,== 判断两个引用是否引用同一个对象,而 equals() 判断引用的对象是否等价。 **2. 等价性**
> [散列](https://github.com/CyC2018/Interview-Notebook/blob/master/notes/Java%20%E5%AE%B9%E5%99%A8.md#%E6%95%A3%E5%88%97) # 四、继承 ## 访问权限 Java 中有三个访问权限修饰符:private、protected 以及 public,如果不加访问修饰符,表示包级可见。 可以对类或类中的成员(字段以及方法)加上访问修饰符。成员可见表示其它类可以用成员所在类的对象访问到该成员;类可见表示其它类可以用这个类创建对象。在理解类的可见性时,可以把类当做包中的一个成员,然后包表示一个类,那么就可以类比成员的可见性。 protected 用于修饰成员,表示在继承体系中成员对于子类可见。但是这个访问修饰符对于类没有意义,因为包没有继承体系。 > [浅析 Java 中的访问权限控制](http://www.importnew.com/18097.html) ## 抽象类与接口 **1. 抽象类**
抽象类和抽象方法都使用 abstract 进行声明。抽象类一般会包含抽象方法,但是少数情况下可以不包含,例如 HttpServlet 类,但是抽象方法一定位于抽象类中。抽象类和普通类最大的区别是,抽象类不能被实例化,需要继承抽象类才能实例化其子类。 ```java public abstract class GenericServlet implements Servlet, ServletConfig, Serializable { // abstract method abstract void service(ServletRequest req, ServletResponse res); void init() { // Its implementation } // other method related to Servlet } ``` > [深入理解 abstract class 和 interface](https://www.ibm.com/developerworks/cn/java/l-javainterface-abstract/) **2. 接口**
接口是抽象类的延伸,Java 为了安全性而不支持多重继承,一个类只能有一个父类。但是接口不同,一个类可以同时实现多个接口,不管这些接口之间有没有关系,所以接口弥补不支持多重继承的缺陷。从 Java 8 开始,接口也可以拥有默认的方法实现。 ```java public interface Externalizable extends Serializable { void writeExternal(ObjectOutput out) throws IOException; void readExternal(ObjectInput in) throws IOException, ClassNotFoundException; } ``` **3. 比较**
| | **抽象类** | **接口** | | --- | --- | --- | | 1 | An abstract class can extend only one class or one abstract class at a time | An interface can extend any number of interfaces at a time | | 2 | An abstract class can extend another concrete (regular) class or abstract class | An interface can only extend another interface | | 3 | An abstract class can have both abstract and concrete methods | An interface can have only abstract methods | | 4 | In abstract class keyword "abstract" is mandatory to declare a method as an abstract | In an interface keyword "abstract" is optional to declare a method as an abstract | | 5 | An abstract class can have protected and public abstract methods | An interface can have only have public abstract methods | | 6 | An abstract class can have static, final or static final variable with any access specifier | interface can only have public static final (constant) variable | > [Difference Between Abstract Class and Interface in Java](https://beginnersbook.com/2013/05/abstract-class-vs-interface-in-java/) **4. 使用选择**
使用抽象类: - 需要在几个相关的类中共享代码; - 需要能控制继承来的方法和字段的访问权限,而不是都为 public。 - 需要继承非静态(non-static)和非常量(non-final)字段。 使用接口: - 需要让不相关的类都实现一个方法,例如不相关的类都可以实现 Compareable 接口中的 compareTo() 方法; - 需要使用多重继承。 > [When to Use Abstract Class and Interface](https://dzone.com/articles/when-to-use-abstract-class-and-intreface) ## super **1. 访问父类的成员**
如果子类覆盖了父类的中某个方法的实现,那么就可以通过使用 super 关键字来引用父类的方法实现。 ```java public class Superclass { public void printMethod() { System.out.println("Printed in Superclass."); } } ``` ```java public class Subclass extends Superclass { // Overrides printMethod in Superclass public void printMethod() { super.printMethod(); System.out.println("Printed in Subclass"); } public static void main(String[] args) { Subclass s = new Subclass(); s.printMethod(); } } ``` **2. 访问父类的构造函数**
可以使用 super() 函数访问父类的构造函数,从而完成一些初始化的工作。 ```java public MountainBike(int startHeight, int startCadence, int startSpeed, int startGear) { super(startCadence, startSpeed, startGear); seatHeight = startHeight; } ``` > [Using the Keyword super](https://docs.oracle.com/javase/tutorial/java/IandI/super.html) ## 重载与重写 - 重写存在于继承体系中,指子类实现了一个与父类在方法声明上完全相同的一个方法; - 重载即存在于继承体系中,也存在于同一个类中,指一个方法与已经存在的方法或者父类的方法名称上相同,但是参数类型、个数、顺序至少有一个不同。应该注意的是,返回值不同,其它都相同不算是重载。 # 五、String ## String, StringBuffer and StringBuilder **1. 是否可变**
String 不可变,StringBuffer 和 StringBuilder 可变。 **2. 是否线程安全**
String 不可变,因此是线程安全的。 StringBuilder 不是线程安全的;StringBuffer 是线程安全的,使用 synchronized 来同步。 > [String, StringBuffer, and StringBuilder](https://stackoverflow.com/questions/2971315/string-stringbuffer-and-stringbuilder) ## String 不可变的原因 **1. 可以缓存 hash 值**
因为 String 的 hash 值经常被使用,例如 String 用做 HashMap 等。不可变的特性可以使得 hash 值也不可变,因此就只需要进行一次计算。 **2. String Pool 的需要**
如果 String 已经被创建过了,那么就会从 String Pool 中取得引用。只有 String 是不可变的,才可能使用 String Pool。

**3. 安全性**
String 经常作为参数,String 不可变性可以保证参数不可变。例如在作为网络连接参数的情况下如果 String 是可变的,那么在网络连接过程中,String 被改变,改变 String 对象的那一方以为现在连接的是其它主机,而实际情况却不一定是。 **4. 线程安全**
String 不可变性天生具备线程安全,可以在多个线程中使用。 > [Why String is immutable in Java?](https://www.programcreek.com/2013/04/why-string-is-immutable-in-java/) ## String.intern() 使用 String.intern() 可以保证所有相同内容的字符串变量引用相同的内存对象。 > [揭开 String.intern() 那神秘的面纱](https://www.jianshu.com/p/95f516cb75ef) # 六、基本类型与运算 ## 包装类型 八个基本类型:boolean 1 byte 8 char 16 short 16 int 32 float 32 long 64 double 64 基本类型都有对应的包装类型,它们之间的赋值使用自动装箱与拆箱完成。 ```java Integer x = 2; // 装箱 int y = x; // 拆箱 ``` new Integer(123) 与 Integer.valueOf(123) 的区别在于,Integer.valueOf(123) 可能会使用缓存对象,因此多次使用 Integer.valueOf(123) 会取得同一个对象的引用。 ```java public static void main(String[] args) { Integer a = new Integer(1); Integer b = new Integer(1); System.out.println("a==b? " + (a==b)); Integer c = Integer.valueOf(1); Integer d = Integer.valueOf(1); System.out.println("c==d? " + (c==d)); } ``` ```html a==b? false c==d? true ``` valueOf() 方法的实现比较简单,就是先判断值是否在缓存池中,如果在的话就直接使用缓存池的内容。 ```java public static Integer valueOf(int i) { final int offset = 128; if (i >= -128 && i <= 127) { // must cache return IntegerCache.cache[i + offset]; } return new Integer(i); } ``` The following is the list of primitives stored as immutable objects: - boolean values true and false - all byte values - short values between -128 and 127 - int values between -128 and 127 - char in the range \u0000 to \u007F 自动装箱过程编译器会调用 valueOf() 方法,因此多个 Integer 对象使用装箱来创建并且值相同,那么就会引用相同的对象,这样做很显然是为了节省内存开销。 ```java Integer x = 1; Integer y = 1; System.out.println(c==d); // true ``` > [Differences between new Integer(123), Integer.valueOf(123) and just 123 ](https://stackoverflow.com/questions/9030817/differences-between-new-integer123-integer-valueof123-and-just-123) ## switch A switch works with the byte, short, char, and int primitive data types. It also works with enumerated types and a few special classes that "wrap" certain primitive types: Character, Byte, Short, and Integer. In the JDK 7 release, you can use a String object in the expression of a switch statement. switch 不支持 long,是因为 swicth 的设计初衷是为那些只需要对少数几个值进行等值判断,如果值过于复杂,那么还是用 if 比较合适。 > [Why can't your switch statement data type be long, Java?](https://stackoverflow.com/questions/2676210/why-cant-your-switch-statement-data-type-be-long-java) switch 使用查找表的方式来实现,JVM 中使用的指令是 lookupswitch。 ```java public static void main(String... args) { switch (1) { case 1: break; case 2: break; } } public static void main(java.lang.String[]); Code: Stack=1, Locals=1, Args_size=1 0: iconst_1 1: lookupswitch{ //2 1: 28; 2: 31; default: 31 } 28: goto 31 31: return ``` > [How does Java's switch work under the hood?](https://stackoverflow.com/questions/12020048/how-does-javas-switch-work-under-the-hood) # 七、反射 每个类都有一个 **Class** 对象,包含了与类有关的信息。当编译一个新类时,会产生一个同名的 .class 文件,该文件内容保存着 Class 对象。 类加载相当于 Class 对象的加载。类在第一次使用时才动态加载到 JVM 中,可以使用 Class.forName('com.mysql.jdbc.Driver.class') 这种方式来控制类的加载,该方法会返回一个 Class 对象。 反射可以提供运行时的类信息,并且这个类可以在运行时才加载进来,甚至在编译时期该类的 .class 不存在也可以加载进来。 Class 和 java.lang.reflect 一起对反射提供了支持,java.lang.reflect 类库包含了 **Field** 、**Method** 以及 **Constructor** 类。可以使用 get() 和 set() 方法读取和修改 Field 对象关联的字段,可以使用 invoke() 方法调用与 Method 对象关联的方法,可以用 Constructor 创建新的对象。 IDE 使用反射机制获取类的信息,在使用一个类的对象时,能够把类的字段、方法和构造函数等信息列出来供用户选择。 > [深入解析 Java 反射(1)- 基础](http://www.sczyh30.com/posts/Java/java-reflection-1/) **Advantages of Using Reflection:**
- **Extensibility Features** : An application may make use of external, user-defined classes by creating instances of extensibility objects using their fully-qualified names. - **Class Browsers and Visual Development Environments** : A class browser needs to be able to enumerate the members of classes. Visual development environments can benefit from making use of type information available in reflection to aid the developer in writing correct code. - **Debuggers and Test Tools** : Debuggers need to be able to examine private members on classes. Test harnesses can make use of reflection to systematically call a discoverable set APIs defined on a class, to insure a high level of code coverage in a test suite. **Drawbacks of Reflection:**
Reflection is powerful, but should not be used indiscriminately. If it is possible to perform an operation without using reflection, then it is preferable to avoid using it. The following concerns should be kept in mind when accessing code via reflection. - **Performance Overhead** : Because reflection involves types that are dynamically resolved, certain Java virtual machine optimizations can not be performed. Consequently, reflective operations have slower performance than their non-reflective counterparts, and should be avoided in sections of code which are called frequently in performance-sensitive applications. - **Security Restrictions** : Reflection requires a runtime permission which may not be present when running under a security manager. This is in an important consideration for code which has to run in a restricted security context, such as in an Applet. - **Exposure of Internals** :Since reflection allows code to perform operations that would be illegal in non-reflective code, such as accessing private fields and methods, the use of reflection can result in unexpected side-effects, which may render code dysfunctional and may destroy portability. Reflective code breaks abstractions and therefore may change behavior with upgrades of the platform. > [Trail: The Reflection API](https://docs.oracle.com/javase/tutorial/reflect/index.html) # 八、异常 Throwable 可以用来表示任何可以作为异常抛出的类,分为两种: **Error** 和 **Exception**,其中 Error 用来表示编译时系统错误。 Exception 分为两种: **受检异常** 和 **非受检异常**。受检异常需要用 try...catch... 语句捕获并进行处理,并且可以从异常中恢复;非受检异常是程序运行时错误,例如除 0 会引发 Arithmetic Exception,此时程序奔溃并且无法恢复。

> [Java 入门之异常处理](https://www.tianmaying.com/tutorial/Java-Exception) > [Java 异常的面试问题及答案 -Part 1](http://www.importnew.com/7383.html) # 九、泛型 ```java public class Box { // T stands for "Type" private T t; public void set(T t) { this.t = t; } public T get() { return t; } } ``` > [Java 泛型详解](https://www.ziwenxie.site/2017/03/01/java-generic/) > [10 道 Java 泛型面试题](https://cloud.tencent.com/developer/article/1033693) # 十、注解 Java 注解是附加在代码中的一些元信息,用于一些工具在编译、运行时进行解析和使用,起到说明、配置的功能。注解不会也不能影响代码的实际逻辑,仅仅起到辅助性的作用。 > [注解 Annotation 实现原理与自定义注解例子](https://www.cnblogs.com/acm-bingzi/p/javaAnnotation.html) # 十一、特性 ## 面向对象三大特性 > [封装、继承、多态](https://github.com/CyC2018/Interview-Notebook/blob/master/notes/%E9%9D%A2%E5%90%91%E5%AF%B9%E8%B1%A1%E6%80%9D%E6%83%B3.md#%E5%B0%81%E8%A3%85%E7%BB%A7%E6%89%BF%E5%A4%9A%E6%80%81) ## Java 各版本的新特性 New highlights in Java SE 8 1. Lambda Expressions 2. Pipelines and Streams 3. Date and Time API 4. Default Methods 5. Type Annotations 6. Nashhorn JavaScript Engine 7. Concurrent Accumulators 8. Parallel operations 9. PermGen Error Removed New highlights in Java SE 7 1. Strings in Switch Statement 2. Type Inference for Generic Instance Creation 3. Multiple Exception Handling 4. Support for Dynamic Languages 5. Try with Resources 6. Java nio Package 7. Binary Literals, Underscore in literals 8. Diamond Syntax > [Difference between Java 1.8 and Java 1.7?](http://www.selfgrowth.com/articles/difference-between-java-18-and-java-17) > [Java 8 特性 ](http://www.importnew.com/19345.html) ## Java 与 C++ 的区别 Java 是纯粹的面向对象语言,所有的对象都继承自 java.lang.Object,C++ 为了兼容 C 即支持面向对象也支持面向过程。 比较详细的内容: | Java | C++ | | -- | -- | | Java does not support pointers, templates, unions, operator overloading, structures etc. The Java language promoters initially said "No pointers!", but when many programmers questioned how you can work without pointers, the promoters began saying "Restricted pointers." Java supports what it calls "references". References act a lot like pointers in C++ languages but you cannot perform arithmetic on pointers in Java. References have types, and they're type-safe. These references cannot be interpreted as raw address and unsafe conversion is not allowed. | C++ supports structures, unions, templates, operator overloading, pointers and pointer arithmetic.| | Java support automatic garbage collection. It does not support destructors as C++ does. | C++ support destructors, which is automatically invoked when the object is destroyed. | | Java does not support conditional compilation and inclusion. | Conditional inclusion (#ifdef #ifndef type) is one of the main features of C++. | | Java has built in support for threads. In Java, there is a `Thread` class that you inherit to create a new thread and override the `run()` method. | C++ has no built in support for threads. C++ relies on non-standard third-party libraries for thread support. | | Java does not support default arguments. There is no scope resolution operator (::) in Java. The method definitions must always occur within a class, so there is no need for scope resolution there either. | C++ supports default arguments. C++ has scope resolution operator (::) which is used to to define a method outside a class and to access a global variable within from the scope where a local variable also exists with the same name. | | There is no _goto_ statement in Java. The keywords `const` and `goto` are reserved, even though they are not used. | C++ has _goto_ statement. However, it is not considered good practice to use of _goto_ statement. | | Java doesn't provide multiple inheritance, at least not in the same sense that C++ does. | C++ does support multiple inheritance. The keyword `virtual` is used to resolve ambiguities during multiple inheritance if there is any. | | Exception handling in Java is different because there are no destructors. Also, in Java, try/catch must be defined if the function declares that it may throw an exception. | While in C++, you may not include the try/catch even if the function throws an exception. | | Java has method overloading, but no operator overloading. The `String` class does use the `+` and `+=` operators to concatenate strings and `String`expressions use automatic type conversion, but that's a special built-in case. | C++ supports both method overloading and operator overloading. | | Java has built-in support for documentation comments (`/** ... */`); therefore, Java source files can contain their own documentation, which is read by a separate tool usually `javadoc` and reformatted into HTML. This helps keeping documentation maintained in easy way. | C++ does not support documentation comments. | | Java is interpreted for the most part and hence platform independent. | C++ generates object code and the same code may not run on different platforms. | > [What are the main differences between Java and C++?](http://cs-fundamentals.com/tech-interview/java/differences-between-java-and-cpp.php) ## JRE or JDK - JRE is the JVM program, Java application need to run on JRE. - JDK is a superset of JRE, JRE + tools for developing java programs. e.g, it provides the compiler "javac" # 参考资料 - Eckel B. Java 编程思想[M]. 机械工业出版社, 2002. - Bloch J. Effective java[M]. Addison-Wesley Professional, 2017.