Haskell - Types and Type Class


Advertisements

Haskell is a functional language and it is strictly typed, which means the data type used in the entire application will be known to the compiler at compile time.

Inbuilt Type Class

In Haskell, every statement is considered as a mathematical expression and the category of this expression is called as a Type. You can say that "Type" is the data type of the expression used at compile time.

To learn more about the Type, we will use the ":t" command. In a generic way, Type can be considered as a value, whereas Type Class can be the considered as a set of similar kind of Types. In this chapter, we will learn about different Inbuilt Types.

Int

Int is a type class representing the Integer types data. Every whole number within the range of 2147483647 to -2147483647 comes under the Int type class. In the following example, the function fType() will behave according to its type defined.

fType :: Int -> Int -> Int 
fType x y = x*x + y*y
main = print (fType 2 4) 

Here, we have set the type of the function fType() as int. The function takes two int values and returns one int value. If you compile and execute this piece of code, then it will produce the following output −

sh-4.3$ ghc -O2 --make *.hs -o main -threaded -rtsopts 
sh-4.3$ main
20

Integer

Integer can be considered as a superset of Int. This value is not bounded by any number, hence an Integer can be of any length without any limitation. To see the basic difference between Int and Integer types, let us modify the above code as follows −

fType :: Int -> Int -> Int 
fType x y = x*x + y*y 
main = print (fType 212124454 44545454454554545445454544545)

If you compile the above piece of code, the following error message will be thrown −

main.hs:3:31: Warning:            
   Literal 44545454454554545445454544545 is out of the Int range -
   9223372036854775808..9223372036854775807 
Linking main ...

This error occurred because our function fType() expecting one Int type value, and we are passing some real big Int type value. To avoid this error, Let us modify the type "Int" with "Integer" and observe the difference.

fType :: Integer -> Integer -> Integer 
fType x y = x*x + y*y 
main = print (fType 212124454 4454545445455454545445445454544545) 

Now, it will produce the following output −

sh-4.3$ main
1984297512562793395882644631364297686099210302577374055141

Float

Take a look at the following piece of code. It shows how Float type works in Haskell −

fType :: Float -> Float -> Float 
fType x y = x*x + y*y 
main = print (fType 2.5 3.8)

The function takes two float values as the input and yields another float value as the output. When you compile and execute this code, it will produce the following output −

sh-4.3$ main
20.689999 

Double

Double is a floating point number with double precision at the end. Take a look at the following example −

fType :: Double -> Double -> Double 
fType x y = x*x + y*y 
main = print (fType 2.56 3.81)

When you execute the above piece of code, it will generate the following output −

sh-4.3$ main 
21.0697

Bool

Bool is a Boolean Type. It can be either True or False. Execute the following code to understand how the Bool type works in Haskell −

main = do  
   let x = True 
   
   if x == False 
      then putStrLn "X matches with Bool Type" 
   else putStrLn "X is not a Bool Type" 

Here, we are defining a variable "x" as a Bool and comparing it with another Boolean value to check its originality. It will produce the following output −

sh-4.3$ main
X is not a Bool Type 

Char

Char represent Characters. Anything within a single quote is considered as a Character. In the following code, we have modified our previous fType() function to accept Char value and return Char value as output.

fType :: Char-> Char 
fType x = 'K' 
main = do  
   let x = 'v' 
   print (fType x) 

The above piece of code will call fType() function with a char value of 'v' but it returns another char value, that is, 'K'. Here is its output −

sh-4.3$ main 
'K'

Note that we are not going to use these types explicitly because Haskell is intelligent enough to catch the type before it is declared. In the subsequent chapters of this tutorial, we will see how different types and Type classes make Haskell a strongly typed language.

EQ Type Class

EQ type class is an interface which provides the functionality to test the equality of an expression. Any Type class that wants to check the equality of an expression should be a part of this EQ Type Class.

All standard Type classes mentioned above is a part of this EQ class. Whenever we are checking any equality using any of the types mentioned above, we are actually making a call to EQ type class.

In the following example, we are using the EQ Type internally using the "==" or "/=" operation.

main = do 
   if 8 /= 8 
      then putStrLn "The values are Equal" 
   else putStrLn "The values are not Equal"

It will yield the following output −

sh-4.3$ main 
The values are not Equal 

Ord Type Class

Ord is another interface class which gives us the functionality of ordering. All the types that we have used so far are a part of this Ord interface. Like EQ interface, Ord interface can be called using ">", "<", "<=", ">=", "compare".

Please find below example where we used “compare ” functionality of this Type Class.

main = print (4 <= 2) 

Here, the Haskell compiler will check if 4 is less than or equal to 2. Since it is not, the code will produce the following output −

sh-4.3$ main 
False

Show

Show has a functionality to print its argument as a String. Whatever may be its argument, it always prints the result as a String. In the following example, we will print the entire list using this interface. "show" can be used to call this interface.

main = print (show [1..10]) 

It will produce the following output on the console. Here, the double quotes indicate that it is a String type value.

sh-4.3$ main 
"[1,2,3,4,5,6,7,8,9,10]" 

Read

Read interface does the same thing as Show, but it won’t print the result in String format. In the following code, we have used the read interface to read a string value and convert the same into an Int value.

main = print (readInt "12") 
readInt :: String -> Int 
readInt = read 

Here, we are passing a String variable ("12") to the readInt method which in turn returns 12 (an Int value) after conversion. Here is its output −

sh-4.3$ main 
12

Enum

Enum is another type of Type class which enables the sequential or ordered functionality in Haskell. This Type class can be accessed by commands such as Succ, Pred, Bool, Char, etc.

The following code shows how to find the successor value of 12.

main = print (succ 12) 

It will produce the following output −

sh-4.3$ main
13

Bounded

All the types having upper and lower bounds come under this Type Class. For example, Int type data has maximum bound of "9223372036854775807" and minimum bound of "-9223372036854775808".

The following code shows how Haskell determines the maximum and minimum bound of Int type.

main = do 
   print (maxBound :: Int) 
   print (minBound :: Int) 

It will produce the following output −

sh-4.3$ main
9223372036854775807
-9223372036854775808

Now, try to find the maximum and minimum bound of Char, Float, and Bool types.

Num

This type class is used for numeric operations. Types such as Int, Integer, Float, and Double come under this Type class. Take a look at the following code −

main = do 
   print(2 :: Int)  
   print(2 :: Float) 

It will produce the following output −

sh-4.3$ main
2
2.0

Integral

Integral can be considered as a sub-class of the Num Type Class. Num Type class holds all types of numbers, whereas Integral type class is used only for integral numbers. Int and Integer are the types under this Type class.

Floating

Like Integral, Floating is also a part of the Num Type class, but it only holds floating point numbers. Hence, Float and Double come under this type class.

Custom Type Class

Like any other programming language, Haskell allows developers to define user-defined types. In the following example, we will create a user-defined type and use it.

data Area = Circle Float Float Float  
surface :: Area -> Float   
surface (Circle _ _ r) = pi * r ^ 2   
main = print (surface $ Circle 10 20 10 ) 

Here, we have created a new type called Area. Next, we are using this type to calculate the area of a circle. In the above example, "surface" is a function that takes Area as an input and produces Float as the output.

Keep in mind that "data" is a keyword here and all user-defined types in Haskell always start with a capital letter.

It will produce the following output −

sh-4.3$ main
314.15927
Advertisements