Interviewer: Tell me what the generic mechanism of Java is.
(Learning video sharing: java video tutorial)
Lite version answer
When developing Java, by using diamond syntax < >, Make a class, interface or method have a group of classes that accept a generic reference as a parameter. The generic class is ultimately defined by the rules in angle brackets. The generic class is usually represented by a capital letter, usually the letter T, and is later passed by the Java compiler During compilation, the generics will be erased, replaced according to the specific class used, and the class bytecode will be generated, so the generics are transparent to the Jvm.
Deluxe version answer
Generics were officially introduced in JDK1.5
Background of the introduction of generics in Java
Java collections allow multiple types to be put in , for example
List list=new ArrayList(); list.add("String"); list.add(1024);
This was a common usage before JDK1.5. Even if it is placed under today's JDK, it is allowed. Then the usage of this collection will cause some problems. I am afraid that only the person who created the collection knows what types are placed in the collection. Other callers cannot determine it at all. In large projects, problems are very likely to occur. If the caller forcefully transfers an object in the collection, once the type is wrong , an error will be reported. This kind of error cannot be found in the compilation stage at that time, and can only be found when running. In order to solve similar problems, JDK introduced generics.
Generic types
After the introduction of generics, it is possible to check whether the type meets the requirements during the compilation phase. To a large extent, blind type conversion is eliminated. Generics mainly work in the compiler. After compilation, the JVM cannot perceive generics when it is running. Generics are divided into ordinary generics and wildcard generics.
1. Ordinary generics
As the name suggests, this type of generics supports passing in any type when calling, but when calling, the generics on the left and right sides of the equal sign must be consistent. The diamond on the right side of JDK1.7 can be omitted.
class Test<T>{...} //聲明時(shí) Test<Integer> test = new Test<Integer>(); //調(diào)用時(shí) Test<Integer> test = new Test(); //1.7調(diào)用時(shí)
3. Bounded generics
<?>//無界泛型,任意類型都匹配 <T extends ClassA>//有界泛型 - 繼承自某父類 <T extends InterfaceB>//有界泛型 - 實(shí)現(xiàn)某接口 <T extends ClassA & InterfaceB & InterfaceC ... >//有界泛型 - 多重邊界 <T super ClassC>//有界泛型 - 指定類父類
Generic erasure
As mentioned just now, generics only work when writing code and compiling, and Jvm loads classes The runtime is invisible and transparent because the compiler erases generics when compiling. To put it simply and loosely, it means: remove the <> angle brackets on the class or method, and follow the rules in the angle brackets. Change to a specific class, so the Jvm runtime does not know that there have been generics in this code before. This is done for compatibility. As mentioned earlier, generics were only introduced in JDK1.5. In this way, even if generics are introduced, now Some JVMs do not need major changes, they only need to change the compiler. For this kind of syntax that the compiler can perceive, but the virtual machine cannot, people call this syntax sugar (suger). The compiler After desugering, the dry information is obtained and handed over to the virtual machine for execution.
(Recommended graphic tutorial: java introductory tutorial)
Generic erasure mechanism
//Pair的泛型 public class Pair<T> { private T mFirst; private T mSecond; public T getmFirst() { return mFirst; } public void setmFirst(T mFirst) { this.mFirst = mFirst; } public T getmSecond() { return mSecond; } public void setmSecond(T mSecond) { this.mSecond = mSecond; } } //Pair的原始類型 //無論何時(shí)定義一個(gè)泛型類型,都會(huì)自動(dòng)提供一個(gè)相應(yīng)的 原始類型 public class Pair { private Object mFirst; private Object mSecond; public Object getFirst() { return mFirst; } public void setFirst(Object mFirst) { this.mFirst = mFirst; } public Object getSecond() { return mSecond; } public void setmSecond(Object mSecond) { this.mSecond = mSecond; } } //如果調(diào)用Pair<T extends String>編譯擦除后得到如下= class Pair{ private String mFirst; private String mSecond; ... }
When generics are used as input parameters of a method, After erasing, it is replaced with the lower bound of a wildcard generic, such as the add method
public static void insertElements(List<? super A> list){ //Add進(jìn)來的都是層級(jí)“小于等于”A的,也就是下界 list.add(new A()); list.add(new B()); list.add(new C()); }
When the generic is used as a return parameter of a method, it is replaced with the upper bound of a wildcard generic after erasure, such as the get method
public void processElements(List<? extends A> elements){ //for循環(huán)挨個(gè)get list 里的數(shù)據(jù),都當(dāng)做A,也就是都當(dāng)做上界 for(A a : elements){ System.out.println(a.getValue()); } }
No matter how much we talk about generic type erasure, no matter how many blogs we read, the official definition is not as accurate as the official definition. Here is a section of Oracle’s official document on type erasure
Type Erasure Generics were introduced to the Java language to provide tighter type checks at compile time and to support generic programming. To implement generics, the Java compiler applies type erasure to: Replace all type parameters in generic types with their bounds or Object if the type parameters are unbounded. The produced bytecode, therefore, contains only ordinary classes, interfaces, and methods.Insert type casts if necessary to preserve type safety.Generate bridge methods to preserve polymorphism in extended generic types.Type erasure ensures that no new classes are created for parameterized types; consequently, generics incur no runtime overhead.
For more interview questions, please click: java interview questions and answers
The above is the detailed content of Let's talk about the generic mechanism of Java. For more information, please follow other related articles on the PHP Chinese website!

Hot AI Tools

Undress AI Tool
Undress images for free

Undresser.AI Undress
AI-powered app for creating realistic nude photos

AI Clothes Remover
Online AI tool for removing clothes from photos.

Clothoff.io
AI clothes remover

Video Face Swap
Swap faces in any video effortlessly with our completely free AI face swap tool!

Hot Article

Hot Tools

Notepad++7.3.1
Easy-to-use and free code editor

SublimeText3 Chinese version
Chinese version, very easy to use

Zend Studio 13.0.1
Powerful PHP integrated development environment

Dreamweaver CS6
Visual web development tools

SublimeText3 Mac version
God-level code editing software (SublimeText3)

Hot Topics

There are three common methods to traverse Map in Java: 1. Use entrySet to obtain keys and values at the same time, which is suitable for most scenarios; 2. Use keySet or values to traverse keys or values respectively; 3. Use Java8's forEach to simplify the code structure. entrySet returns a Set set containing all key-value pairs, and each loop gets the Map.Entry object, suitable for frequent access to keys and values; if only keys or values are required, you can call keySet() or values() respectively, or you can get the value through map.get(key) when traversing the keys; Java 8 can use forEach((key,value)->

Optional can clearly express intentions and reduce code noise for null judgments. 1. Optional.ofNullable is a common way to deal with null objects. For example, when taking values ??from maps, orElse can be used to provide default values, so that the logic is clearer and concise; 2. Use chain calls maps to achieve nested values ??to safely avoid NPE, and automatically terminate if any link is null and return the default value; 3. Filter can be used for conditional filtering, and subsequent operations will continue to be performed only if the conditions are met, otherwise it will jump directly to orElse, which is suitable for lightweight business judgment; 4. It is not recommended to overuse Optional, such as basic types or simple logic, which will increase complexity, and some scenarios will directly return to nu.

The core workaround for encountering java.io.NotSerializableException is to ensure that all classes that need to be serialized implement the Serializable interface and check the serialization support of nested objects. 1. Add implementsSerializable to the main class; 2. Ensure that the corresponding classes of custom fields in the class also implement Serializable; 3. Use transient to mark fields that do not need to be serialized; 4. Check the non-serialized types in collections or nested objects; 5. Check which class does not implement the interface; 6. Consider replacement design for classes that cannot be modified, such as saving key data or using serializable intermediate structures; 7. Consider modifying

In Java, Comparable is used to define default sorting rules internally, and Comparator is used to define multiple sorting logic externally. 1.Comparable is an interface implemented by the class itself. It defines the natural order by rewriting the compareTo() method. It is suitable for classes with fixed and most commonly used sorting methods, such as String or Integer. 2. Comparator is an externally defined functional interface, implemented through the compare() method, suitable for situations where multiple sorting methods are required for the same class, the class source code cannot be modified, or the sorting logic is often changed. The difference between the two is that Comparable can only define a sorting logic and needs to modify the class itself, while Compar

Method reference is a way to simplify the writing of Lambda expressions in Java, making the code more concise. It is not a new syntax, but a shortcut to Lambda expressions introduced by Java 8, suitable for the context of functional interfaces. The core is to use existing methods directly as implementations of functional interfaces. For example, System.out::println is equivalent to s->System.out.println(s). There are four main forms of method reference: 1. Static method reference (ClassName::staticMethodName); 2. Instance method reference (binding to a specific object, instance::methodName); 3.

To deal with character encoding problems in Java, the key is to clearly specify the encoding used at each step. 1. Always specify encoding when reading and writing text, use InputStreamReader and OutputStreamWriter and pass in an explicit character set to avoid relying on system default encoding. 2. Make sure both ends are consistent when processing strings on the network boundary, set the correct Content-Type header and explicitly specify the encoding with the library. 3. Use String.getBytes() and newString(byte[]) with caution, and always manually specify StandardCharsets.UTF_8 to avoid data corruption caused by platform differences. In short, by

There are three common ways to parse JSON in Java: use Jackson, Gson, or org.json. 1. Jackson is suitable for most projects, with good performance and comprehensive functions, and supports conversion and annotation mapping between objects and JSON strings; 2. Gson is more suitable for Android projects or lightweight needs, and is simple to use but slightly inferior in handling complex structures and high-performance scenarios; 3.org.json is suitable for simple tasks or small scripts, and is not recommended for large projects because of its lack of flexibility and type safety. The choice should be decided based on actual needs.

How to quickly create new emails in Outlook is as follows: 1. The desktop version uses the shortcut key Ctrl Shift M to directly pop up a new email window; 2. The web version can create new emails in one-click by creating a bookmark containing JavaScript (such as javascript:document.querySelector("divrole='button'").click()); 3. Use browser plug-ins (such as Vimium, CrxMouseGestures) to trigger the "New Mail" button; 4. Windows users can also select "New Mail" by right-clicking the Outlook icon of the taskbar
