 
- 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 - Type Casting
Like other programming languages, Scala provides type casting. It is also known as coercion. You should be aware about type casting in Scala because it has type erasure property. So, it can be dangerous. If you type casting in the wrong way then there can be bugs in your program/ project.
Scala Type Casting
In Scala, there are three methods to convert one type of an object to another type. These are −
- Value Type casting
- Type casting using asInstanceOf[T] method
- Type casting using match statement
We will discuss these methods in brief.
Value Type Casting
You can convert one type to another type is defined as −
Byte --> Short --> Int --> Long --> Float --> Double
                  ^
                  |
                 Char
 
You can convert one type to another as arrows suggested above. For example, you can convert Int to Long but reverse is not possible, that is the wrong way. It can cause loss of value and error.
Example
object Demo {
   def main(args: Array[String]): Unit = {
      val shortValue: Short = 0xF
      val intValue: Int = shortValue
      println(s"Short value: $shortValue")
      println(s"Int value (from Short): $intValue")
    
      // The following line will not compile
     // val shortValue2: Short = intValue
   }
}
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
Short value: 15 Int value (from Short): 15
Note that there is no need for special methods and syntax for type casting.
Type Casting Using asInstanceOf[T]
You can type cast using asInstanceOf[T] method. You can also use the companion method in Scala in conjunction with it.
Consider these classes −
class T1 class T2 extends T1 class T3
Now, use asInstanceOf[T] for type casting −
class T1 class T2 extends T1 class T3
Example
class T1
class T2 extends T1
class T3
object Demo {
   def main(args: Array[String]): Unit = {
      val customT2 = new T2
      val customT3 = new T3
      val customT1: T1 = customT2.asInstanceOf[T1]
    
      println(s"customT2 is instance of T1: ${customT2.isInstanceOf[T1]}")
      println(s"customT3 is instance of T1: ${customT3.isInstanceOf[T1]}")
    
      // The following line will cause a run-time error
      // val anotherT1 = customT3.asInstanceOf[T1]
   }
}
 
This is the same as in the Java programming language. Now we will see how it can be improved.
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
customT2 is instance of T1: true customT3 is instance of T1: false
Type Casting Using Pattern Matching
You can use asInstanceOf[T] and isInstanceOf[T] as discussed above. But if you have complex logic which is based on object type. Then you end up with if-else statements which are difficult to maintain. You can use Pattern Matching here to solve this problem.
Example
Try(retrieveAnotherExample()) match {
   case Success(result) if result.isEmpty =>
      // Handle empty result
   case Success(result) =>
      // Process successful response
   case Failure(e: AnotherSpecificException) =>
      // Handle a specific exception
   case Failure(e: YetAnotherException) =>
      // Handle another specific exception
   case Failure(e: IOException) =>
      // Generic IO/network error handling here
   case Failure(t) =>
      // Handle other non-IOException errors
}
There are various benefits of using pattern matching here. You can handle errors, integrate with legacy code that cannot be changed. You can use Scala functions for simpler code. Let users focus on app code and not create extra classes.
For legacy Java code, you need to integrate it carefully. Some frameworks have unclear results. To use in Scala, you should cast to clear types.
Type Erasure
You need to be aware about type erasure in Scala. It happens due to JVM handling generic types. You can parameterize a trait or class in generic types. Generic types are mostly used with collections classes of Scala.
Example
val temperatures: Array[Double] = Array(25.5, 30.0, 22.3, 28.7)
val studentGrades: Map[String, Double] = Map("Alice" -> 85.5, "Bob" -> 92.0, "Charlie" -> 78.3)
As you can see here, collections can only accept objects of a particular type. But, this is a compiler time feature. At runtime, both collections look like any other type Array or Map.
To understand this concept, consider following example,
object Demo {
  def main(args: Array[String]): Unit = {
    val temperatures: Array[Double] = Array(25.5, 30.0, 22.3, 28.7)
    val studentGrades: Map[String, Double] = Map("Alice" -> 85.5, "Bob" -> 92.0, "Charlie" -> 78.3)
    
    println(s"Temperatures: ${temperatures.mkString(", ")}")
    println(s"Student Grades: ${studentGrades.mkString(", ")}")
    
    val l3 = List(4.5, 6.7, 8.1)
    val l4 = List("x", "y", "z")
    
    println(s"l3 is instance of List[Double]: ${l3.isInstanceOf[List[Double]]}")
    println(s"l4 is instance of List[String]: ${l4.isInstanceOf[List[String]]}")
  }
}
 
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
Temperatures: 25.5, 30.0, 22.3, 28.7 Student Grades: Alice -> 85.5, Bob -> 92.0, Charlie -> 78.3 l3 is instance of List[Double]: true l4 is instance of List[String]: true
Type Ascriptions
Type ascriptions are same type annotations. At compile time, ascriptions are up-casting for values to give hints for type-checking.
There can be many types to apply to an expression, because of different types of inheritance. You can provide the value that you expect from a specific expression using this ascription type. There is a special symbol (:) for ascription type in Scala.
For example,
val result = Some(42.5): Option[Double]
In the above example, we have provided expression type Some(42.5) explicitly as Option[Double].
To pass a sequence type to a vararg method, you need to use type ascription: ': _* special symbol. For example,
object Demo {
  def main(args: Array[String]): Unit = {
    val result = Some(42.5): Option[Double]
    println(s"Result with type ascription: $result")
    def varargFn(numbers: Int*): Int = numbers.sum
    val inputNumbers = Seq(1, 2, 3, 4)
    println(s"Sum of input numbers: ${varargFn(inputNumbers: _*)}")
  }
}
In the above example, we convert 'input' from Seq[Int] to vararg using type ascription.
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
Result with type ascription: Some(42.5) Sum of input numbers: 10
Note that type ascription is different from asInstanceOf(). You can use type ascription during compilation and it avoids runtime errors. Whereas, asInstanceOf() method is used at runtime.