6.4 KiB
title, weight
| title | weight |
|---|---|
| Lecture 1. Introduction | 5 |
Why do we choose Java?
According to TIOBE Programming Community Index Java is one of the most demanded languages in the programming field.
Img. 1 - TIOBE Programming Community Index
Also, Java has a lot of advantages for beginners such as:
- It's fairly easy
- It has a broad spectre of usage
- Server
- Desktop
- Mobile devices
- Smart-cards
What does Java consist of?
Java consists of multiple main components. The first one being the Java compiler. The process of converting human-readable text to machine code is called compilation.
Img.2 - Simplified Compilation Process
The behaviour of the Java compiler is described by the Java Language Specification or JLS.
Let's talk about compilers a little bit. For example if we will take C++. If we take C++ compiler for x86-64 platform and Windows operating system, and launch the compiler on source code it will turn it directly into assembly code.
Img. X -- C++ code compilation process under x86-64 architecture.
But what if I want to run my program on Linux instead of Windows. I will need to take a different compiler under different operating system and recompile my code using a new compiler. It's not very convenient.
Java tries to protect us from that. By converting the Java source code into Java byte code. And then the byte code will be ran on the Java virtual machine which will run our program on the native processor.
Img. X -- Java code compilation
This approach allows to change only JVM according to our platform (x86_64, ARM, ...) and operating system (Windows, MacOS, GNU-Linux, ...) while byte code stays the same. We can only write our code once, than compile it and run under everywhere.
As the motto says:
Write once --
debugrun everywhere.
The third component of Java is the standart library which is included in the JVM.
There are multiple redactions of Java-platforms:
- Standart edition
- For regular aplications
- Enterprise edition
- For server aplications
- Micro-edition
- For mobile aplications
- Isn't in use nowadays 🪦
- Java Card
- Sim- and smart-cards
There also were multiple versions of Java throughout its history.
- JDK 1.0 (Jan 1996)
- J2SE 1.2 (Dec 1998)
- Collections Framework
- J2SE 5.0 (Sep 2004)
- Generics
- Java SE 8 (Mar 2014)
- Streams and Lambdas
- Java SE 9 (Sep 2017)
- Modules
- Java SE 10 (Mar 2018)
var
- Java 11 (Sep 2018)
jshell
- Java 17 [LTS-old] (Sep 2021)
- Previous stable version
- Many little changes
- Java 21 [LTS] (Sep 2023)
- Current stable version
- Java 25 (Sep 2025)
- Next version
Java comes in two parts: JDK - Java Development Kit and JVM - Java Virtual Machine.
There is also a JRE - Java Runtime Environment. For example if we want to run our code somewhere on the server we don't need to compile it there because we have our byte code and we just need JRE to run it.
Some of the most popular JVMs right now are:
- OpenJDK
- Eclipse
- Azul Systems
- Excelsior JET
The disadvantage of such system is in the connection between JVM and a native processing unit. In case of C++ compiler that we reviewed earlier the source code is compiled directly into machine-code but in case with Java it is compiled into byte-code. And so the problem is to develop such a JVM that would quickly turn our byte-code into machine-code. Anyway it takes extra time. That's why it mostly will be slower than direct compilation into machine-code. So ultimately while we have the advantage of compiling out code only once, we have the disadvantage of turning byte-code into machine-code slower.
None the less, there is a way to speed up this process which is called JIT - Just In Time compilation. How does it work? While our program is running some of the instructions or functions turns directly into processor commands.
What is garbage collection?
For example we have int which is represented with 4 bytes of data which is directly stored in memory.
But what if we have a String. How many memory cells does this string take? We don't know. We will say that our String that has length of 5 symbols is stored at 0x12347865. We defined an address where this string is located in memory. And somewhere in the memory of our programm will be a large buffer where the 5 cells will be stored.
But what do we do with that buffer? We won't need it forever and so sometimes we need to clear that buffer. In case with C/C++ whoever created the memory for that string is in charge of clearing it. There are also languages with garbage collection such as Java and Python. The purpose of the Garbage Collector is to automatically find variables that we no longer need and clear their memory.
Suppose we have some code like this.
if (someCondition) {
x = [1, 3, 7] // first link
// some code here
y = x // second link
// some code here
} // no links
After the if-statement we no longer need x or y. Every variable in this case x and y is the link to our array ([1, 3, 7]). After we left the if-statement the amount of links to the array is zero, so we can safely delete the array. This approach is implemented in Python. The problem is that if we have to objects linked to one another and there are no external links, they will not be deleted by garbage collector since there link counter is 1.
Img. X -- Edge case for link counting
The advanced way of implementing the garbage collection is traversing the graph of links which is implemented in Java, C# and Go.
What other advantages does Java have?
- It's easy (in terms of syntax)
- It's secure and stable
- It supports Unicode
- It supports multithreading
- It has backwards compatibility
How should Java code look like?
You can go and learn about it here.