Class, Primary Constructor, Secondary Constructor and Init Block
fun main(args: Array) {
var student = Student("Steve", 10)
println(student.id)
}
class Student(var name: String) {
var id: Int = -1
init {
println("Student has got a name as $name and id is $id")
}
constructor(n: String, id: Int): this(n) {
// The body of the secondary constructor is called after init block
this.id = id
}
}
Inheritance
fun main(args: Array) {
var dog = Dog()
dog.bread = "labra"
dog.color = "black"
dog.bark()
dog.eat()
var cat = Cat()
cat.age = 7
cat.color = "brown"
cat.meow()
cat.eat()
var animal = Animal()
animal.color = "white"
animal.eat()
}
open class Animal { // Super class / Parent class / Base class
var color: String = ""
fun eat() {
println("Eat")
}
}
class Dog : Animal() { // Sub class / Child class / Derived class
var bread: String = ""
fun bark() {
println("Bark")
}
}
class Cat : Animal() { // Sub class / Child class / Derived class
var age: Int = -1
fun meow() {
println("Meow")
}
}
by default Class are:
*public
*final
So for inheritance you need to make a class --OPEN--
Override
fun main(args: Array) {
var dog = MyDog()
println(dog.color)
dog.eat()
}
open class MyAnimal {
open var color: String = "White"
open fun eat() {
println("Animal Eating")
}
}
class MyDog : MyAnimal() {
var bread: String = ""
override var color: String = "Black"
fun bark() {
println("Bark")
}
override fun eat() {
super.eat()
println("Dog is eating")
}
}
Overriding Member Functions and variable
Just like Kotlin classes, members of a Kotlin class are also public and final by default.
To allow a member function to be overridden, you need to mark it with the open modifier.
Moreover, The derived class that overrides a base class function must use the override modifier, otherwise, the compiler will generate an error
Inheritance with Primary and Secondary Constructors
fun main(args: Array) {
var dog = TheDog("Black", "Pug")
}
open class TheAnimal { // Super class / Parent class / Base class
var color: String = ""
constructor(color: String) {
this.color = color
println("From Animal: $color")
}
}
class TheDog : TheAnimal { // Sub class / Child class / Derived class
var bread: String = ""
constructor(color: String, breed: String): super(color) {
this.bread = breed
println("From Dog: $color and $breed")
}
}
Inheritance with Primary and Secondary Constructors
Visibility Modifiers
open class Person { // Super class
private val a = 1
protected val b = 2
internal val c = 3
val a = 10
}
class Indian: Person() { // Sub class
fun printTest() {
print(a)
// a is not visible
// b, c, d are visible
}
}
class TestClass {
fun testing() {
var person = Person()
print(person.a)
// a, b are not visible
// c, d are visible
}
}
public always is visible
protected visible inside sub class, not visible inside other class
internal visible from module
private just visible from yourClass.kt
Abstract Class
fun main(args: Array) {
// var person = MyPerson() // Not allowed. You cannot create instance of abstract class
var person = Indian() // Allowed. Abstract Super class reference variable
// pointing to child class object.
person.name = "Steve"
person.eat()
person.goToSchool()
}
abstract class MyPerson { // you cannot create instance of abstract class
abstract var name: String
abstract fun eat() // abstract properties are 'open' by default
open fun getHeight() {} // A 'open' function ready to be overridden
fun goToSchool() {} // A normal function
}
class Indian: MyPerson() {
override var name: String = "dummy_indian_name"
override fun eat() {
// Our own code
}
}
The Role or Abstract class is to just provide set of methods and properties
Abstract Class are Partlally defined class
Abstract Methods have no body when declared
Abstract Property cannot be initialized when declared
CONCLUSION
you cannot create intance/objects of ABSTRACT class
you need to override ABSTRACT methods, propertics inside Derived class
Interface
fun main(args: Array) {
var myButton = MyButton()
myButton.onTouch()
myButton.onClick()
}
interface MyInterfaceListener { // You cannot create the instance of interface
fun onTouch() // Methods in interface are abstract by default
fun onClick() { // Normal methods are public and open by default but NOT FINAL
println("MyInterfaceListener: onClick")
}
}
interface MySecondInterface { // You cannot create the instance of interface
fun onTouch() { // Normal Method
println("MySecondInterface: onTouch")
}
fun onClick() { // Normal methods are public and open by default but NOT FINAL
println("MySecondInterface: onClick")
}
}
class MyButton: MyInterfaceListener, MySecondInterface {
override fun onTouch() {
super.onClick()
super.onClick()
}
override fun onClick() {
super.onTouch()
}
}
Interface can contains both NORMAL methods and ABSTRACT methods
But they contain only ABSTRACT PROPERTY
Interface is not class
You can not create instance of an INTERFACE
similar to an ABSTRACT class
Data Clases
fun main(args: Array) {
var user1 = User("Sam", 10)
var user2 = User("Sam", 10)
println(user1.toString())
if (user1 == user2)
println("Equal")
else
println("Not equal")
var newUser = user1.copy(id = 25)
println(newUser)
}
data class User(var name: String, var id: Int)
Any class contains functions such as:
equals():Boolean
hasCode():Int
toString:String
------------------
Kotlin creates a copy() too
Sealed
sealed class Expr
data class Const(val number: Double) : Expr()
data class Sum(val e1: Expr, val e2: Expr) : Expr()
object NotANumber : Expr()
fun eval(expr: Expr): Double = when(expr) {
is Const -> expr.number
is Sum -> eval(expr.e1) + eval(expr.e2)
NotANumber -> Double.NaN
// the `else` clause is not required because we've covered all the cases
}
Sealed classes are used for representing restricted class hierarchies
when a value can have one of the types from a limited set, but cannot have any other type
They are, in a sense, an extension of enum classes:
- the set of values for an enum type is also restricted,
- but each enum constant exists only as a single instance
- whereas a subclass of a sealed class can have multiple instances which can contain state