Skip to content

This repo is a full guide to learn java language in addition to software engineering principles, also it contains mini-applications on java from scratch -basic concepts- to small | mid-sized java projects

License

Notifications You must be signed in to change notification settings

Sarah627/Java_miniatures

Repository files navigation

Java Miniatures

Java Miniatures is a comprehensive repository dedicated to honing Java programming skills and delving into software engineering concepts. As a passionate Computer Science graduate aspiring to excel in the field of Software Engineering, I've curated this repository to serve as a platform for my continuous learning journey and to showcase my expertise in Java programming.

What the Repository Does:

The repository encompasses a wide range of Java topics, starting from fundamental concepts and progressing towards more advanced software engineering principles. Each section of the repository is meticulously crafted to provide clear explanations, code examples, and practical projects, all aimed at deepening understanding and mastery of Java and its application in real-world software development scenarios. The repo will include the fundamental Computer Science topics.

Challenges Faced and Future Features:

Throughout this learning journey, I've encountered various challenges, both technical and conceptual. However, each challenge has served as an opportunity for growth and learning. In the future, I aim to implement additional features such as interactive coding exercises, more extensive project examples, and collaborative learning opportunities to further enhance the repository's utility and effectiveness as a learning resource.

Important:

This repository IS NOT A ROADMAP FOR LEARNING Java exactly, you can use different arrangement e.g. If you want to learn generics first before knowing about frameworks (e.g.spring boot) it won't cause a problem, I use this repo more likely as a documentation for my programming skills progress, how my code, thinking, experience in using Java change over time in addition to aiming to be a good reference in future (ISA) for whom are concerned about learning Java and software engineering.

Table of Contents

Part One : Pick up your weapon !

Java Basics
  1. Elementary Programming
  2. Control Structures Part 1: (Selections)
  3. Mathematical Functions,Characters, and Strings
  4. Control Structures Part 2: (Iterators)
  5. Methods
  6. Arrays
Object-Oriented Programming
  1. Objects and Classes
  2. Object-Oriented Thinking
  3. Inheritance and Polymorphism
  4. Exception Handling and Text I/O
  5. Abstract Classes and Interfaces
Java in Action
  1. Build tools:

    • Ant
    • Maven
    • Gradle
  2. building GUI

    • JavaFX
    • Java Swing
  3. Web Frameworks

    • Spring
    • Spring-boot
  4. ORM

    • JPA
    • Hibernate
Advanced Java Topics
  1. Binary I/O
  2. Generics
  3. How JVM works
  4. Multithreading and Parallel Programming
  5. Networking
  6. Java Database Programming
Data Structures
  1. Recursion
  2. Stacks & Queues
  3. Lists
  4. Trees
  5. Priority Queues
  6. Maps & Hash Tables
  7. Search Trees
Algorithms
  1. Asymptotic analysis
  2. Sorting
  3. Searching
  4. Graphs
  5. Strings

Part Two: Software Engineering Principles

  1. OOA & OOD
  2. SOLID Principles
  3. Design Patterns
  4. Refactoring Techniques
  5. Software Development Methodologies

Part Three: Projects

  1. Small-scale Java Projects
  2. Medium-scale Software Engineering Projects
  3. Real-world Application Examples

Cross-Part knowledge

Challenges and Exercises

  • Coding Challenges
  • Problem-solving Exercises
  • Algorithm Implementations

Resources

  • Online Tutorials and Courses
  • Recommended Books
  • Useful Articles and Videos

Contributions

  • How to Contribute
  • Guidelines for Contributors
  • Community Engagement Opportunities

Part One : Pick Up Your Weapon!

1 - Java Basics

Getting-Started :D 🔥

according to Java Docs official website Oracle Docs.

'Java is a general-purpose, concurrent, strongly typed, class-based object-oriented programming language'

  • general puprose means you can build applications using this language in a wide range of different domains (Desktop applications, client/server applications/ web applications, Enterprise applications).

  • concurrent means the ability of the language to execute multiple processes in the same time. (This is kinda an advanced concept for beginners and maybe cs students in first 2 or three semesters)

  • strongly-typed means the strict attachment of language syntax to define variables data types. (This will be explained and noticed as we go deep in the language)

  • class based object oriented means this language structure is based on object-oriented concepts and use of classes.

Don't worry these characteristics of the language will be more clear as we go down in this repo


Installing JDK

Installing JDK

To get started with Java we must install JDK on our local machine, (here, I'm considering you are a beginner or totally newcomer in programming world or even did not visit virtualization concepts|tools before).

Installing JDK is a must it provides the necessary tools and libraries to develop, compile, and run Java applications.

click on the following link oracle and install the latest JDK versions on your own device

screenshot

download and install JDK based on your os and processor architecture set

What's JDK?

"JDK" stands for Java Development Kit, it's the development environment for building Java applications, it consists of JVM -Java Virtual Machine-, JRE -Java Runtime Environment- and development tools like compiler etc.

JDK

Java is machine independent language, or it's more descriptive to refer to it as platform-independent; you can run it on any platform, while the JVM is platform-dependent; different JVM is designed for each platform (Linux, Windows, macOS).

In other words, if we have three local devices, each of which has a different platform installed in it e.g.(windows, mac, linux), all is required to install a JVM for each device and the Java files can run on any machine of them after compiling it just one time.

Once you downloaded the latest version of JDK here you have two options:

  1. if it's your first time downloading

    you need to edit system environment variables first

    These instructions for Windows users only ⚠:

    • open your Program Files directory navigate till you find the jdk directory
    • open the bin directory then copy its path path image
    • press win key on your keyboard ⌨ then type in search bar down edit system environment variables
    • click on Environment Variables, from user variables choose path then click on edit
    • on path window click add new, then paste the path you copied earlier and press enter
    • click on move up to make it on the top of the list and voilà!
  2. if you've already installed earlier JDK version

    • open your cmd(win key, type cmd | win key + R then type cmd)

      • write the following:

            Java -version
        

      Java version on cmd

      I have JDK version 17 installed on my own, now we are going to switch it to the new version of JDK

      • we'll repeat the steps of the previous option (editing system environment variables, etc.)
      • Check your Java version again in cmd (restart it if you didn't close it already)

      and Voila!

      new jdk version is working well


1. Elementary Programming

1. Elementary Programming

Remember this part mentioned earlier ? class-based

Java is a class-based object-oriented programming language

Let's see how this concept was applied!

Writing your first Hello world program!:-D "programmers' traditional custom"

prerequisites

if you didn't install the jdk yet, go to this part and install it

It's better if you have an IDE, till this point it's not mandatory yet ,but then it will be as we are not at the age of DOS anymore.

  1. If you don't have an IDE: (note: I won't let you stay lazy forever, you'll have to install it)

    • create a new directory (folder) for your Java practice, right-click, new, text document

    • yes, that's it just empty text document, rename this document "HelloWorld.Java"

    • open the document, as mentioned before Java is class based so to write any instructions they must be inside classes, now let's type our first class:

         public class HelloWorld{}

    public & class are both keywords all written in lower case letters

    HelloWorld is the name of the class ,and it must be written in pascal case (the first letter of each word is capital)

    inside the curly braces {} we are going to illustrate our code

    • keywords or reserved words are predefined words in a programming language with a specific use, you can't use them for naming (simply, because the compiler will be confused if it's the word used by the language to identify some order to be executed by the program ,or it's just a programmer defined variable)

    • every Java program should at least have one class, to execute the program we should have a main method (in other words method its name is "main")

    • inside our created class we define the main method as follows:

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

      public, static, void, String are all keywords

      naming methods using camelCase in Java if it consists of more than one word otherwise all lower case

    • the curly braces {} mean block of code, here we write the lines of code that will be executed, their place is after every class, method, control statements, specific keywords e.g. static keyword. They also represent a specific scope (we'll talk about this soon).

    • inside the curly braces of the main method we will print our first hello world

         System.out.println("Hello World!");

      Do Not Forget The Semicolon!

      any string is enclosed in double quotes

      System.out.println() is the method we'll use for printing anything on the terminal or console screen to user.

    • to run your program open your cmd win + R type cmd then enter ↩, in the same directory, if you cannot navigate to the directory where the Java file lies, simply in the address bar of your folder window type cmd as follows: cmd of the current directory

    • inside cmd type Javac {yourfilename.Java} (don't forget the extension)

        Javac replacethis.Java
      

    cmd

    • write the name of the file right with its correct format ,or you'll face error: file not found

    Javac stands for Java compile, you'll notice that a .class file is created on your directory this is the bytecode. you compile one time and run the code infinitely.

    • to run the program write Java {yourfilename} (note here without the extension).

    and Voila!

program run

what does program looks like

you can check the file here in this directory Hello.Java file.

Now we consider you a real programmer

queen's knight

Java general knowledge and syntax:

To explain data types and variables, and how Java works with code, here’s a clear and beginner-friendly explanation:

1. Variables and Data Types:

In Java, a variable is a symbol that represents a value stored in the computer’s memory. Before you can use a variable, you need to declare it with a data type, which tells Java what kind of value it will hold.

Primitive Data Types:
  • int: Stores integers (whole numbers), e.g., int age = 25;
  • double: Stores real numbers (floating-point), e.g., double radius = 3.5;
  • char: Stores single characters, e.g., char grade = 'A';
  • boolean: Stores true or false values, e.g., boolean isJavaFun = true;

Example:

int radius = 5;  // Declare a variable of type int to store the radius
double area = 3.14 * radius * radius;  // Declare a variable of type double to store the calculated area

In this example, radius holds an integer value, and area holds a floating-point value calculated using radius. Java ensures that each variable can only store values that match its declared type.

2. How Java Works with Code (Compilation and Bytecode):

When you write Java code, the process of transforming it into an executable program involves several steps:

  1. Compilation: Java source code (written in .Java files) is compiled by the Java Compiler (Javac) into bytecode, which is stored in .class files. Bytecode is a low-level, platform-independent code.

    compilation process

  2. Java Virtual Machine (JVM): The JVM executes the bytecode. Since JVMs are available on many platforms (Windows, macOS, Linux), the same Java bytecode can run on any platform without modification, making Java portable. This is why Java is often described as "write once, run anywhere."

  3. Portability: Java programs are portable because the JVM abstracts away the underlying operating system. As long as the target machine has a JVM, the compiled bytecode will run on it.

JVM

Example of Compilation:

  • MyProgram.Java (source code) is compiled into MyProgram.class (bytecode).
  • The JVM then reads MyProgram.class and runs the program on any platform.
3. Types of Java (SE, EE, ME):

Java comes in several editions, each designed for different types of applications:

  • Java SE (Standard Edition): Provides core libraries and APIs for building general-purpose desktop and server-side applications. This is where most beginners start, and it includes the base language features, APIs for data structures, concurrency, and input/output.

  • Java EE (Enterprise Edition): Built on top of Java SE, Java EE is used to develop large-scale, distributed, multi-tiered applications, especially for enterprise environments. It includes tools for handling web services, databases, and more complex networking.

  • Java ME (Micro Edition): A subset of Java SE, Java ME is designed for mobile devices and embedded systems, with a smaller footprint for applications running on limited hardware like smartphones and IoT devices.

  • every programming language has a syntax defines it, simply like grammar in English defines how you form a sentence has context, similarly in programming the syntax allows you to write program instructions correct or with a context for the compiler.

  • curly braces{} define a block of code -in other words lines of code- and also define scope.

  • yup, just like C language semicolon is the statements terminator ;

  • // double slashes for single line comments and /* comments lines go here */ for multiple lines of comments (for those who are new to programming, comments are sentences or lines won't be treated as code or instruction they will be ignored by the compiler, programmers use comments to clarify points about their code, like comments show incomplete implementation or reference to the original source code or even to clear confusion about the code they wrote)

  • Java is case-sensitive language (you get syntax error if you mistyped a keyword or wrote its letter capital rather than small or vice versa), e.g.

    Public void fun(){...} --> syntax error (Public is not a keyword)

  • parenthesis () follow functions names, flow control keywords like for, while, if...etc.

  • while square brackets follow 'Array' types [] e.g.

    String[] -> indicates array of type String.

  • later in advanced Java programming we will get familiar with the ArrayList, we will find the angle brackets <> follow ArrayLists keyword in which contain the Type of the ArrayList (ArrayList is simply resizable array) e.g.

    ArrayList<String>.

  • every Java program must have at least one class.

  • Java program start execution from main method.

  • string is enclosed in double quotes while character is enclosed in single quotes.

errors types:

  • Syntax errors: mistyping, missing punctuation

  • Runtime errors: occur while program is actually running if the IDE indicated a process cannot be carried out or through input errors.

  • Logical errors (semantic) -cruel one😈-: the code actually works fine but doesn't give you the expected output e.g. you wanted to make simple program that multiply numbers. now look at the following code:

    int x = 2, y = 2;

    int result = x + y;

    System.out.println(result); >>>> 4 looks fine nothing wrong, no errors, the result is right :)

    try again making y = 3 instead of 2

    int x = 2, y = 3;

    int result = x + y;

    System.out.println(result); >>> 5 here there is a problem, the code is working fine as no errors appear but wait a minute 😅, a plus " + " sign where inserted instead of " * " multiplication sign.

    here where the error takes place, it's not in syntax, the code works fine but doesn't give you the expected results, these types of errors are dedicated and corrected using tests we will come to this point later.

for more on errors types here is a useful link not for newcomers , this is nice introduction to errors types of errors in Java.

Dealing with data in Java

  • a variable is used to store data, you can think of variables as where we store values, these values can be numeric, string (group of characters), reference, etc.

  • basic syntax to declare a variable in Java

    datatype {variable-name} = value;

    • declaring means you announce to your compiler about storage place of name (your variable name) and type (your variable type) and it has to reserve or allocate this amount of memory space.
    • definition is when you assign a value to this identifier or the variable, equal sign = is the assignment operator.
    • assignment is right-to-lef operation -this is super important-, as you say the value 3 is assigned to variable x of type integer for this statement int x = 3; .
    • obviously, we don't have to mention "not to forget the damn semicolon ; after each statement you write", thank you :) .
  • naming conventions:

    • camel case.
    • do not start naming a variable with a number
    • you can start your variable name with dollar sign $ or with underscore _.
    • variables cannot be reserved words, e.g. you cannot name your variable 'static' gives you syntax error.
    • for clean code early teaching purpose, choose meaningful names for your variables describe them.
    • do not contain spaces in your variable name, keep using camel case if your variable name is more than one word.
  • data types:

    • In general, they are two types numerical or non-numerical, numerical like integers, floating point or decimal numbers, non-numerical like strings "Hello my name is" or boolean like True or False.

    • In Java, they are two categories, primitive data types and reference data types, they are classified based on how they store data, primitive copies only values while reference hold the address to the value they just copied, if you couldn't understand this no problem it will be clear later.

    • primitive data types are:

      • byte
      • short
      • int
      • long
      • float
      • double
      • boolean
      • char
    • reference data type - the only one you'll use for now - is String (take care of the capital letter).

    • data types tell the compiler how much space to reserve for that variable in memory.

      see this

  • note as variable word declares, its value can be changes during the execution of the program, from this point let's get familiar with constants.

  • constants cannot change their value, we declare them using final keyword then data type follows final then the constant name, e.g.

    final float PI = 3.14;

    • Upper case as naming convention for constants.
  • note you cannot re-declare variable but you can re-assign i, e.g.

    redeclaration definition

simple operations:

  • you can perform simple operations using binary arithmetic operators: + , - , * , / , %

    • 'asterisk' * is used for multiplication, 'single slash' / is used for division, and % is called modulus gives you the remainder, e.g.

      remainder

    • note that = means assignment not equality.

taking user input

  • for taking custom input from user, either from console or keyboard input, we are going to start using classes.

  • In Java to take user input using three classes: Console, BufferedReader, and Scanner. For now, we will deal with Scanner only.

  • First, we need to import the class, so using the keyword import we will invoke the Scanner class as follows:

    import scanner class

    • note that even in import we don't forget the semicolon :) . import scanner
  • Then, we create a new object from Scanner class

    creating scanner object

    • we created object of name input of Scanner class using new operator.

    Scanner class methods

  • importing statement is placed before class declaration.

  • simple Java program takes input from user to calculate the area of a circle: you can simply see the file here: ComputeCircleArea.Java

    circle area

  • so, at any point you want to take user input:

    1. import Java.util.Scanner;
    2. inside main method block of code Scanner input = new Scanner(System.in);
    3. based on what data you want to take, e.g. prompt user to enter integer: int number = input.nextInt(); and that's it.

Jshell interactive tool as Java REPL (Read, Evaluate, Print, Loop)

  • Read-Evaluate-Print Loop (REPL), which evaluates declarations, statements, and expressions as they are entered and immediately shows the results.

  • it's an interactive tool for quickly prototyping Java code, all without the need for a public static void main or the need to compile your code before executing it.

  • it's super easy open your cmd or powershell and type jshell

  • try simple declaring variable called hello, var hello = "welcome to jshell"

jshell

notes:

First Chapter is Finished!!!


2. Control Structure Part 1: (_Selections_)

2. Control Structure Part 1:

In this section you will learn the way you take decision.

According to documentation:

control flow statements, break up the flow of execution by employing decision-making, looping, and branching, enabling your program to conditionally execute particular blocks of code.

Types of Control Flow statements:

  1. Decision-making statements.
  2. Iteration (Looping) statements.
  3. Branching statements.

in this section we'll talk about Selections | Conditionals | Decision-Making statements.

Conditional keywords: if, else, else if,switch,case,break.

1. If statements:

Conditional statement evaluate a (condition) if it's true or false to perform set of instructions:

if condition

A boolean expression is expression will yield true or false, e.g. is 4 + 5 = 9? if their sum equal to nine so the answer is yes otherwise no. so, an expression will be given to be evaluated.

now, the question is ' How to build a conditional ' :

before anything let's learn about the second category you hear in this guide, relational operators: == <= >= != > <

note that assignment operator = is not the same as == equality operator, equality operator mean if the value of smth equals some value on the other hand the assignment operator assigns value to variable.

Conditionals evaluate boolean expressions, boolean expression are expression expected to have true or false as answer, so we are kinda building comparison statements.

if example

it's simply like saying: is dividing 7 by 10 equals 70? if yes please print the result ,and if 'not' do nothing.

  • Checking for alternatives: using else if we can check for alternatives, e.g. here is a simple program to check if a number is positive or negative or equal to 0:

    conditional example

  • you can find more examples in Test Files Elementary, e.g. addition Quiz

we can check using if then another if but what we hit the goal from first time why do we continue checking? that's why we use else if instead of to skip the rest of conditional statement checking if we already satisfied one.

notes:

  • any boolean expression is enclosed by parenthesis ().
  • if by mistake you placed a semicolon after if statement if ( y == x ); is like doing this: if ( y == x ){} means do nothing.
  • avoid redundant testing of boolean expression e.g. if (even == true) better do this if (even), so if even have a number other than zero it still evaluate to true or if even a boolean variable it will evaluate to its boolean value.
  • avoid equality test to float variables to avoid precision problem of floating point numbers.
  • you can simplify boolean variable assignment: do this

now we could build simple conditionals using of, else and if else. What if we want to check two conditions at the same time, can we? actually yes, using compound boolean expressions we can check two conditions at the same time, but first let's get familiar with the logical operators:

|| OR operator

&& AND operator

! NOT operator

^ XOR operator

if you are a computer science, or electrical engineering student you are already familiar with these operators as you took this knowledge in Logic design course -probably it's name nearly like this-, if you or not here's a simple explanation ,and I'll provide you a link for a video to simplify the concept more:

OR operator || check if one of the expressions at least is true or both of them. e.g. (5+6 == 11 || 5+7 == 11) evaluates to true because the first expression is true. it will only evaluate to false if both of them are false.

AND operator && check if both of the expressions must be true if one of them is false so the whole evaluation of both of them will be false.

NOT operator ! is to negate any value to oppose any expression in other words: !(expression) means to flip the evaluation value of this expression if it's evaluated to true the not operator will make it false and so on. same when we use != we mean not equal.

XOR operator ^ is used to check if both expressions evaluates to different answers, simply if both of the expression is evaluated to true or both false it will return to us false, if they are different (true , false or false , true) return to us true.

we can use DeMorgan's law to best practise these compound boolean expression conditionals

!(condition1 && condition2) == ! condition1 && ! condition2 using DeMorgan's.

note && and || are called lazy operator as they perform short circuit concept as if false && anything evaluates to false without checking the second expression same in true || anything will evaluate to true without checking the second expression.

1. Switch case statements:

we use switch case statements with menu-like - means many choices - problems, e.g. choosing operation to be done on numerical values, based on day of week chosen ;perform some action:

switch case flow chart

switch case syntax

This is a simple program perform basic operations on the entered numbers: you can check it from here switch program

switch case example

  • if break is not used, switch will evaluate all other cases, we use default if the expression doesn't match any case.

lastly we will learn about the boolean ternary operator (boolean expression)? (if true)expression1 : (if false)expression2 this operator return exp1 if it evaluates to true and exp if it evaluates to false, e.g. :

ternary operator

notes:

Second Chapter is Finished!!!


3. Mathematical Functions,Characters, and Strings

3. Mathematical Functions,Characters, and Strings

In this section you will learn about some built-in mathematical functions in addition to introducing two data types (Character and String).

lets breakdown the concepts we are going to learn:

  • Methods
  • Important built-in methods
  • example using these methods
  • Character datatype
  • Operations
  • Casting
  • String Reference Type
  • Important methods for dealing with Strings
  • Java String pool

before we go in methods with more details on the next Chapter, we can have quick definition help us to figure out what these block of codes mean.

Methods can be used to define reusable code and organize and simplify coding, and make code easy to maintain.

We also refer to methods as routines and other words will be mentioned later, for now remember the word "reusable".

note: if you experienced other languages other than Java you may find that methods are called functions, but with a programming language which is attached to Object-Oriented concepts you'll find the word 'method' is the convention here.

methods can be:

  1. user defined
  2. built-in

Before we go deeper in this

Why do we need methods?

Coffee Shop analogy:

coffeeShop

Imagine we have the following situation:

Imagine you run a coffee shop. When customers come in and order coffee, the barista follows specific steps to make the coffee. If you have many customers, repeating these steps for each order can be exhausting and time-consuming

Here is the ongoing scenario:

  1. 10 customers order coffee.
  2. The barista follows the same steps for each coffee order.

Identifying the problem: steps repetition for every customer in addition to time consumed

Offered Solution: Introducing a coffee machine

To improve efficiency, you introduce a coffee machine. The machine has buttons for different types of coffee. Now, the barista simply presses a button, and the coffee machine follows the predefined steps to make the coffee.

Analogy to Methods:

  • Without Methods: The barista repeats the same steps (code) for each customer.
  • With Methods: The barista uses the coffee machine (method) to make coffee efficiently.

Java Code Simulation:

Without Methods (Exhausting Process)

without methods

With Methods (Effective Process)

with methods

Analysis to what happened:

  • DRY Principle: By using methods, we encapsulate the repetitive steps of making coffee into a single method makeCoffee. This follows the DRY principle by avoiding repetition of the same code.

  • Built-In vs. User-Defined Methods:

    Built-In Methods: These are like the common coffee machines available in the market. They perform common tasks and are part of the Java Standard Library. For example, System.out.println is a built-in method used to print messages to the console.

    User-Defined Methods: These are like the custom coffee machine designed specifically for your coffee shop. They perform specific tasks defined by the programmer. In our example, makeCoffee is a user-defined method that encapsulates the steps to make coffee.

For now in this chapter we're going to discuss built-in methods only.


more about built-in functions

easy definition for these functions found on NinjaOne blog:

Built-in functions refer to those pre-defined functions that come along with a programming language. They are intrinsic components of the language’s library, ready to be used without requiring any additional installation or importation. These functions have been designed to perform common tasks, thus facilitating efficient and swift coding.

for short: built-in functions or methods are pre-defined methods in the language library itself which are already implemented for you to perform common tasks.

now lets getting familiar with them; first, I prefer to categorize them, so it's easily inside my head to choose the suitable one for the suitable situation:

We'll use Math class for this mission, Math class in Java contains all mathematical constants and methods you'll need to use, according to Oracle documentation:

The class Math contains methods for performing basic numeric operations such as the elementary exponential, logarithm, square root, and trigonometric functions.


Built-in Functions in Math Class:

  1. Trigonometric Methods
  2. Exponential Methods
  3. Service Methods (rounding, min-max-abs, random)

1- Trigonometric Methods:

Method Description
sin(radians) Returns the trigonometric sine of an angle in radians.
cos(radians) Returns the trigonometric cosine of an angle in radians.
tan(radians) Returns the trigonometric tangent of an angle in radians.
toRadians(degree) Returns the angle in radians for the angle in degrees.
toDegrees(radians) Returns the angle in degrees for the angle in radians.
asin(a) Returns the angle in radians for the inverse of sine.
acos(a) Returns the angle in radians for the inverse of cosine.
atan(a) Returns the angle in radians for the inverse of tangent.

2- Exponential Methods:

Method Description
exp(x) Returns e raised to power of x (ex).
log(x) Returns the natural logarithm of x (ln(x) = loge(x)).
log10(x) Returns the base 10 logarithm of x (log10(x)).
pow(a, b) Returns a raised to the power of b (ab).
sqrt(x) Returns the square root of x (2x) for x7 =0.

3- Service Methods:

  • Rounding Methods
  • The min, max, and abs Methods
  • The random()Method

Rounding Methods

The best way to understand rounding methods in Math class in Java is by example:

ciel method

floor method

rint method

round method

The min, max, and abs Methods

  • The min and max methods return the minimum and maximum numbers of two numbers

  • The abs method returns the absolute value of the number

Here is a worked example:

min max abs methods

The random() Method

In my own opinion random method is super important in a lot of situations, for example in programs used for making test models, in the preceding chapters and projects you'll figure this out

This method generates a random double value greater than or equal to 0.0 and less than 1.0

random Method

Simple Application !

angle between two vectors there is a little application "Calculate Theta value" calculates the value of the angle between any two vectors using dot product you can take a look on the application here, whereas you can find how can we use Math Class built-in functions are used. Calculate Theta

For ,now we addressed the following:

  • Methods
  • Important built-in methods
  • example using these methods

Character data type and String reference type:

Character Data Type:

Character data type is used to represent single character surrounded by single quotes. char firstLetter = 'S'

Caution ⚠

characters must be surrounded by single quotes while Strings must be surrounded by double quotes, thus "S" is a String and 'S' is a character my this is not important in other languages like: Python but here it's essential to differentiate between them as Java s a strongly static typed language

If you are not a computer science student ,or you are not familiar with the idea of how computers deals with data, so expand this section, if you are familiar go to the next points


How computers deal with data

How computers deals with data?

You need to know what machine language is. Machine language is the language that computers actually understand. This language consists of 0's and 1's based on the fact that computers, at their core, consist of transistors that can be switched to two states: ON or OFF (1 or 0). From this, the binary system was born.

zeros and ones

The processes are carried out in a central processing unit (CPU), which performs the necessary mathematical operations for processing incoming data. It handles all computing tasks required for running the operating system and applications.

CPU

Anything represented in a high-level language, like "Human Language," is too complex for computers to understand directly. This is where the role of the CPU and other computing components comes in, translating this data into something understandable to the computer, known as "machine language."

Similarly, computers don't understand characters directly; they need a translator. This can be a lengthy process that we won't delve into deeply here, but the main idea is conveyed.

Computers only understand binary data, binary code, or machine language. They represent everything (text, images, sound, etc.) using binary.

Today, no one deals with 0's and 1's directly. This level of operation is abstracted within the computer hardware.

Here are videos will help you so much for understanding the topic:

A good article about how data is stored and how computers deal with it: How data is stored in computers

Since you may not be familiar with computer science, I'll recommend a superior playlist. The videos are relatively short and very informative: Crash Course computer science

  • note You can watch this playlist to get an idea of what computer science is and how it has evolved over the decades. It will help you pick up keywords, history, important names, applications, etc. However, it's not for in-depth study.

  • Watch the playlist like your daily series; you'll learn amazing things about this field that has revolutionized humanity in the last decade!

Another one, but this time you cannot just watch and enjoy, also I want to inform its lectures are long, but super informative recommended from almost everyone ,and it's considered as one of the most common and best introductions to computer science, CS50 - from Harvard CS50

Note: You don't need to master this topic to understand character data types. Just grasp the main idea. If you want to delve deeper into computer science, these playlists and courses are great resources. These playlists take time, and it's okay to keep learning without finishing them first. You can do both in parallel.

I address the problem of getting bored due to the huge and often complicated information for the first time to understand. When you get bored and decide to stop, you risk losing out on important knowledge. You may end up with incomplete knowledge about a language that could be crucial for your future job. Instead, you could spend that time learning something enjoyable, like how to bake the perfect cinnabon! 🥯😂


Characters and Encoding

  • Computers use binary numbers internally. A character is stored in a computer as a sequence of 0s and 1s.

  • Mapping a character to its binary representation is called encoding. (encoding is converting data from one format to another)

  • There are different ways to encode a character. How characters are encoded is defined by an encoding scheme.

  • Java supports Unicode, an encoding scheme established by the Unicode Consortium to support the interchange, processing, and display of written texts in the world’s diverse languages.

  • Before Unicode, ASCII encoding scheme were used; a standard data-encoding format for electronic communication between computers encoding scheme, it was 7 bits (every bit can represent either one or zero) so it could hold only 128 characters English characters with some punctuation and control features then it was expanded to be 8 bits to hold 256 characters.

  • other encoding schemes are widely used today like UTF-8 which is algorithmic mapping in addition to providing backward compatibility with ASCII.

Character Encoding Analogy! Teaching Colors with a Numbered Painting Sketch

If we want to teach a child how to get used with colors and their names we use the Numbered Painting Sketch

  • Encoding: in the context of The Numbered Painting Sketch, Encoding is like the painting sketch where each sector of the drawing is assigned (encoded to) a unique number (1 for red color, two for green, 21 for mint green, etc.).

numbered painting sketch

  • Just as each sector in the sketch has a number to indicate which color to use, encoding assigns a unique number to each character so that it can be stored and identified correctly.

  • encoding schema or Unicode: The Color Table

    color table

    Unicode is like the color table that shows which number corresponds to which color.

    The table ensures that no matter what color you need, there's a specific number assigned to it.

    Similarly, Unicode assigns a unique code point to every character in every language, ensuring a universal standard.

  • UTF-8: The Flexible Coloring Kit

    UTF-8 is like a coloring kit that adapts to the complexity of the drawing.

    • In the coloring kit: Simple, common colors (like "Red") might be represented by one marker.

    • More complex, less common colors might be represented by more markers.

    • This makes the kit efficient and flexible, capable of handling all sorts of drawings (characters) without wasting resources.

    blending colors


Now we can introduce the data type :

you define the character data type using char keyword, then give it a name and assign a character value to it.

Caution⚠ don't forget that characters must be surrounded by single quotes ' '.

Why did we introduce the character encoding with all this staff? to understand the following

characters

Here you can see that character data type can store numeric value and weird code starts with \u but how?

In Java, the char data type can store numeric values, but these values are not interpreted as numbers by Java. Instead, they are treated as their corresponding ASCII or Unicode characters. For example, the character 'a' has an ASCII value of 97. In Unicode, the same character is represented as \u0061.

note The increment and decrement operators can also be used on char variables to get the next or preceding Unicode character. For example, the following statements display character b:

character operations increment

also, numerical operations are allowed ,but you'll get numeric output

char operations

Escape Sequences for Special Characters

if you know already C you maybe familiar with these characters

Escape characetr is a character preceded by a backslash \ is an escape sequence and has special meaning to the compiler.

from Oracle Java tutorial:

When an escape sequence is encountered in a print statement, the compiler interprets it accordingly. For example, if you want to put quotes within quotes you must use the escape sequence, \", on the interior quotes. To print the sentence

She said "Hello!" to me. you would write:

escape sequence

Character Name
\b Backspace
\t Tab
\n Linefeed
\f Form feed
\r Carriage Return
\\ Backslash
\" Double Quote

Don't worry if you are not familiar with some terms like "carriage return" here is clarification:

  • Carriage return means to return to the beginning of the current line without advancing downward. (Abbreviated CR)

  • Linefeed means to advance downward to the next line (Abbreviated LF or NL) CRLF is used for the pair "\r\n".

  • Formfeed means advance downward to the next "page".


Casting

Now let's move to new topic (Casting) which will be expanded later when you got familiar with OOP.

Casting : is a process that converts a variable's data type into another data type, casting may be implicit (auto) or explicit.

Characters (char) can be converted to numerical data types because characters are internally represented by numeric values according to the Unicode standard, thus character data type can be cast into any numeric type and vice versa.

  • note 1: character data type is 2 bytes so only the lower 16 bits of data are used when casting from numeric into character

  • note 2: if you cast a floating number data to character, first the float data will be cast into integer then character

casting one

When a char is cast into a numeric type, the character’s Unicode is cast into the specified numeric type.

casting two

⚠ note Implicit casting can be used if the result of a casting fits into the target variable. Otherwise, explicit casting must be used.

  • All data type reserves different amount of space in memory, so it's important to take this in consideration when casting, char is 2 bytes, boolean is one byte while integer is 4 bytes.

  • Two characters can be compared using the relational operators just like comparing two numbers. This is done by comparing the Unicode of the two characters.

    • for example:

    comparing characters

taking the advantage that we got familiar with built-in functions, let's use the built-in functions which can be used to perform some tests on characters:

Method Description
isDigit(ch) Returns true if the specified character is a digit.
isLetter(ch) Returns true if the specified character is a letter.
isLetterOrDigit(ch) Returns true if the specified character is a letter or digit.
isLowerCase(ch) Returns true if the specified character is a lowercase letter.
isUpperCase(ch) Returns true if the specified character is an uppercase letter.
toLowerCase(ch) Returns the lowercase of the specified character.
toUpperCase(ch) Returns the uppercase of the specified character.

note to use character built-in function, it's like Math class you have to import Character Class as follows:

Import Character class

Character data type is finished!!!


String reference type

A string is a sequence of characters.

"Hello, World!" >>> the most popular string ever in programming world!

Strings are enclosed by double quotes " ".

again and again characters are surrounded by single quotes ,and it's a must ,or they will be considered as strings

note : Strings are immutable 'you cannot change their value' ⚠

Reference Types in Java

In Java, there are two main categories of data types: primitive types and reference types.

Primitive Types: These include int, char, boolean, etc. They store simple values.

Reference Types: These include objects like Strings, arrays, and custom classes. They store references (or addresses) to the actual data in memory.

In other words, Any class can be used as a reference type, and any variable declared using this reference type is called reference variable or an object.

  • For now, the class is like a template and the object is the custom version you make from this template.
  • Objects and Classes will be discussed in details LATER :)

So, like Scanner class you have to follow the naming convention and capitalize the first letter of the class name:

 ```Java
 String sentence = "Hello, Developer!"; 
 ```

STRINGS IS A SUPER IMPORTANT TOPIC IN ALMOST ALL AREAS OF PROGRAMMING

SO, KEEP YOUR FOCUS ON THIS TOPIC

String Operations

Now we are going to discuss the most important concepts that are necessary to know about strings:

  • How to get the length of a string
  • Concatenating Strings
  • Trimming Strings
  • Find character/s | substring/s in a given string
  • Comparing Strings
  • Converting and reading strings as user input.

For short this is a table containing all the needed methods to perform the mentioned tasks:

Method Description
length() Returns the number of characters in this string.
charAt(index) Returns the character at the specified index from this string.
concat(s1) Returns a new string that concatenates this string with string s1.
toUpperCase() Returns a new string with all letters in uppercase.
toLowerCase() Returns a new string with all letters in lowercase.
trim() Returns a new string with whitespace characters trimmed on both sides.
equals(s1) Returns true if this string is equal to string s1.
equalsIgnoreCase(s1) Returns true if this string is equal to string s1; it is case insensitive.
compareTo(s1) Returns an integer greater than 0, equal to 0, or less than 0 to indicate whether this string is greater than, equal to, or less than s1.
compareToIgnoreCase(s1) Same as compareTo except that the comparison is case insensitive.
startsWith(prefix) Returns true if this string starts with the specified prefix.
endsWith(suffix) Returns true if this string ends with the specified suffix.
contains(s1) Returns true if s1 is a substring in this string

1- Getting String Length

  • you'll use strings almost for everything, getting a string length is important if you are going to iterate over this string for any task or to check if the returned data is matching from their lengths.
  • an instance method length() returns to you the length of the given string.
  • here comes another question, What is an instance method?
    • till now, we've passed by static and instance methods, static methods like Math.rint(), Math.sin()... , length()method here is the first instance method we meet.
    • regarding mentioning classes and objects earlier without details, for now instance methods are methods tied to a specific object (relies on specific data) and static methods are methods which are not tied to a specific object they only cling to the class itself that's why you invoke them using the class name itself and not the reference variable (object).
    • clarification:
      • Think of a school where students and classrooms play important roles:

        • Instance Methods: Imagine each student has their own notebook. The notes in these notebooks are personal and unique to each student. For example, a student's grades or personal notes are tied to that specific student. In the same way, instance methods in a class belong to specific objects (or instances) of that class. These methods can access and modify the instance variables of the object they belong to.
      • notebooks

        • Static Methods: Now, think about the whiteboard in a classroom. The whiteboard is a shared resource that any teacher can use to teach any student. The information on the whiteboard is not tied to any specific student; it can be used by all students collectively. Similarly, static methods belong to the class itself, not to any particular instance. These methods can be called without creating an instance of the class and cannot access instance variables directly. Instead, they work with class-level data.

        board

        • this is the difference noticed when you use static methods of Math class like round(), as rounding number doesn't rely on specific data, but length() from String class is tied to specific object data, in our case here sentence string above, you return the length of this specific string object.

you can check this example here about how to use length() method Check Password Length example

check password


Learn by example:

we're going to explain each method and common use cases by examples in addition to clarifying any ambiguity:

  • charAt(index) method

    let's clarify a point, string as mentioned earlier is a sequence of characters, people who already familiar with programming or already tried C/C++ knew that string is an array of characters, each character is indexed.

    The (string variable).charAt(index) method can be used to retrieve a specific character in a string s,where the index is between 0 and (string variable).length()–1.

    note ⚠ a common programming error "StringIndexOutOfBoundsException." if you tried to access character which is our of array bounds as we count from zero, so charAt(string.length()) will expose you to this error :)

     String message = "Welcome to Java";
    
     System.out.println(message.length());
    

    Java indexed string

To Someone Totally New to Programming
  • Imagine you have a row of lockers, and each locker has a number on it, starting from 0. These numbers help you find and open a specific locker quickly.

  • A string is like a row of lockers, but instead of holding things, it holds characters (letters, numbers, symbols, etc.). Each character in the string has a number called an index, which tells you its position in the row.

For example, in the string "Hello":

  • The first character 'H' is at index 0
  • The second character 'e' is at index 1
  • The third character 'l' is at index 2
  • The fourth character 'l' is at index 3
  • The fifth character 'o' is at index 4

So, if you want to find or use a specific character, you can refer to its index number, just like you'd find a specific locker by its number.

note in programming by default we count from 0 index

example on using charAt(index) method: email validation

charAt()

  • concat(s1) method:

    concatenation is one of the most important concepts you need to learn about strings, concatenation means merging two strings together <3

    concatenation is achieved using multiple ways, one shortcut is by using plus sign '+' between two strings as follows:

      ``` Java
          String name = "Sarah";
          String greeting = "Hello, " + name;
          System.out.println(greeting);
          //prints >>> Hello, Sarah 
      ```
    

    also += operator can be used with strings:

       ```Java 
          String studentsNames = "";  //empty string
          studentsNames += input.next();
          // enetrs Mariam 
          System.out.println(studentsNames);
          // prints Mariam as the empty string is concatenated with the user input "Mariam"
      ```
    

    using s2.concat(s1) method:

    ```Java
    
         String spongepopFandomgreeting = "Hi Hi,"; 
         String name = "Captain";
         spongepopFandomgreeting = spongepopFandomgreeting.concat(name);
         System.out.println(spongepopFandomgreeting);
        //prints Hi Hi, Captain
    
    ```
    

    note the reassignment happened above with spongepopFandomgreeting because the method concat(s1) returns a string ,so it must be stored in string variable or update an existing string variable.

    you can concatenate strings with any data type resulting a new string with all the concatenated values

    string concatenation

  • trim(), toUpperCase(), toLowerCase() methods:

    The toLowerCase() method returns a new string with all lowercase letters, and the toUpperCase() method returns a new string with all uppercase letters. For example:

     ``` Java
         String convertToLower = "Welcome".toLowerCase(); // returns a new string **welcome**.
         String convertToUpper ="Welcome".toUpperCase(); //returns   a new string **WELCOME**.
      ```
    

    The trim() method returns a new string by eliminating whitespace characters from both ends of the string. The characters ' ', \t, \f, \r, or \n are known as whitespace characters. For example,

    ``` Java        
        String trimmedSentence = "\t Good Night \n".trim(); //returns a new string Good Night.
    ```  
    
  • equals(s1) & equalsIgnoreCase(s1) methods:

    Let’s examine the following examples:

    example using == operator

    example using equals() method

    What Happened?

    This introduces the concept of the String Pool in the heap.

    From the reference used to formulate part 1:

    the == operator checks only whether string1 and string2 refer to the same object; it does not tell you whether they have the same contents. Therefore, you cannot use the == operator to find out whether two string variables have the same contents. Instead, you should use the equals() method.

    The reference illustrated the concept straight forward, let me highlight that:

    • "the == operator checks only whether string1 and string2 refer to the same object; it does not tell you whether they have the same contents"

    • let us dig deeper, what does that mean?

      • This is where the String Pool concept comes into play.

      • The String Pool is a special area in the heap memory where Java stores string literals. When you create a string using a literal (e.g., String s = "hello";), Java first checks if an identical string already exists in the pool. If it does, the existing reference is used; otherwise, a new string is added to the pool. due to the immutability characteristic of Java Strings what happens in string pool ensures security, reusability, and memory management. But how does it work?

      • Reference types in Java store a reference to an object, not the actual value. If you're familiar with C, you can think of it like a pointer, but let's clarify further:

        reference data types

      • When you change the value of reference type as follows:

           String greeting = "Hello";
           greeting = "Hi";
           System.out.println(greeting); //prints Hi
      • You’re not changing the value of greeting directly; instead, you’re changing the reference to another object that has the value "Hi".

      String pool example

      String pool example 2

      • As you observed in the figures, you just created another string literal has the value of "Hi"

      How security can be achieved here in this situation:

      • magine you have multiple string objects with the same value "default" as follows:

           String str1 = "default";
           String str2 = "default";
           String str3 = "default";
      • They all point to the same value in the string pool

        changing references

      • What if one of them have been changed its value to "def" for example?

        • Now you know the answer! the value hasn't been changed the string reference variable just has been changed its reference to another object in the pool :)

        changed reference

        • Here, security is ensured. For instance, imagine a situation which multiple threads use the same string literal, and one thread tries to change the value, the others won’t be affected because the string reference variable used by the thread just changes its reference without affecting the others.
        • Reusability and memory management are also ensured. The JVM first checks if an identical string already exists in the pool. If it does, the existing reference is used; otherwise, a new string is added to the pool. This prevents memory waste for the same literal while maintaining security due to string immutability.
      • In case of using new operator, here you create string object outside of the string pool, in the heap:

      new operator

      • The new operator creates an object outside the String Pool, so even if you create multiple objects with the same exact string literal using the new operator, they won’t point to the same object, as seen in myName.
  • str1.compareTo(str2) is a method compares two strings lexicographically.

    • What's meant by lexicographically?

      • The comparison is based on the Unicode value of each character in the strings.
    • The method returns value less than zero if str1 has fewer characters or considered less than str2 comparing the Unicode value of their characters, returns more than zero if the opposite and 0 if they are equal.

    • to understand observe the next example:

      comparing strings

    • try it out:

      ```Java
          String greeting = "Hello";
          String greeting2 = "Hello";
          System.out.println(greeting.compareTo(greeting2)); //prints 0
          String salut = "Hi";
          String salut2 = "Hola";
          // prints -6 because letter i unicode value is less than o by 6 characters
          System.out.println(salut.compareTo(salut2)); 
          String welcome = "welcome";
          String welcome2 = "welco";
          //prints +2 because of the two characters excess
          System.out.println(welcome.compareTo(welcome2));
          //if we inverted the strings prints -2 because of the two characters are missing
          System.out.println(welcome2.compareTo(welcome));
      ``` 
      
  • str1.compareToIgnoreCase(str2) does the same job but here the case won't be considered :

    comparing strings ignore case

  • startsWith(prefix), endsWith(suffix), contains(s1) methods:

    • startsWith(prefix)Returns true if this string starts with the specified prefix.
    • endsWith(suffix) Returns true if this string ends with the specified suffix.
    • contains(s1) Returns true if s1 is a substring in this string.

    substrings

Obtaining Substrings

  • You can obtain either a character or a substring from a string or return their indices:

    Method Description
    substring(beginIndex) Returns this string’s substring that begins with the character at the specified beginIndex and extends to the end of the string, as shown down in the Figure
    substring(beginIndex,endIndex) Returns this string’s substring that begins at the specified beginIndex and extends to the character at index endIndex – 1, shown down in the Figure. Note the character at endIndex is not part o f the substring
    indexOf(ch) Returns the index of the first occurrence o f ch in the string. Returns −1 if not matched.
    indexOf(ch, fromIndex) Returns the index of the first occurrence of ch after fromIndex in the string. Returns −1 if not matched.
    indexOf(s) Returns the index of the first occurrence o f string s in this string. Returns −1 if not matched.
    indexOf(s, fromIndex) Returns the index of the first occurrence of string s in this string after fromIndex. Returns −1 if not matched.
    lastIndexOf(ch) Returns the index of the last occurrence e of ch in the string. Returns −1 if not matched.
    lastIndexOf(ch,fromIndex) Returns the index of the last occurrence of ch before fromIndex in this string. Returns −1 if not matched.
    lastIndexOf(s) Returns the index of the last occurrence of string s. Returns −1 if not matched.
    lastIndexOf(s, fromIndex) Returns the index of the last occurrence of string s before fromIndex. Returns −1 if not matched.
  • example on indexOf(ch) method

example on indexOf method

Conversion Between Strings and Numbers:

using parseInt() method from Integer class you can convert from string to integer, in other words parsing integers from a string.

  ```Java
      int parsedInteger = Integer.parseInt("2351");
      System.out.println(parsedInteger); // 2351 
  ```

you can use parseDouble() from Double class also for the same matter but for parsing floating point numbers:

  ```Java
     double parsedDouble = Double.parseDouble("34.223");
     System.out.println(parsedDouble); // 34.223
  ```

If the string is not a numeric string, the conversion would cause a runtime error:

  ```Java
    int numberFromAString = Integer.parseInt("Iloveyou3000");
    System.out.println(numberFromAString); // runtime error (throws an exception) as the string is not numeric it has some characters
  ```

note: you don't have to import Integer or Double classes because they are included in Java.lang library thus they are imported by default

you can convert any numerical value to a string by just concatenate it with a double quotes as follows:

  ```Java
  int number = 345;
  String numericalString = number +"";
  System.out.println(numericalString);   // 345
  ```

Formatting Console Output:

Good news for these who loves c/c++, you can use the System.out.printf method to display formatted output on the console just like printf in c:

```Java
  double amount = 12618.98;
  double interestRate = 0.0013;
  double interest = amount * interestRate;
  System.out.printf("Interest is $%4.2f", interest);
```

For those who are new to the idea, you need to get familiar with what a format specifier mean:

format specifier specifies how an item should be formatted, simple format specifier consists of a percent sign (%) followed by a conversion code.

The following is a table of format specifiers used in Java:

Format Specifier Output
%b formats boolean value
%c formats character
%d formats integer
%f formats floating point number
%e formats scientific notation
%s formats string

Observe in the next example how do we use the format specifiers in formatting an output as follows:

format specifiers

note Items must match the format specifiers in order, in number, and in exact type otherwise a runtime error will occur.

Another important point is considering that we are formatting console output, so in some cases we need to display leading zeros or adding thousands separators ,or even we don't need to display all these numbers after the floating point.

Controlling the width and precision in a format specifier, helps in achieving what were mentioned above:

components of a format specifier

you can add leading zeros but with integer conversion character d as follows:

``` Java
    int thousands = 52497823;
    System.out.printf("%010d", thousands); // 0052497823
```

you can alter the leading zeros flag with the thousands flag or left justifying flag - and so on.

  • What if the field width is less or more than the item digits or characters, if the field width is smaller than the item it will be automatically increased but if it's more than the item digits or characters spaces are added before them:

spaces 1

spaces 2

you can see more about formatting in these links on Oracle website: Java tutorial

Finally, the third chapter is FINISHED!!! I THOUGHT IT WAS A CURSE, WASN'T IT?


4. Control Structure Part 2: (_Iterators_)

4. Control Structure Part 2: (Iterators)

Remember from the first chapter about Control Flow Statements, that controlling the execution flow of a code block can be done with the help of looping and branching.

Also, we've mentioned the types of control flow statements branching and Iteration statements.

In this Chapter we're going to learn about Iteration (loop):

  • Consider a situation where you need to print scores of a specific academic year of 200 students, logically writing 200 print statements which almost print the same sentence except the name of student and his/her score would be a super exhaustive task.
  • Regarding what you've learnt from this section in the last chapter DRY principle, you may think of using methods; but is that really the suitable choice for this task? for short, no.
    • you can use methods to encapsulate the right structure.
  • A control structure in which you can perform a number of tasks or execute a block of code repeatedly is called Loop.

Loops (Iterations):

  • A loop is a control structure which controls how many times an operation or a block of code can be executed.

  • Types of Loops:

    • Counter Controlled
    • Sentinel Controlled

types of loops

  • Simply, the difference between the counter controlled and sentinel controlled is that you know the number of iterations in advance while in sentinel loops you don't know how many iterations will be actually executed because it depends on other action like user input for example.

Loop General Structure:

  1. Loop-Continuation-Condition
  2. Loop-Body
  3. iteration

Observe the following figure:

loop structure

  • The loop-continuation-condition is a boolean condition in which ,if it was evaluated to true so the loop continues otherwise 'false' the loop terminates.

    • The boolean condition controls the execution of the loop thus adjusting this condition is super important.
    • You can be exposed to off-by-one error (one more or less iteration) if you didn't adjust this condition for example:
          public class Main{
              public static void main(String[] args) {
                
                //this program is for printing Hello,World 10 times
                for(int i =0; i<=10; i++ )
                    System.out.println(i+1 + "- Hello,World");
                    // the sentence will be printed 11 times! off-by-one 
              }
            }
    • This kind of error is logical or semantic errors, the code actually works fine but the code doesn't yield the right output.
  • The loop-Body is the block of code which contains the statements to be repeated.

  • Executing the Loop-Body successfully one time is considered as one iteration.

  • Loop Design strategy:

    There is two keys to design a loop:

    1. Identify the code needed to be repeated
    2. Identify what can terminate the loop
  • Types of loop statements in Java:

    • pretest loops
      • for loop
      • while loop
    • post-test loops
      • do-while loop

Types of Loop statements in Java

  1. The Pretest Loops: loops which already test the condition first before the execution:

    • For loop:

      • for loops have concise syntax:

      For Loop Java Syntax

      • try it out ! replace each statement with actual code.

        for(`initial action`; `continuation-condition` ;`action-after-each-iteration`){
          block of statements to be repeated
        } 
        
      • the statements between the parenthesis after for keyword construct the control structure of the loop.

      • the initial action is the loop control variable initialization.

      • the control variable controls the number of iterations in the loop.

      • the action-after-iterations is the statement which adjusts the control variable.

      • each statement of the loop control structure is terminated using a semicolon ; like the rest of the statements in Java

        • note⚠ you can insert a semicolon by mistake after the control structure itself inside the parenthesis which results in loop termination ,or it means do nothing as follows:

          for(int i = 0; i<10;i++);   //<----- note the semicolon after the closing parenthesis here will case to print nothing as the loop has been terminated.
          {
            System.out.println("Hello);
          } 
          
      • examples of using for-loops:

        1. Simple program to print students name followed by their IDs, the IDs start from 2020000 to 2020009:
            public class Main{
                public static void main(String[] args) {
                    //This program is for printing the students IDs:
                    int initialID = 202000;
                    String[] studentNames = {"Reem","Ali","Maged","Mariam","Mary","Ahmed","Mohammed","Mustafa","Shehzad","Sally"};
                    for(int i = 0; i<10; i++){
                        System.out.printf("Student's name: %s , student ID: %d \n ",studentNames[i],initialID+i);
                    }
                }
            }  

      loop example 1 2. Simple program to print the multiples of a number entered by user and stop at of before 100:

      import Java.util.Scanner;
      public class Main
      {
          public static void main(String[] args) {
              // This program is for printing the multiples of any number entered by user till 100:
              // prompt user to enter a number
              System.out.println("Please enter a number");
              // define the Scanner Object
              Scanner input = new Scanner(System.in);
              int number = input.nextInt();
              int multiple = number;
              // input.nextInteger() is token based so a \n must be inserted or the next print lines 
              // will be right beside the taken input -confusing-
              System.out.println();
              // note that you can statements separated by comma in the initial action to initialize 
              // variables or control variable but it's more common to intialize control variables
              // and the same in after-iteration-action statement,you can add statements
              // you want to be executed after each iteration
              for(int i = 0; multiple <= 100; i++,multiple+=number){
                  System.out.printf("Multiple number %d of number %d is %d \n ",i+1,number, multiple);
              }
          }
      } 

      loop example 1

    • While loop:

      • A while loop executes statements repeatedly while the condition is true. Its syntax is simple:

           ``` Java
          
             while (loop-continuation-condition) {
               // Loop body
               Statement(s);
             }
           ```
        
      • While loop general syntax:

      while loop general syntax

      • while loop strcture is a little different from for loop structure, observe the following example:

      while loop example

      • before the loop two variabels were declared and initialized sum & i (the counter).
        • in contrast with for loop, in while loop you declare and initialize the counter outside the loop and use it inside the loop body.
      • the boolean expression i<10 specifies that continue iteration if i is smaller than 10, otherwise; terminate the loop.
      • inside the loop body we have two statements:
        • the first statement commulate the sum
        • the second one is incementing the counter -or we will experience an infinite loop-
      • then afer the loop termination the next statements in your code will be executed.
      • while loop can be used for both: sentinel and counter controlled loops, in the previous example we used a counter variable i to control the loop thus we knew the number of iterations in advance.

      while loop can be used for both

      • the following example illustrates a sentinel controlled loop using while:

        example number 2 while loop

      • the loop terminates based on user input, if the user entered the correct answer, the loop will terminate, otherwise; continues.

      • using random built-in function to generate random numbers, notice the casting before using the function (int) , multiplying by 10 to move the range, because random generates numbers between 0 and 1 exlcusive.

      • a scanner object is created to take user's input then a while loop is crafted to recieve the user answer for the question.

      • if the user entered the right answer, it termintes and then the next line in the code executes.

      • you can check a similar example but with grading system here: ItertativeAdditionQuiz.

  2. The Post-test Loops: loops which loop at least one time before testing the condition:

    • Do-While Loop:
      • do-While loop is a variation of while loop but here the body of the loop is executed first before testing the condition.

      • Use a do-while loop if you have statements inside the loop that must be executed at least once.

      • do-while syntax:

        do while loop syntax

      • try it out!

        do {
          // Loop body;
          Statement(s);
        } while(loop-continuation-condition);   
        
      • One important thing to notice about do-while loop is the semicolon after the closing parentheses of the loop-continuation-condition.

      • examples:

        • Menu example, when you prompt the user to choose from a menu, the menu must be executed at least once if the user chose to exit from the first time. you can see that in the following example:
          • note that it's just an example to clarify the concept of while, due to the missing functionalities like opening other files as it's considered an advanced topic currently.
          • you can check it out at DoWhileLoopTestExample.

        do while loop test example

        • Guessing number example:

        guess number example

Which loop to use:

Generally, you would use a for loop when you know the number of repetitions in advance, such as when you need to display a message a hundred times. A while loop is appropriate when the number of repetitions is uncertain, like when reading numbers until the input is 0. If you need the loop body to execute at least once before checking the continuation condition, a do-while loop can be used instead of a while loop.

Nested loops:

Simply a loop inside another loop are considered as nested loops.

Nested loops consist of an outer loop and one or more inner loops. Each time the outer loop is repeated, the inner loops are reentered, and started anew.

"from Introduction to Java Programming and Data Structures Comprehensive Version 12th Edition by Daniel Y. Liang"

  • Nested loops often used to loop over something has more than one dimension like tables, 2D arrays, sorting and so on. you need to keep track of the columns and rows in the same time.

  • One common example of using nested loops is printing patterns:

    nested loop pattern example

  • try it out!

  • ``` Java
    public class Main{
      public static void main(String[] args) {
        int rows = 5;
            for (int i = 1; i <= rows; i++) {
                for (int j = 1; j <= i; j++) {
                    System.out.print("* ");
                }
                System.out.println();
            }
      }
    }
    ```      
    
  • note: using nested loops are computationally expensive (time complexity O(n^2)), so they are avoided as much as possible but some algorithms are built using nested for loops.

Notes about Loops in Java:

  • Avoid using floating-point numbers in the continuation condition because of the precision of the floating numbers as floating-point numbers are represented in approximation in computers by nature:

    • An example illustrated from the same textbook mentioned above explained the precision problem caused because of using floating-point numbers in the continuation condition in addition to the statements inside the loop body itself:
    • The exact sum should be 50.50 but the loop output is 50.499985

    why you have to avoid using floating point numbers

    • If double data type is used instead of 'float' will improve the output a little but not the best.
  • Using break and 'continue' keywords:

    • They add more control over you loop ,but they are not secure so overusing them is not recommended.
    • break is used to break out of the loop or of the condition 'it was used with switch before' while continue is used to break out of the iteration itself.
    • Using continue is to stop preceding in the current running iteration and start the next iteration.

    using continue in a loop

  • An example on using Loops with the previously learnt concepts is shown in this example: isPalindrome

  • Checking the word is palindrome or not is a common programming problem you can search about it and check.

Finally, the forth chapter is FINISHED!!!


5.Methods

5. Methods

In this chapter you are about learning the first step towards the abstraction.

We already mentioned before on a successive chapter that methods are used to define reusable code.

Now we are going to delve deeper in Methods concepts in Java context, the following are the point's I'm going to discuss:

  1. Method definition.
  2. Method Invocation
  3. Pass by Value
  4. Modularizing Code
  5. Overloading
  6. Scope
  7. Method abstraction

1. Method Definition:

  • Before we discussed how does a method work, but we've mentioned that we were talking about the built-in methods.

  • Now, we are going to deal with custom-built methods, in other words you'll create it yourself.

  • This chapter will demonstrate the concept of How to Define a method.

  • A method consists of:

    • modifier
    • return type
    • method name
    • list of parameters
    • body
  • These components are shown in the following figure:

method definition

  • The modifier and its importance will be mentioned in a future chapter (OOP).

  • The return type, methods are of two types :

    1. value-returning methods
    2. void methods (return nothing).
  • Method Name and Parameter List which define the method signature -crucial for a future concept 'Overloading'-.

  • The parameter list is the variables, objects, arrays,etc. we pass to the method which are used in the processing performed be the method to finish a specific task, parameters actually are optional some methods have empty parameter list.

  • The method body like the other loop body the collection of statements which are grouped together to perform one task not for iteration like the loop body.

Abstraction:
  • Methods provide a form of abstraction, but what's meant by an abstraction by the way?
    • abstraction is the process of hiding the complex details and showing only the essential features of an object or a method. In programming, this allows developers to work with higher-level concepts without needing to understand the intricate implementation behind them.
  • May this figure clear your understanding about the abstraction idea and methods:

method abstraction

  • now let's try to refine an old example, for example let's refine the calculator example.
  • Let's encapsulate these lines of code in a small code statement consists only of a method call.
details of how we refined the example to make use of method concept

after the closing curly braces of the main method

  1. define a new method with a meaningful name calculate.
  2. add these two modifiers for now public and then static.
  3. define the return type, here it's void it can be double ,but we have to do some edits in switch statement.
  4. define the parameter list here (two integers the numbers which the operation will be performed on, and the operator code -also an integer-).
  5. move the long switch case in the method.
  6. invoke it in the main method. (remember anything you want to execute you must invoke it in the main method or in the main class which contains the main method).

details

invoking

  • final result after refining the calculator example:

calculator example refined

  • imagine the method is in another class the code will be like this:

calculator example refined

  • now everything looks really abstracted you don't know how this method process the operations ,but you know what it does!

  • another point worth mentioning is that the parameter list is called the formal parameters, when you actually passing values to the method then it's called actual parameters or arguments:

    • parameter list: formal paramters.
    • passing actual values: arguments.
  • some methods return values after finishing their task and others doesn't such as any print method it doesn't return value but prints some values on console.

  • methods returning values must contain the return keyword, the method terminates its execution when approach this keyword.

  • Some programming languages refer to methods as procedures and functions. A value-returning method is called a function and a void method is called a procedure.

little story:

The first programming language I was exposed to in the middle school was visual basic 'VB' maybe in 2016 or earlier, I don't even remember its version or how I coded using this language ,but it was the point where it all started 🤍, but I can remember well my teacher when she was differentiating between procedure and function, in general a value-returning method was called a function and a void method was called a procedure, I can remember this part because of the way we had been taught, I'm Egyptian and our native language is Arabic, so procedure translation in Arabic is masculine and function is a feminine word, so my teacher told us that mom comes back from the market with many groceries and surprises while dad he doesn't care and comes back with nothing, this was for joking she didn't mean anything mean but from the actual reality that your mom is more caring and loving than your dad by nature -again without any offense to either gender but if you live here in Egypt you'll enjoy and understand this kind of jokes 😄-.

2. Method Invocation:

  • To execute the method, you have to call or invoke it.

    • invoking or calling is simply by typing the method name and pass its parameters if they exist as follows:

    Invoking a method

    1. Number 1 and 3 are examples on parameter list and no parameter methods respectively.
    2. Number 2, have two parts the left part is the class name 'Math' as it's a static method ,and it should be invoked using the class name then the method name with an access operator . or the period.
Value-Returning Method vs Void Method:
  • Void method is used as a statement that can standalone while the value returning-value method is treated as a value as shown in the previous example, the value returned by the method calculateDistance can have two destinations:

    1. Is to be stored in a variable or array or whatever of the same type, or
    2. To be passed to the print statement 'if it can be printed directly' to be printed on the console.
  • A returning-value method must have a return keyword followed by the returned value to terminate the method execution, while the void method terminates when its execution approaches the closing curly braces.

  • Another note is that your method implementation must guarantee that the method will return the value at the end, look at the following example:

    • A return statement is required for a value-returning method. The method given in (a) is logically correct, but it has a compile error because the Java compiler thinks this method might not return a value.

    return example

    • To fix this problem, delete if (n < 0) in (a), so the compiler will see a return statement to be reached regardless of how the if statement is evaluated, as shown in (b).
Program Control, Call and Caller:
  • The program control refers to the order in which individual instructions, statements, and function calls are executed in a program

  • Program control is commonly known by the flow of execution, the normal flow is from top to bottom until a control structure or a method call is executed the program control moves to them until they end their tasks and terminates successfully.

  • When a method is called, the program's control is transferred to that method:

    • The current execution of code is paused.
    • Program control jumps to the method definition.
    • After the method finishes execution, control returns to the point where the method was called.
  • Understanding this point helps in other times when you want to debug your program, it aids you with the understanding of where you can stick a breaking point to check where a problem started.

  • The program that calls the function is called a caller.

  • The main method is just like any other method, except that it is invoked by the JVM to start the program.

    • The main method’s header is always the same. It includes the modifiers public and static, return value type void, method name main, and a parameter of the String[] type. String[] indicates the parameter is an array of String, a subject addressed in the next chapter.
    • The statements in main may invoke other methods that are defined in the class that contains the main method or in other classes.
Call Stack
  • What is the Call Stack?

    • The call stack is a special part of a computer’s memory where Java keeps track of all the methods that are currently being executed.
    • Think of it as a list or a stack of tasks that the program is handling. When a new task (method) is called, it gets added to the top of the list. Once the task is done, it is removed, and the program goes back to the previous task.
  • How Does the Call Stack Work?

    • Let’s break it down step by step:

      1. Program starts executing:

        • When you run a Java program, the main method is the first method that gets executed. The program control goes to the main method, and the first thing that happens is that a stack frame is created for main().
        • A stack frame or activation record is a structure that holds all the information needed for the method to run (local variables, parameters, and the point to return after the method is done).
      2. Calling another method:

        • When main() calls another method (let’s say printHello()), a new stack frame is created for printHello(), and it is added to the top of the call stack.
        • The method at the top of the call stack is always the one that is currently being executed. This means that main() is paused, and the program now focuses on executing printHello().
      3. Method completion:

        • Once printHello() finishes, its stack frame is removed (or popped off) from the call stack.
        • Program control then returns to main(), and main() continues executing from where it left off.
      4. Stack Frame Removal:

        • After the main() method finishes, its stack frame is removed, and the program ends.
    • Visualizing the Call Stack

      public class CallStackExample {
          public static void main(String[] args) {
              System.out.println("Start of main()");
              printHello();
              System.out.println("End of main()");
          }
      
          public static void printHello() {
              System.out.println("Hello from printHello()");
          }
      }

      When this program runs:

      1. main() starts and is placed on the call stack.
      2. Inside main(), the printHello() method is called, so a stack frame for printHello() is added.
      3. Once printHello() is done, its stack frame is removed, and the program returns to main().

      Call Stack Visualization:

      call stack

  • Why Does the Call Stack Matter?

    • Managing Method Calls: The call stack ensures that each method runs correctly and in the right order.
    • Memory Management: Each method’s local variables and parameters are stored in its stack frame, so Java knows how much memory to allocate and free when methods start and finish.
  • Important Points to Remember:

    • The call stack operates on a Last-In, First-Out (LIFO) basis, meaning the last method called is the first one to finish and be removed from the stack.
    • Each time a method is called, a new stack frame is created, and when the method completes, the stack frame is removed.
    • Stack Overflow: If too many methods are called without finishing, the call stack can become too large and overflow. This is called a stack overflow, which typically happens in cases of infinite recursion.
  • Real-world Analogy

    Imagine you are baking a cake (running a program). If you realize mid-way that you need eggs (another method), you pause the cake process and go get the eggs. Once you have the eggs (the method finishes), you return to baking (resume the original method). The call stack works similarly, ensuring that each task gets completed in the right order without losing track of the original task.

3. Pass By Value:

  • In Java, when you pass a variable to a method, you are passing only the value stored in that variable, not the actual memory address or reference of the variable itself. This is known as pass by value.

  • Every variable in Java has a memory address where its value is stored. When we manipulate a variable, we are working with the value at that memory location.

  • Passing by value means that when you call a method and pass a variable as an argument, Java copies the value stored in the variable and passes that copy to the method. Inside the method, any changes made to the copied value do not affect the original variable outside the method.

  • The original variable and the copy passed into the method are stored in different memory locations. Modifying the copy only affects the memory location of the copy, leaving the original untouched.

  • The original variable will only be modified if the method returns a new value, and you reassign the result to that variable.

  • This behavior applies to primitive data types (like int, float, boolean) and can be confusing with reference types (like objects and arrays), but the core principle remains the same: Java always passes values, not references.

  • let's demonstrate an example:

    ``` Java
    
        public class PassByValueExample {
    
          public static void main(String[] args) {
              int original = 10;
              System.out.println("Before calling method: " + original);
              
              modifyValue(original); // Passes the value, not the reference
              
              System.out.println("After calling method: " + original); // Original value remains unchanged
          }
    
        public static void modifyValue(int value) {
            value = 20; // Only modifies the copy of the value
            System.out.println("Inside method: " + value);
        }
      }
    
    ```
    
  • The value of original remains the same after calling the method.

  • Other languages which deal with pointers like C/Cpp can pass references to methods causing direct change to the original variables.

  • Another important point is that you must pass arguments in the same order as s their respective parameters in the method signature. This is known as parameter order association.

  • The arguments must match the parameters in order, number, and compatible type, as defined in the method signature. Compatible type means you can pass an argument to a parameter without explicit casting, such as passing an int value argument to a double value parameter.

  • observe this example:

    ``` Java 
        public class Main{
            public static void main(String[] args) {
                // passing incompatible types to a method: 
                double base = 10;
                double height = 5;
              double area = calculateTriangleArea(base, height);
              System.out.println("The triangle area is: "+ area);
            }
          
            public static double calculateTriangleArea(float base, int height){
                
                return 0.5 * base * height;
            }
          }
    ```  
    

    incompatible types

4. Modularizing Code:

  • In Java, methods are one of the key ways to modularize code, which means breaking down a large, complex problem into smaller, more manageable pieces. This not only makes the code easier to understand but also allows for better organization and re-usability. Here's how it works:
  1. Encapsulation:
    Encapsulation is the concept of bundling data and methods that operate on that data into a single unit (like a class). By using methods, we hide the internal implementation details and expose only the functionality that other parts of the program need to know about. This allows for independent methods to be developed and modified without affecting the rest of the code.

  2. Abstraction:
    Abstraction involves hiding the complex details of what a method does and focusing on what the method achieves. By defining clear method names and purposes, the inner workings of a method are abstracted away, letting you focus on the bigger picture. This makes code easier to understand and maintain.

  3. Isolation and Independence:
    Since each method is self-contained, it can be tested and debugged in isolation, ensuring that changes to one method do not affect other parts of the code. This allows developers to modify or improve individual methods without worrying about unintended consequences elsewhere in the program.

  4. Re-usability:
    One of the greatest benefits of methods is their re-usability. Once a method is written, you can call it multiple times from different parts of the program, reducing redundancy and keeping the code clean and efficient.

  • The demonstrated the calculator example here clarifies the point:

    ``` Java
            public class Calculator {
    
            public static void main(String[] args) {
                int num1 = 10;
                int num2 = 5;
    
                // Modularity in action: calling independent methods for each operation
                System.out.println("Addition: " + add(num1, num2));
                System.out.println("Subtraction: " + subtract(num1, num2));
                System.out.println("Multiplication: " + multiply(num1, num2));
                System.out.println("Division: " + divide(num1, num2));
            }
    
            // Each method encapsulates its own logic, providing abstraction
            public static int add(int a, int b) {
                return a + b;
            }
    
            public static int subtract(int a, int b) {
                return a - b;
            }
    
            public static int multiply(int a, int b) {
                return a * b;
            }
    
            public static int divide(int a, int b) {
                if (b != 0) {
                    return a / b;
                } else {
                    System.out.println("Error: Division by zero.");
                    return 0;
                }
            }
        }
    
    ```
    
  • Until this point, abstraction concept is being introduced to you bit by bit until we approach the OOP chapters; then the concept will be clear.

5. Overloading:

  • In Java, method overloading is a technique that allow methods to behave differently based on various factors.

  • Method Overloading or Compile-time Polymorphism - will be discussed later in OOP chapters - Method overloading occurs when multiple methods in the same class have the same name but different parameter lists (different number of parameters, types, or both). This is resolved at compile time, meaning the Java compiler decides which method to call based on the method signature.

    • Purpose: To create several versions of the same method that can handle different input data.
    • When Used: When you need the same method name to perform different operations based on the input parameters.
  • observe the following example:

    ``` Java
    
        public class Calculator {
    
        // Overloaded methods: same name but different parameters
          public int add(int a, int b) {
            return a + b;
          }
    
          public double add(double a, double b) {
              return a + b;
          }
    
          public int add(int a, int b, int c) {
              return a + b + c;
          }
    
          public static void main(String[] args) {
              Calculator calc = new Calculator();
    
              // Calls different add methods based on the arguments
              System.out.println(calc.add(5, 10));       // Calls the first add method (int, int)
              System.out.println(calc.add(5.5, 10.2));   // Calls the second add method (double, double)
              System.out.println(calc.add(5, 10, 15));   // Calls the third add method (int, int, int)
          }
        }
    
    ```
    
  • Sometimes there are two or more possible matches for the invocation of a method, but the compiler cannot determine the most specific match. This is referred to as ambiguous invocation. Ambiguous invocation causes a compile error. Consider the following:

    ```Java
       public class AmbiguousOverloading {
          public static void main(String[] args) {
          System.out.println(max(1, 2));
          }
          public static double max(int num1, double num2) {
            if (num1 > num2)
              return num1;
            else
              return num2;
          }
          public static double max(double num1, int num2) {
            if (num1 > num2)
              return num1;
            else
              return num2;
          }
        }
    ```  
    
  • Both max(int, double) and max(double, int) are possible candidates to match max(1, 2). Because neither is more specific than the other, the invocation is ambiguous, resulting in a compile error.

6. Scope:

  • The scope of a variable is the part of the program where the variable can be referenced.

  • scope levels:

    scope level

  • A variable defined inside a method is referred to as a local variable.

  • The scope of a local variable starts from its declaration and continues to the end of the block that contains the variable.

  • A local variable must be declared and assigned a value before it can be used.

  • A parameter is actually a local variable. The scope of a method parameter covers the entire method.

    example on scope level

  • A variable declared in the initial-action part of a for-loop header has its scope in the entire loop. However, a variable declared inside a for-loop body has its scope limited in the loop body from its declaration

  • You can declare local variables with the same name in different blocks within a method. However, you cannot declare the same local variable twice in the same block or in nested blocks. In nested blocks, this is not allowed because the variable declared in the outer block remains accessible within the inner block. On the other hand, if the variable is declared in the inner block, it will not be accessible outside that block.

    nested blocks scope

7. Stepwise Refinement:

  • The first step in developing a software is to understand the abstraction concept.

  • The concept is already introduced earlier in the chapter abstraction.

  • Abstraction concept is achieved using encapsulation, encapsulation is hiding the implementation details from the user, thus if the implementation changed for any reason, the user is not aware he/she just keep using the method or the class.

  • A key point in designing large programs is to divide the program into sub-programs using divide-and-conquer approach or commonly known by stepwise refinement.

  • You can follow top-down design strategy to develop the program.

  • To make problem-solving flow as smoothly as possible, you'll apply method abstraction concept to isolate details from design and only later implements the details either using bottom-up/top-up approaches.

Student portal example:
  • Imagine we have a student's portal for faculty-x, we won't discuss all the features of the portal, but we are going to focus on just one feature: print the current student's transcript.

  • The top-down approach will be applied to break up the big problem into smaller manageable problems:

  • print student's transcript should print the following:

    • student's name
    • ID
    • Academic year (based on a key on the ID)
    • completed hours
    • CGPA
    • Current semester GPA
  • The Big idea is to print the student's transcript:

    student's transcript

  • Divide the big idea into two sub-problems:

    • Input Data
    • Print transcript

    two sub-problems

  • Now focus on each sub-problem independently; for example, in Print Transcript we have to divide it into another two sub-problems (the number of sub-problems is not mandatory to be two but this what our case requires at the current time):

    • GPA calculator
    • Print Transcript

    GPA calculator and Print transcript

  • Translate your focus on GPA Calculator, to calculate the GPA you have to sum all the weights of the registered courses multiplied by their credit hours and divided over th total registered hours this semester, so calculate weights method will be added:

    calculate weights

  • let's sum up the whole problem: (I've added some additional blocks)

    the big picture

  • Now after applying the Top-Down approach in designing the feature, let's move to implementation:

    • I used the bottom-up approach to implement the methods:

      1. I started by the getLiteralGrades methods, as I wanted to print the literal grade of each course, then pass these literals to another method calculates the weight of each course solely.
      public static String[] getLetterGrade(int[] grades){
        String[] letterGrades = new String[grades.length];
        int index = 0;
        for(int grade : grades){
            if(grade < 50)
                letterGrades[index++] = "F";
            else if(grade < 60)
                letterGrades[index++] = "D";
            else if (grade < 65)
                letterGrades[index++] = "D+";
            else if (grade < 70)
                letterGrades[index++] = "C";
            else if (grade < 75)
                letterGrades[index++] = "C+";
            else if (grade < 80)
                letterGrades[index++] = "B";
            else if (grade < 85)
                letterGrades[index++] = "B+";
            else if (grade < 90)
                letterGrades[index++] = "A";
            else
                letterGrades[index++] = "A+";
        }
      
        return letterGrades;
      }
      1. Then calculateWeights method came into the play:
          public static float[] calculateWeights(String[] letterGrade){
          float[] weights = new float[letterGrade.length];
          int index = 0;
          for(String grade : letterGrade){
              switch(grade){
                  case "A+" -> weights[index++] = 4.0f;
                  case "A" -> weights[index++] = 3.75f;
                  case "B+" -> weights[index++] = 3.4f;
                  case "B" -> weights[index++] = 3.1f;
                  case "C+" -> weights[index++] = 2.8f;
                  case "C" -> weights[index++] = 2.5f;
                  case "D+" -> weights[index++] = 2.25f;
                  case "D" -> weights[index++] = 2;
                  default -> weights[index++] = 1;
              }
          }
          return weights;
        }

      notice the enhanced switch statement, read about how to use it here.

      1. the implementation of gpaCalculator method based on the mathematical calculation of the GPA:

      $$\frac{\sum_{x = 1}^{n} (course : weight × credit : hours)}{total:completed:hours}$$

      where n is the number of registered courses, each course literal grade has a weight, the sume of the weights multiplied by credit hours over the total completed hours is the GPA.

          public static float gpaCalculator(float[] weights,
                                          int totalHours,
                                          int numberOfRegisteredCourses,
                                          int[] creditHours){
            float sum = 0.0f;
            for(int i = 0; i < numberOfRegisteredCourses; i++){
                sum += weights[i] * creditHours[i];
            }
            return sum / totalHours;
        }
      1. The implementation of cumulativeGPA method comes as following:

        based on the mathematical calculation:

      $$ CGPA = \frac {(Old GPA × Old Credit Hours)+(New GPA × New Credit Hours)​}{Old Credit Hours + New Credit Hours }$$

          public static float cumulativeGPA(float gpa, float oldGPA, 
                                          int totalCompletedHours,
                                          int oldCompletedHours){
            
            return (gpa * totalCompletedHours + oldGPA * oldCompletedHours)/(oldCompletedHours+totalCompletedHours);
        }
      1. Then the studentPrintTranscript method:
          public static void printTranscript(float cgpa , float gpa, 
                                          String name, String id, 
                                          String[] letterGrades, String[] coursesNames,
                                          int oldCompletedHours, int currentSemesterCompletedHours){
            String classification = id.substring(7);
            int index = 0;
      
            // identify the academic year of the student from his/her id
            int academicYear = switch(classification){
                case "Fr" -> 1;
                case "So" -> 2;
                case "Jr" -> 3;
                case "Sr" -> 4;
                default -> 0;
            };
      
            //print transcript
            System.out.printf(
                    "Student's Name: %s \n" +
                    "ID: %s \n" +
                    "Academic Year: %d \n" +
                    "Current Semester GPA: %.2f \n" +
                    "CGPA: %.2f \n " +
                    "Total Completed Hours: %d \n",
                    name,
                    id,
                    academicYear,
                    gpa,
                    cgpa,
                    (oldCompletedHours+currentSemesterCompletedHours));
      
            // print student's literal grades of each course:
            for(String grade : letterGrades){
                System.out.println("Course Name: " + coursesNames[index++] + " grade: "+ grade);
            }
        }
      1. actually I decided to refactor the printTranscript method into print and estimateTheAcaddemicYear methods, to make each method atomic (do one mission).
          public static void printTranscript(float cgpa , float gpa,
                                       String name, String id,
                                       String[] letterGrades, String[] coursesNames,
                                       int oldCompletedHours, int currentSemesterCompletedHours, int academicYear){
      
            int index = 0;
            //print transcript
            System.out.printf(
                    "Student's Name: %s \n" +
                    "ID: %s \n" +
                    "Academic Year: %d \n" +
                    "Current Semester GPA: %.2f \n" +
                    "CGPA: %.2f \n " +
                    "Total Completed Hours: %d \n",
                    name,
                    id,
                    academicYear,
                    gpa,
                    cgpa,
                    (oldCompletedHours+currentSemesterCompletedHours));
      
            // print student's literal grades of each course:
            for(String grade : letterGrades){
                System.out.println("Course Name: " + coursesNames[index++] + " grade: "+ grade);
            }
        }
      
      public static int estimateTheAcademicYear(String id){
      
          String classification = id.substring(7);
          switch(classification){
              case "Fr" -> { return 1; }
              case "So" -> { return 2; }
              case "Jr" -> { return 3; }
              case "Sr" -> { return 4; }
              default -> { return 0; }
          }
      
        }
      1. It's better to enter the related data as collections, we are going to learn about the first and the basic collection Arrays in the next chapter, I used Arrays reference type to facilitate the passing and receiving of data.
      2. I constructed methods to process the input collections:
          public static int[] processingNumericalInput(){
            int[] arr = new int[SIZE];
            Scanner input = new Scanner(System.in);
            for(int i = 0; i<SIZE;i++){
                arr[i] = input.nextInt();
            }
            return arr;
          }
      
        public static String[] processingLiteralInput(){
            String[] arr = new String[SIZE];
            Scanner input = new Scanner(System.in);
            for(int i = 0; i<SIZE;i++){
                arr[i] = input.nextLine();
            }
            return arr;
          }
      1. Then let's sum all up in the main method where we pass everything:
          public static void main(String[] args) {
      
          // Scanner object
          Scanner input = new Scanner(System.in);
      
          // declare identifiers
          String name, id;
          String[] coursesNames , letterGrades ;
          int[] creditHours, grades ;
          int totalHours, oldCompletedHours, academicYear;
          float[] weights;
          float cgpa,gpa,oldGPA;
      
          // processing inputs:
          System.out.println("enter your name: ");
          name  = input.nextLine();
          System.out.println("Welcome! "+name);
          System.out.println("enter your id: ");
          id  = input.nextLine();
          System.out.println("enter current semester courses names: ");
          coursesNames = processingLiteralInput();
          System.out.println("enter credit hours for each course in the same order as courses names: ");
          creditHours = processingNumericalInput();
          System.out.println("enter grades in the same order as courses names: ");
          grades = processingNumericalInput();
          System.out.println("enter total completed hours this semester: ");
          totalHours = input.nextInt();
          System.out.println("enter your old gpa: ");
          oldGPA = input.nextFloat();
          System.out.println("enter the previous completed hours before this semester: ");
          oldCompletedHours = input.nextInt();
      
          // print the transcript:
          letterGrades = getLiteralGrades(grades);
          weights = calculateWeights(letterGrades);
          gpa = gpaCalculator(weights,totalHours,SIZE,creditHours);
          cgpa = cumulativeGPA(gpa,oldGPA,totalHours,oldCompletedHours);
          academicYear = estimateTheAcademicYear(id);
          printTranscript(cgpa,gpa,name,id,letterGrades,coursesNames,oldCompletedHours,totalHours,academicYear);
      
      
      }
    • You can find the full project here: Student's Portal

Fifth Chapter is Finished!!! ​


6. Arrays

6. Arrays

In this chapter, you are about to learn the first step towards the data structures world and a crucial reference type used in almost everything. We already mentioned before, in a successive chapter, that methods are used to define reusable code to not repeat the code lines applying the DRY heuristic. In the same context, we're going to be familiar with the very basic data structure : array.

Array stores a fixed-size sequential collection of elements of the same type which allow to not repeat the declaration for large number of variables of the same type.

Concepts: 1. Array Basics (Declaration, size, accessing elements.etc.) 2. Copying Arrays. 3. Passing Arrays to Methods. 4. Returning an Array from a Method. 5. Variable-Length Argument Lists. 6. Arrays Class. 7. Command Line Arguments. 8. Two Dimensional Arrays.

Array Basics:

  • If you are familiar with C/C++ the array declaration would be like that:

        int array[] = {1,2,3,4};
  • This is allowed in Java but not used, the declaration convention of arrays in Java is quiet similar:

         int[] array = {1,2,3,4};
  • yes, just the place of the brackets is different.

  • Array is a reference type, which means the array reference variables contains the address of the array not the array content itself. try the following:

    ```Java
        int[] elements = {0,3,5,2,51,54,78,9,3}; 
        System.out.println(elements); //returns an ambiguos string -- memory location
    ```
    
  • It's important at this point to differentiate between declaring an array and allocating memory for it, declaring an array is just declaring an array reference variable to point to the array elements later, allocating space actually happens when you add elements to the array.

  • Unlike the primitive datatypes; declaring an array doesn't allocate space in memory, only when you start to assign values to array elements.

    ```Java
        int[] elements; // this is declaration without initialization
        //System.out.println(elements); //error          
    ``` 
    
  • now you have to be familiar with null which is the value of the reference variable if the reference variable doesn't contain a reference to an array.

  • you get NullPointerException when you try to access elements[0] if you didn't initialize elements array already, here the reference variable points to null.

  • Using null keyword with reference variables as a 'value':

    ```Java
        int[] elements = null;
        System.out.println(elements); //null
    ```
    
  • How empty array could be:

    ``` Java
        int[] integers = {};
        System.out.println(integers.length); // 0  
    ```    
    
  • So in general,the syntax of declaring array as follows:

    elementType arrayRefVar[index] = new elementType[arraySize];

  • This statement does two things: (1) it creates an array using new elementType[arraySize] and (2) it assigns the reference of the newly created array to the variable arrayRefVar.

  • you can declare an array, specify its size then assign elements to it:

    ``` Java
          int[] myList = new int[10]; 
          myList[7] = 5;
    ```
    
  • for example this is one way to how you can initialize an array:

       myList2[0] = 5.6;
       myList2[1] = 4.5;
       myList2[2] = 3.3;
       myList2[3] = 13.2;
       myList2[4] = 4.0;
       myList2[5] = 34.33;
       myList2[6] = 34.0;
       myList2[7] = 45.45;
       myList2[8] = 99.993;
       myList2[9] = 11123;
  • After you specify the size of an array, you cannot change it later, you can obtain the size of an array using arrRefVar.length:

    ``` Java
          System.out.println(mylist.length); // 10
    ```
    
  • reminding you again that arrays in Java are 0-based index (means you start counting and accessing array elements from 0 index)

  • in mylist example if you tried to access mylist[10] will result in ArrayOurOfBounds runtime error.

  • Java has a shorthand notation, known as the array initializer, which combines the declaration, creation,and initialization of an array in one statement:

    ``` Java
          double[] myList = {1.9, 2.9, 3.4, 3.5};
    ```
    
  • important note to mention :you cannot separate declaration from initialization in arrays, you can use the array initializer only while you declaring the array.

    ``` Java
          float[] arv;
          //arv = {1.1,3.4,5.6}; //error 
    ```
    
  • It was mentioned that the array size cannot be altered after declaration, once the space is allocated the elements of the array take default values before you assign values to them:

    Data Type Default Value
    Numeric 0
    Char \u0000
    Boolean false
  • Accessing array elements is done using its index, remember again arrays are zero indexed means if you want to access the second array element it will be like this : elements[1];

  • The index values range from 0 to arrayReferenceVariable.length -1

  • An indexed variable can be used in the same way as a regular variable. For example, the following code adds the values in myList[0] and myList[1] to myList[2]:

    ```Java  
          myList[2] = myList[0] + myList[1];
    ```        
    
  • You can insert, print or traverse array using loops:

    • for loops

    • for each

      • for each loops are enhanced loops can be used to forward traverse arrays or collections in single steps.
    • using ordinary loop:

        for (int i = 0; i < myList.length; i++) {
            System.out.print(myList[i] + " ");
        }
    • using for each:

          for (int num : mylist) {
            System.out.println(num);}
  • Using for loops are more common than for each

    • Let's initialize arrays using the same logic of iteration:
      • prompt user to enter array values:

            Scanner input  = new Java.util.Scanner(System.in);
            for (int i = 0; i< mylist.length; i++){
            myList[i] = input.nextDouble();
            }
        ```
      • initialize the array using random function from Math class:

            for (int i = 0; i < mylist.length; i++){
              mylist[i] = (Math.random() * 100); // inserting numbers ranges from 0.0 to 100.0 exclusive
            }
  • A question might come up here: why don’t we use the enhanced for-each loop? It’s a good question. for-each loops are excellent for traversing (reading) elements in a step-by-step manner, but they’re not suitable for modifying elements. This is because the loop works with a copy of each element, rather than directly accessing them through their indices.

  • another note is that if you tried to print elements of char[] type array, you can use System.out.print directly unlike the other array types which require iteration structure to handle the traverse.

        char[] name = {'A','l','i'};
        System.out.println(name); // Ali 
  • This behavior can be observed with printf, as it doesn’t automatically treat char[] as a string due to internal differences in how printf handles format specifiers compared to print and println. Unlike print and println, printf relies on format specifiers that don’t automatically interpret char[] as a sequence of characters. Therefore, converting char[] to a String is necessary with printf to avoid printing the reference address.

Copying Arrays:

  • Once you entered the world of reference types, everything changes.
    • Even copying, you cannot copy contents an array into another array just like the following:

         int[] copyArray = mylist; // that's wrong

      copying array using assignment operator

    • you've just copied the reference of mylist array into copyArray, so copyArray have the same reference, not the content.

    • There are three ways to copy arrays:

      1. Use a loop to copy individual elements one by one.

      2. Use the static arraycopy method in the System class.

      3. Use the clone method to copy arrays; this would be more clear as we approach deeper in OOP.

        1. Using loop
        2. Using arraycopy
          public class Main
          {
            public static void main(String[] args) {
              int[] array = new int[6];
              for(int i=0; i<array.length; i++){
                  array[i] = (int)(Math.random() * 6);
                  System.out.println(array[i]);
              }
            
              // to copy this array into another array using loops:
              System.out.println("-----------------");
            
              int[] copiedArray = new int[6];
            
              for(int j=0; j<copiedArray.length; j++){
                  copiedArray[j] = array[j];
                  System.out.println(copiedArray[j]);
              }
            
            }
          }

      note that arraycopy method doesn't follow the naming convention in Java.

          //formal declaration of the method:
          public static void arraycopy(Object src, int srcPos, Object dest, int destPos, int length);
          // where src is the source you would copy from, srcPos --> where you want to start copying, the same for the next two parameters, the last parameter is the number of elements would you copy?
          
              import Java.util.Arrays;
              public class Main
              {
                public static void main(String[] args) {
                  int[] array = new int[6];
                  for(int i=0; i<array.length; i++){
                      array[i] = (int)(Math.random() * 6);
                      System.out.println(array[i]);
                  }
                  
                  // to copy this array into another array using copyarray:
                  System.out.println("-----------------");
                  
                  int[] copiedArray = new int[6];
                  System.arraycopy(array,0,copiedArray,0,array.length);
                  for(int num: copiedArray) System.out.println(num);
                }
              }

Passing Arrays to Methods:

  • We've already learnt that Java is 'passing by value', that happens when you pass primitive data types in addition to the reference data types as arrays.
  • It's a common pitfall to think Java passes array or objects by reference because you can actually make change after passing them to a method, but it's not, it still passes them by value.
  • The confusion between pass by value and pass by reference in Java often arises due to how reference types like objects and arrays behave when passed to methods.
  • This can be clarified using examples:
    • In Java, everything is passed by value. This means when a variable is passed to a method, a copy of its value is made and used in the method.

      • For Primitive Data Types When a primitive value (like int, float, boolean) is passed, the method gets a copy of the value. Any changes made to the parameter inside the method do not affect the original value.

            public class Demo {
                public static void modifyPrimitive(int number) {
                    number = 10; // Change the local copy
                }
        
                public static void main(String[] args) {
                    int original = 5;
                    modifyPrimitive(original);
                    System.out.println(original); // Outputs: 5
                }
            }
      • Explanation:

        • The number parameter in modifyPrimitive gets a copy of original's value.
        • Changing number does not affect the original variable in the main method.
      • For Reference Types When an object or array is passed, the method receives a copy of the reference (the memory address). The reference still points to the same object in the heap. Modifying the object through this reference affects the original object.

        public class Demo {
            public static void modifyArray(int[] numbers) {
                numbers[0] = 10; // Modify the object
            }
        
            public static void main(String[] args) {
                int[] array = {1, 2, 3};
                modifyArray(array);
                System.out.println(array[0]); // Outputs: 10
            }
        }
      • Explanation:

        • The numbers parameter in modifyArray gets a copy of the reference to array.
        • Both numbers and array point to the same array in the heap.
        • Modifying numbers[0] changes the shared array, so the original array in main is affected.
    • Why It’s Not Pass by Reference

      • In pass by reference, the method would receive a direct reference to the variable itself, allowing it to reassign the original reference entirely.
    • However, in Java:

      • The reference is copied and passed to the method.

      • If you reassign the reference inside the method, it does not affect the original reference.

        Example:

        public class Demo {
            public static void reassignReference(int[] numbers) {
                numbers = new int[]{10, 20, 30}; // Reassign the reference
            }
        
            public static void main(String[] args) {
                int[] array = {1, 2, 3};
                reassignReference(array);
                System.out.println(array[0]); // Outputs: 1
            }
        }

      Explanation:

      • Inside reassignReference, the numbers parameter is assigned a new array.
      • This does not affect the array in main, as the original reference was not changed.

4. Returning Arrays From Methods:

  • You can pass arrays when invoking a method. A method may also return an array.

      public class ArrayReturnExample {
      // Method to return an array of even numbers
      public static int[] generateArray(int size) {
          int[] array = new int[size]; 
          for (int i = 0; i < size; i++) {
              array[i] = i * 2; 
          }
          return array; // Return the array
      }
      public static void main(String[] args) {
          int[] result = generateArray(5); // Call the method and get the array
          for (int num : result) {        // Print the elements of the array
              System.out.print(num + " ");
          }
      }
      }

5. Variable Length Parameter Lists:

  • we can pass variable number of parameter list, means that you can pass one or two or more parameters but should be of the same type, here java treats the var param list as Array.

  • In the method declaration, you specify the type followed by an ellipsis (...).

  • Only one variable-length parameter may be specified in a method.

  • This parameter must be the last parameter. Any regular parameters must precede it.

  • Look at the following example:

     ```Java 
        public class VarargsExample {
        // Method with variable-length parameter list
        public static int sum(int... numbers) {
            int total = 0;
              for (int num : numbers) {
                  total += num; // Add each number to total}
              return total;}
        public static void main(String[] args) {
            System.out.println(sum(1, 2, 3));       // Output: 6
            System.out.println(sum(10, 20, 30, 40)); // Output: 100
            System.out.println(sum());             // Output: 0 (no arguments passed)
          }
        }
     ```
    

6. Arrays Class in Java:

  • The java.util.Arrays class contains various static methods for sorting and searching arrays, comparing arrays, filling array elements, and returning a string representation of the array. These methods are overloaded for all primitive types.
  1. Sorting Arrays Using Arrays.sort() The sort() method sorts the elements of an array in ascending order.

    import java.util.Arrays;
    
    public class SortExample {
        public static void main(String[] args) {
            int[] numbers = {5, 1, 12, -5, 16};
            Arrays.sort(numbers); // Sort in ascending order
            System.out.println(Arrays.toString(numbers)); // Output: [-5, 1, 5, 12, 16]
        }
    }
  2. Parallel Sorting Using Arrays.parallelSort() parallelSort() divides the array into sub-arrays and sorts them in parallel, combining the results at the end.

    import java.util.Arrays;
    
    public class ParallelSortExample {
        public static void main(String[] args) {
            int[] numbers = {9, 3, 7, 1, 6};
            Arrays.parallelSort(numbers); // Parallel sorting
            System.out.println(Arrays.toString(numbers)); // Output: [1, 3, 6, 7, 9]
        }
    }
  3. Searching for an Element Using Arrays.binarySearch() binarySearch() searches for a specified value in a sorted array. It returns the index of the element if found, or a negative value if not.

    import java.util.Arrays;
    
    public class BinarySearchExample {
        public static void main(String[] args) {
            int[] numbers = {1, 3, 5, 7, 9};
            int index = Arrays.binarySearch(numbers, 5); // Search for 5
            System.out.println("Index of 5: " + index); // Output: Index of 5: 2
            
            int notFoundIndex = Arrays.binarySearch(numbers, 4); // Search for 4
            System.out.println("Index of 4: " + notFoundIndex); // Output: Index of 4: -3
        }
    }
  4. Converting an Array to a String Using Arrays.toString() toString() converts the contents of an array into a readable string representation.

    import java.util.Arrays;
    
    public class ToStringExample {
        public static void main(String[] args) {
            String[] fruits = {"Apple", "Banana", "Cherry"};
            System.out.println(Arrays.toString(fruits)); // Output: [Apple, Banana, Cherry]
        }
    }
  5. Filling an Array Using Arrays.fill() fill() assigns a specified value to every element in the array.

    import java.util.Arrays;
    
    public class FillExample {
        public static void main(String[] args) {
            int[] numbers = new int[5];
            Arrays.fill(numbers, 42); // Fill all elements with 42
            System.out.println(Arrays.toString(numbers)); // Output: [42, 42, 42, 42, 42]
        }
    }
  6. Comparing Arrays Using Arrays.equals() equals() checks if two arrays are equal, element by element.

    import java.util.Arrays;
    
    public class EqualsExample {
        public static void main(String[] args) {
            int[] arr1 = {1, 2, 3};
            int[] arr2 = {1, 2, 3};
            int[] arr3 = {3, 2, 1};
            
            System.out.println(Arrays.equals(arr1, arr2)); // Output: true
            System.out.println(Arrays.equals(arr1, arr3)); // Output: false
        }
    }
  7. Copying Arrays Using Arrays.copyOf() copyOf() creates a copy of the array with the specified length.

    import java.util.Arrays;
    
    public class CopyOfExample {
        public static void main(String[] args) {
            int[] numbers = {10, 20, 30};
            int[] copy = Arrays.copyOf(numbers, 5); // Copy with size 5
            System.out.println(Arrays.toString(copy)); // Output: [10, 20, 30, 0, 0]
        }
    }
  8. Splitting Arrays Using Arrays.copyOfRange() copyOfRange() creates a copy of a specified range from the original array.

    import java.util.Arrays;
    
    public class CopyOfRangeExample {
        public static void main(String[] args) {
            int[] numbers = {1, 2, 3, 4, 5};
            int[] subArray = Arrays.copyOfRange(numbers, 1, 4); // Copy index 1 to 3 (4 exclusive)
            System.out.println(Arrays.toString(subArray)); // Output: [2, 3, 4]
        }
    }
  9. Comparing Lexicographically Using Arrays.compare() compare() compares two arrays lexicographically.

    import java.util.Arrays;
    
    public class CompareExample {
        public static void main(String[] args) {
            int[] arr1 = {1, 2, 3};
            int[] arr2 = {1, 2, 4};
            int result = Arrays.compare(arr1, arr2);
            System.out.println(result); // Output: -1 (arr1 < arr2)
        }
    }
  10. Checking Array Mismatches Using Arrays.mismatch() mismatch() returns the index of the first mismatch between two arrays, or -1 if they're identical.

    import java.util.Arrays;
    
    public class MismatchExample {
        public static void main(String[] args) {
            int[] arr1 = {1, 2, 3};
            int[] arr2 = {1, 2, 4};
            int mismatchIndex = Arrays.mismatch(arr1, arr2);
            System.out.println("First mismatch at index: " + mismatchIndex); // Output: First mismatch at index: 2
        }
    }

Arrays Methods Table:

Method Purpose
sort() Sorts array elements in ascending order.
parallelSort() Sorts array elements using parallel computation
binarySearch() Finds an element in a sorted array.
toString() Converts array to a string representation.
fill() Fills array elements with a specified value.
equals() Compares two arrays for equality.
copyOf() Copies array with specified length.
copyOfRange() Copies a range of array elements.
compare() Compares arrays lexicographically.
mismatch() Finds the first index of mismatch.

7. Command Line Arguments:

  • The main method can receive string arguments from the command line.
  • You might have noticed the unique header of the main method, which includes the parameter args of type String[].
  • Essentially, args is an array of strings.
  • Similar to any other method with parameters, the main method can accept arguments, just as you would pass actual parameters when calling a regular method.
public class Main
{
  public static void main(String[] args) {
    System.out.println("Hello, "+ args[0]);
  }
}
  • before running the code, in the command line pass the arguments of main method as follows:
    javac Main.java
    java Main yourname

8. Two Dimensional Arrays:

  • A two-dimensional array in Java is essentially an array of arrays. It can be visualized as a table, where each element is accessed using two indices: one for the row and one for the column.
Declaration and Initialization:
  1. Declaration:

    • A two-dimensional array can be declared as follows:
      int[][] matrix;
  2. Initialization:

    • You can initialize the array at the time of declaration or later.
      • Method 1: During declaration
        int[][] matrix = {
            {1, 2, 3},
            {4, 5, 6},
            {7, 8, 9}
        };
      • Method 2: After declaration
        int[][] matrix = new int[3][3]; // creates a 3x3 matrix
        matrix[0][0] = 1;
        matrix[0][1] = 2;
        matrix[0][2] = 3;
        matrix[1][0] = 4;
        matrix[1][1] = 5;
        matrix[1][2] = 6;
        matrix[2][0] = 7;
        matrix[2][1] = 8;
        matrix[2][2] = 9;
Accessing Elements in a Two-Dimensional Array:
  • You access an element by specifying two indices: one for the row and one for the column.
int value = matrix[1][2]; // Accesses the element in the 2nd row and 3rd column (value is 6)
Iterating Through Two-Dimensional Arrays:
  • You can iterate over a two-dimensional array using nested loops:
for (int i = 0; i < matrix.length; i++) { // Loop through rows
    for (int j = 0; j < matrix[i].length; j++) { // Loop through columns
        System.out.print(matrix[i][j] + " ");
    }
    System.out.println();
}
Example of a Calculator Using a Two-Dimensional Array:
public class Calculator {
    public static void main(String[] args) {
        int[][] operations = new int[3][3]; // A 3x3 matrix for storing results of operations

        // Filling the array with simple operations
        operations[0][0] = 10 + 5;   // addition
        operations[0][1] = 10 - 5;   // subtraction
        operations[0][2] = 10 * 5;   // multiplication
        operations[1][0] = 10 / 5;   // division
        operations[1][1] = 10 % 5;   // modulo
        operations[1][2] = (int) Math.pow(10, 2); // exponentiation
        operations[2][0] = 5 + 10;   // reverse addition
        operations[2][1] = 5 - 10;   // reverse subtraction
        operations[2][2] = 5 * 10;   // reverse multiplication

        // Printing the result of operations
        for (int i = 0; i < operations.length; i++) {
            for (int j = 0; j < operations[i].length; j++) {
                System.out.print(operations[i][j] + " ");
            }
            System.out.println();
        }
    }
}
**Output:**
```
15 5 50 
2 0 100 
15 -5 50 
```

notes:

Last Chapter in Java Basics is Finished Finally 🥳!!!


2- Object Oriented Programming

4 Main concepts of OOP:

  1. Abstraction
  2. Encapsulation
  3. Inheritance
  4. Polymorphism

Object Oriented Thinking:

  • A class is a blueprint , template or a contract which defines properties and behaviours of objects.

  • An object is an instance of a class.

    • an object has a unique :
      • state --> datafield values.
      • behaviour --> methods to perform some operations on the object.
  • The process of making instance of a class is called instantiation.

  • The instantiation is done using a special method is called the constructor.

         ClassName(parameters){
          //lines of code to initialize the class
         }
  • Constructors can:

    • be overloaded
    • be invoked
    • have an access modifier
  • note when you have two classes in one file, one of them only is the public class and the other is not, so the file name must be the same as the public class name.

  • if you didn't specify any constructors, a no-arg constructor is made for your class automatically by the Java compiler.

About

This repo is a full guide to learn java language in addition to software engineering principles, also it contains mini-applications on java from scratch -basic concepts- to small | mid-sized java projects

Topics

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages