 
- Scala - Home
- Scala - Overview
- Scala - Features
- Scala - Environment Setup
- Scala - Build Tool (SBT)
- Scala - REPL
- Scala - Dot & Dotty
- Scala - Basic Syntax
- Scala - Hello World Program
- Scala - Identifiers
- Scala - Keywords
- Scala - Comments
- Scala - Code Blocks
- Scala - Semicolon
- Scala - Constructs
- Scala - Expressions
- Scala - Input and Output
- Scala - Optional Braces
- Scala - Underscore (_)
- Data Types and Variables
- Scala - Data Types
- Scala - Type Bounds
- Scala - Context Bound
- Scala - Variances
- Scala - Type Hierarchy
- Scala - Variables
- Scala - Variable Scopes
- Scala - Literals
- Scala - Numeric Types
- Scala - Boolean Types
- Scala - Char Type
- Scala - Unit Types
- Scala - Strings
- Scala - Arrays
- Scala - Null Type
- Scala - Nothing
- Scala - Any Type
- Scala - AnyRef Type
- Scala - Unified Types
- Scala - Dates and Times
- Scala - Ranges
- Scala - Multidimensional Arrays
- Scala - WrappedArray
- Scala - StringBuilder
- Scala - String Interpolation
- Scala - StringContext
- Scala - Type Casting
- Scala var vs val
- Scala Operators
- Scala - Operators
- Scala - Rules for Operators
- Scala - Arithmetic Operators
- Scala - Relational Operators
- Scala - Logical Operators
- Scala - Bitwise Operators
- Scala - Assignment Operators
- Scala - Operators Precedence
- Scala - Symbolic Operators
- Scala - Range Operator
- Scala - String Concatenation Operator
- Scala Conditional Statements
- Scala - IF ELSE
- Scala - IF-ELSE-IF-ELSE Statement
- Scala - Nested IF-ELSE Statement
- Scala Loop Statements
- Scala - Loop Statements
- Scala - while Loop
- Scala - do-while Loop
- Scala - Nested Loops
- Scala - for Loop
- Scala - break Statement
- Scala - yield Keyword
- Scala Classes & Objects
- Scala - Classes & Objects
- Scala - Constructors
- Scala - Auxiliary Constructor
- Scala - Primary Constructor
- Scala - This Keyword
- Scala - Nested Classes
- Scala - Getters and Setters
- Scala - Object Private Fields
- Scala - Singleton Object
- Scala - Companion Objects
- Scala - Creating Executable Programs
- Scala - Stateful Object
- Scala - Enumerations
- Scala - Polymorphism
- Scala - Access Modifiers
- Scala - Apply Method
- Scala - Update Methods
- Scala - UnapplySeq Method
- Scala - Inheritance
- Scala - Extending a Class
- Scala - Method Overloading
- Scala - Method Overriding
- Scala - Generic Classes
- Scala - Generic Functions
- Scala - Superclass Construction
- Scala Methods & Functions
- Scala - Methods
- Scala - Functions
- Scala - Methods vs Functions
- Scala - Main Methods
- Scala - Functions Call-by-Name
- Scala - Functions with Named Arguments
- Scala - Function with Variable Arguments
- Scala - Recursion Functions
- Scala - Default Parameter Values
- Scala - Functions without Parameters
- Scala - Implicit Parameters
- Scala - Higher-Order Functions
- Scala - Nested Functions
- Scala - Extension Methods
- Scala - Anonymous Functions
- Partially Applied Functions
- Scala - Lazy Val
- Scala - Pure Function
- Scala - Currying Functions
- Scala - Control Abstractions
- Scala - Corecursion
- Scala - Unfold
- Scala - Tail Recursion
- Scala - Infinite Sequences
- Scala - Dynamic Invocation
- Scala - Lambda Expressions
- Scala - Polymorphic Functions
- Scala Collections
- Scala - Collections
- Mutable and Immutable Collections
- Scala - Lists
- Scala - Sets
- Scala - Maps
- Scala - TreeMap
- Scala - SortedMap
- Scala - Tuples
- Scala - Iterators
- Scala - Options
- Scala - NumericRange
- Scala - Infinite Streams
- Scala - Parallel Collections
- Scala Advanced Types
- Scala - Union Types
- Scala - Intersection Types
- Scala - Type Aliases
- Scala - Structural Types
- Scala - Match Expression
- Scala - Singleton Type Operator
- Scala - Abstract Types
- Scala - Dependent Types
- Scala - Abstract Type Bounds
- Scala - Higher-Kinded Types
- Scala - Opaque Type Alias
- Scala - Path-Dependent Types
- Scala - Type Lambdas
- Scala - Type Inference
- Scala - Algebraic Data Types
- Scala Pattern Matching
- Scala - Pattern Matching
- Scala - Guards
- Scala - Variables in Patterns
- Scala - Type Patterns
- Scala - The Matchable Trait
- Scala - Matching Arrays
- Scala - Matching Lists
- Scala - Matching Tuples
- Scala - Exception Handling
- Scala - Extractors
- Scala - Pattern Bindings
- Scala - Regular Expressions
- Scala - Case Classes
- Scala - Partial Functions
- Scala - Packaging and Imports
- Scala - Implicit Imports
- Scala - Export Clauses
- Scala - Nested Packages
- Scala - Chained Packages
- Scala - Package Objects
- Scala Files I/O
- Scala - Files I/O
- Scala - Writing Files
- Scala - Listing Files
- Scala - Deleting Directories
- Scala - Check File Exists
- Scala Advanced Concepts
- Scala - Closures
- Scala - Futures
- Scala - Promises
- Scala - Traits
- Scala - Trait Mixins
- Scala - Layered Traits
- Scala - Trait Linearization
- Scala - Sealed Traits
- Scala - Transparent Traits
- Scala - Process Management
- Scala - Scaladoc
- Scala - Literal Type Arithmetic
- Scala - Inline keyword
- Scala - Def, Var & Val
- Scala - Dropped Features
- Scala Unit Testing
- Scala - Unit Testing
- Scala - uTest
- Scala - MUnit
- Scala - ScalaTest Runner
- Scala - ScalaMock
- Scala - JUnit
- Scala - Mocking
- Scala - BDD Testing
Scala - Polymorphism
This chapter takes you through the concept of polymorphism in Scala programming. Polymorphism is a core principle of object-oriented programming. Methods can process objects differently based on their data type and class. It enhances code flexibility and reusability.
Polymorphism in Scala
Using polymorphism, methods can operate on objects of different types using single interface. Polymorphism can be achieved through method overloading and method overriding.
There are basically two types of Polymorphism in Scala:
- Compile-time polymorphism (Method Overloading)
- Runtime polymorphism (Method Overriding)
Compile-time Polymorphism (Method Overloading)
Method overloading is a compile-time polymorphism. There can be multiple methods with same name but different parameter lists within the same class.
Syntax
The syntax of method overloading -
class ClassName {
  def method(parameter1: Type): ReturnType = {
    // Implementation
  }
  def method(parameter1: Type, parameter2: Type): ReturnType = {
    // Implementation
  }
}
Example
The following example shows method overloading in Scala programming -
class Calculator {
  def add(x: Int, y: Int): Int = {
    x + y
  }
  def add(x: Double, y: Double): Double = {
    x + y
  }
}
object Demo {
  def main(args: Array[String]): Unit = {
    val calculator = new Calculator
    println(calculator.add(10, 20))
    println(calculator.add(10.5, 20.5))
  }
}
Save the above program in Demo.scala. The following commands are used to compile and execute this program.
Command
\>scalac Demo.scala \>scala Demo
Output
30 31.0
In the example, Calculator class has two overloaded add methods. One for integers and one for doubles. Demo object shows how correct method is called based on the parameter types.
Runtime Polymorphism (Method Overriding)
Method overriding is a runtime polymorphism. A method in a subclass has the same name and parameter list as a method in its superclass. The subclass method overrides the superclass method.
Syntax
The syntax of the method overriding -
class SuperClass {
  def method(): Unit = {
    // Superclass implementation
  }
}
class SubClass extends SuperClass {
  override def method(): Unit = {
    // Subclass implementation
  }
}
Example
The following example shows method overriding in Scala -
class Animal {
  def sound(): Unit = {
    println("Animal makes a sound")
  }
}
class Dog extends Animal {
  override def sound(): Unit = {
    println("Dog barks")
  }
}
object Demo {
  def main(args: Array[String]): Unit = {
    val animal: Animal = new Animal
    val dog: Animal = new Dog
    animal.sound()
    dog.sound()
  }
}
Save the above program in Demo.scala. The following commands are used to compile and execute this program.
Command
\>scalac Demo.scala \>scala Demo
Output
Animal makes a sound Dog barks
In the example, Dog class overrides the sound method of the Animal class. Demo object shows how different objects respond to the same method call differently.
Polymorphism with Traits
Traits in Scala can also be used to achieve polymorphism. A trait is similar to an interface in Java. Classes can implement multiple traits.
Syntax
The syntax of polymorphism with traits is -
trait TraitName {
  def method(): Unit
}
class ClassName extends TraitName {
  override def method(): Unit = {
    // Implementation
  }
}
Example
The following example shows polymorphism with traits in Scala -
trait Shape {
  def area(): Double
}
class Circle(val radius: Double) extends Shape {
  override def area(): Double = {
    Math.PI * radius * radius
  }
}
class Rectangle(val width: Double, val height: Double) extends Shape {
  override def area(): Double = {
    width * height
  }
}
object Demo {
  def main(args: Array[String]): Unit = {
    val shapes: List[Shape] = List(new Circle(5), new Rectangle(4, 7))
    shapes.foreach(shape => println(shape.area()))
  }
}
Save the above program in Demo.scala. The following commands are used to compile and execute this program.
Command
\>scalac Demo.scala \>scala Demo
Output
78.53981633974483 28.0
In the example, Shape trait defines a method area. Circle and Rectangle classes implement this trait. These classes provide their own implementations of the area method. Demo object shows polymorphism by storing different shapes in a list and calling the area method on each shape.
Polymorphic Functions
Polymorphic functions can operate on arguments of different types. So the function is more flexible and reusable.
Syntax
The syntax of the polymorphic function is -
def identity[T](x: T): T = x
Example
The following example demonstrates a polymorphic function in Scala programming -
object Demo {
  def identity[T](x: T): T = x
  def main(args: Array[String]): Unit = {
    println(identity(42))
    println(identity("Hello"))
  }
}
Save the above program in Demo.scala. The following commands are used to compile and execute this program.
Command
\>scalac Demo.scala \>scala Demo
Output
42 Hello
In the example, the identity function is a polymorphic function that returns its argument. The Demo object shows how the function can operate on both integers and strings.
Scala Polymorphism Summary
- Using polymorphism, methods can operate on objects of different types using single interface. So, code has more flexibility and reusability.
- Method Overloading (compile-time polymorphism) has multiple methods with the same name but different parameter lists within the same class.
- Method Overriding (runtime polymorphism) has subclass method to override superclass method with the same name and parameter list.
- Traits can be used to achieve polymorphism. So classes to implement multiple traits and provide different method implementations.
- Polymorphic Functions can operate on arguments of different types. So, it increases function flexibility.