练习:Kotlin 基础知识

1. 开始之前

既然你已经努力学习了 Kotlin 编程的基础知识,现在是时候将你学到的知识付诸实践了。

这些练习测试你对所学概念的理解。它们基于现实世界的用例,其中一些你可能以前作为用户遇到过。

按照说明在 Kotlin Playground 中为每个练习找到解决方案。如果你卡住了,一些练习有提示可以帮助你。每个练习的解决方案代码都可以在最后找到,但建议你在查看答案之前先解决这些练习。

按照你舒适的速度完成练习。练习有持续时间估计,但这只是估计,所以你不必遵守它们。花尽可能多的时间仔细解决每个问题。解决方案只是解决练习的一种方法,因此请随意尝试任何你感到舒适的方式。

先决条件

你需要什么

  • Kotlin Playground

2. 移动通知

通常,你的手机会为你提供通知摘要。

在以下代码片段中提供的初始代码中,编写一个程序,根据你收到的通知数量打印摘要消息。消息应包括

  • 当通知少于 100 条时,显示确切的通知数量。
  • 当通知数量为 100 条或更多时,显示 99+
fun main() {
    val morningNotification = 51
    val eveningNotification = 135
    
    printNotificationSummary(morningNotification)
    printNotificationSummary(eveningNotification)
}


fun printNotificationSummary(numberOfMessages: Int) {
    // Fill in the code.
}

完成 printNotificationSummary() 函数,以便程序打印这些行

You have 51 notifications.
Your phone is blowing up! You have 99+ notifications.

3. 电影票价格

电影票的价格通常根据影迷的年龄而有所不同。

在以下代码片段中提供的初始代码中,编写一个程序来计算这些基于年龄的票价

  • 12 岁或以下儿童的票价为 15 美元。
  • 13 至 60 岁人士的标准票价为 30 美元。在星期一,同一年龄段的标准票价可享受 25 美元的折扣。
  • 61 岁及以上老年人的票价为 20 美元。假设影迷的最大年龄为 100 岁。
  • 当用户输入的年龄超出年龄规格时,使用 -1 值表示价格无效。
fun main() {
    val child = 5
    val adult = 28
    val senior = 87
    
    val isMonday = true
    
    println("The movie ticket price for a person aged $child is \$${ticketPrice(child, isMonday)}.")
    println("The movie ticket price for a person aged $adult is \$${ticketPrice(adult, isMonday)}.")
    println("The movie ticket price for a person aged $senior is \$${ticketPrice(senior, isMonday)}.")
}

fun ticketPrice(age: Int, isMonday: Boolean): Int {
    // Fill in the code.
}

完成 ticketPrice() 函数,以便程序打印这些行

The movie ticket price for a person aged 5 is $15.
The movie ticket price for a person aged 28 is $25.
The movie ticket price for a person aged 87 is $20.

4. 温度转换器

世界上主要使用三种温度标度:摄氏度、华氏度和开尔文。

在以下代码片段中提供的初始代码中,编写一个程序,使用以下公式将温度从一个标度转换为另一个标度

  • 摄氏度到华氏度:° F = 9/5 (° C) + 32
  • 开尔文到摄氏度:° C = K - 273.15
  • 华氏度到开尔文:K = 5/9 (° F - 32) + 273.15

请注意,String.format("%.2f", /* measurement */ ) 方法用于将数字转换为带有 2 位小数的 String 类型。

fun main() {
    // Fill in the code.
}


fun printFinalTemperature(
    initialMeasurement: Double, 
    initialUnit: String, 
    finalUnit: String, 
    conversionFormula: (Double) -> Double
) {
    val finalMeasurement = String.format("%.2f", conversionFormula(initialMeasurement)) // two decimal places
    println("$initialMeasurement degrees $initialUnit is $finalMeasurement degrees $finalUnit.")
}

完成 main() 函数,使其调用 printFinalTemperature() 函数并打印以下行。你需要为温度和转换公式传递参数。提示:你可能需要使用 Double 值来避免在除法运算期间出现 Integer 截断。

27.0 degrees Celsius is 80.60 degrees Fahrenheit.
350.0 degrees Kelvin is 76.85 degrees Celsius.
10.0 degrees Fahrenheit is 260.93 degrees Kelvin.

5. 歌曲目录

想象一下,你需要创建一个音乐播放器应用程序。

创建一个可以表示歌曲结构的类。 Song 类必须包含以下代码元素

  • 标题、艺术家、出版年份和播放次数的属性
  • 一个属性,指示歌曲是否流行。如果播放次数少于 1,000 次,则认为它不受欢迎。
  • 一个方法,以这种格式打印歌曲描述

"[标题],由 [艺术家] 演唱,发行于 [出版年份]。"

6. 互联网个人资料

通常,你需要填写包含必填字段和非必填字段的在线网站个人资料。例如,你可以添加你的个人信息并链接到推荐你注册个人资料的其他人员。

在以下代码片段中提供的初始代码中,编写一个程序来打印一个人的个人资料详细信息。

fun main() {    
    val amanda = Person("Amanda", 33, "play tennis", null)
    val atiqah = Person("Atiqah", 28, "climb", amanda)
    
    amanda.showProfile()
    atiqah.showProfile()
}


class Person(val name: String, val age: Int, val hobby: String?, val referrer: Person?) {
    fun showProfile() {
       // Fill in code 
    }
}

完成 showProfile() 函数,以便程序打印这些行

Name: Amanda
Age: 33
Likes to play tennis. Doesn't have a referrer.

Name: Atiqah
Age: 28
Likes to climb. Has a referrer named Amanda, who likes to play tennis.

7. 可折叠手机

通常,当按下电源按钮时,手机屏幕会打开和关闭。相反,如果可折叠手机折叠起来,则按下电源按钮时,可折叠手机上的主要内屏不会打开。

在以下代码片段中提供的初始代码中,编写一个继承自 Phone 类的 FoldablePhone 类。它应该包含以下内容

  • 一个属性,指示手机是否折叠。
  • Phone 类不同的 switchOn() 函数行为,以便它仅在手机未折叠时才打开屏幕。
  • 更改折叠状态的方法。
class Phone(var isScreenLightOn: Boolean = false){
    fun switchOn() {
        isScreenLightOn = true
    }
    
    fun switchOff() {
        isScreenLightOn = false
    }
    
    fun checkPhoneScreenLight() {
        val phoneScreenLight = if (isScreenLightOn) "on" else "off"
        println("The phone screen's light is $phoneScreenLight.")
    }
}

8. 特殊拍卖

通常在拍卖中,出价最高者决定商品的价格。在这个特殊的拍卖中,如果某件商品没有竞拍者,则该商品将自动以最低价出售给拍卖行。

在以下代码片段中提供的初始代码中,你得到一个接受可空 Bid? 类型作为参数的 auctionPrice() 函数

fun main() {
    val winningBid = Bid(5000, "Private Collector")
    
    println("Item A is sold at ${auctionPrice(winningBid, 2000)}.")
    println("Item B is sold at ${auctionPrice(null, 3000)}.")
}

class Bid(val amount: Int, val bidder: String)
 
fun auctionPrice(bid: Bid?, minimumPrice: Int): Int {
   // Fill in the code.
}

完成 auctionPrice() 函数,以便程序打印这些行

Item A is sold at 5000.
Item B is sold at 3000.

9. 解决方案代码

移动通知

该解决方案使用 if/else 语句根据接收到的通知消息数量打印相应的通知摘要消息

fun main() {
    val morningNotification = 51
    val eveningNotification = 135
    
    printNotificationSummary(morningNotification)
    printNotificationSummary(eveningNotification)
}


fun printNotificationSummary(numberOfMessages: Int) {
    if (numberOfMessages < 100) {
        println("You have ${numberOfMessages} notifications.")
    } else {
        println("Your phone is blowing up! You have 99+ notifications.")
    }
}

电影票价格

该解决方案使用 when 表达式根据影迷的年龄返回相应的票价。它还使用一个简单的 if/else 表达式作为其中一个 when 表达式的分支,为标准票价添加附加条件。

else 分支中的票价返回 -1 值,这表示为 else 分支设置的价格无效。更好的实现是让 else 分支抛出异常。你将在以后的单元中学习异常处理。

fun main() {
    val child = 5
    val adult = 28
    val senior = 87
    
    val isMonday = true
    
    println("The movie ticket price for a person aged $child is \$${ticketPrice(child, isMonday)}.")
    println("The movie ticket price for a person aged $adult is \$${ticketPrice(adult, isMonday)}.")
    println("The movie ticket price for a person aged $senior is \$${ticketPrice(senior, isMonday)}.")
}
 
fun ticketPrice(age: Int, isMonday: Boolean): Int {
    return when(age) {
        in 0..12 -> 15
        in 13..60 -> if (isMonday) 25 else 30
        in 61..100 -> 20
        else -> -1
    }
}

温度转换器

该解决方案要求你将函数作为参数传递给 printFinalTemperature() 函数。最简洁的解决方案将 lambda 表达式作为参数传递,使用 it 参数引用代替参数名称,并使用尾随 lambda 语法。

fun main() {    
        printFinalTemperature(27.0, "Celsius", "Fahrenheit") { 9.0 / 5.0 * it + 32 }
        printFinalTemperature(350.0, "Kelvin", "Celsius") { it - 273.15 }
        printFinalTemperature(10.0, "Fahrenheit", "Kelvin") { 5.0 / 9.0 * (it - 32) + 273.15 }
}


fun printFinalTemperature(
    initialMeasurement: Double, 
    initialUnit: String, 
    finalUnit: String, 
    conversionFormula: (Double) -> Double
) {
    val finalMeasurement = String.format("%.2f", conversionFormula(initialMeasurement)) // two decimal places
    println("$initialMeasurement degrees $initialUnit is $finalMeasurement degrees $finalUnit.")
}

歌曲目录

该解决方案包含一个 Song 类,它具有接受所有必需参数的默认构造函数。 Song 类还有一个使用自定义 getter 函数的 isPopular 属性,以及一个打印自身描述的方法。你可以在 main() 函数中创建类的实例并调用其方法来测试实现是否正确。你可以使用下划线编写大数字,例如 1_000_000 值,以使其更易于阅读。

fun main() {    
    val brunoSong = Song("We Don't Talk About Bruno", "Encanto Cast", 2022, 1_000_000)
    brunoSong.printDescription()
    println(brunoSong.isPopular)
}


class Song(
    val title: String, 
    val artist: String, 
    val yearPublished: Int, 
    val playCount: Int
){
    val isPopular: Boolean
        get() = playCount >= 1000

    fun printDescription() {
        println("$title, performed by $artist, was released in $yearPublished.")
    }   
}

当你在实例的方法上调用 println() 函数时,程序可能会打印此输出

We Don't Talk About Bruno, performed by Encanto Cast, was released in 2022.
true

互联网个人资料

该解决方案在各种 if/else 语句中包含空检查,以便根据各种类属性是否为 null 打印不同的文本

fun main() {    
    val amanda = Person("Amanda", 33, "play tennis", null)
    val atiqah = Person("Atiqah", 28, "climb", amanda)
    
    amanda.showProfile()
    atiqah.showProfile()
}


class Person(val name: String, val age: Int, val hobby: String?, val referrer: Person?) {
    fun showProfile() {
        println("Name: $name")
        println("Age: $age")
        if(hobby != null) {
            print("Likes to $hobby. ")
        }
        if(referrer != null) {
            print("Has a referrer named ${referrer.name}")
            if(referrer.hobby != null) {
                print(", who likes to ${referrer.hobby}.")
            } else {
                print(".")
            }
        } else {
            print("Doesn't have a referrer.")
        }
        print("\n\n")
    }
}

可折叠手机

为了使Phone类成为父类,需要通过在类名前添加open关键字来将该类声明为开放类。要重写FoldablePhone类中的switchOn()方法,需要通过在Phone类中的该方法前添加open关键字来将其声明为开放方法。

解决方案包含一个FoldablePhone类,它有一个默认构造函数,该构造函数为isFolded参数包含一个默认参数。FoldablePhone类还包含两个方法,用于将isFolded属性更改为truefalse值。它还重写了从Phone类继承的switchOn()方法。

您可以在main()函数中创建该类的实例,并调用其方法来测试实现是否正确。

open class Phone(var isScreenLightOn: Boolean = false){
    open fun switchOn() {
        isScreenLightOn = true
    }
    
    fun switchOff() {
        isScreenLightOn = false
    }
    
    fun checkPhoneScreenLight() {
        val phoneScreenLight = if (isScreenLightOn) "on" else "off"
        println("The phone screen's light is $phoneScreenLight.")
    }
}

class FoldablePhone(var isFolded: Boolean = true): Phone() {
    override fun switchOn() {
        if (!isFolded) {
            isScreenLightOn = true
        }
    }
    
    fun fold() {
        isFolded = true
    }
    
    fun unfold() {
        isFolded = false
    }
}

fun main() {    
    val newFoldablePhone = FoldablePhone()
    
    newFoldablePhone.switchOn()
    newFoldablePhone.checkPhoneScreenLight()
    newFoldablePhone.unfold()
    newFoldablePhone.switchOn()
    newFoldablePhone.checkPhoneScreenLight()
}

输出结果如下:

The phone screen's light is off.
The phone screen's light is on.

特别拍卖

该解决方案使用?.安全调用运算符和?: Elvis运算符返回正确的价格。

fun main() {
    val winningBid = Bid(5000, "Private Collector")
    
    println("Item A is sold at ${auctionPrice(winningBid, 2000)}.")
    println("Item B is sold at ${auctionPrice(null, 3000)}.")
}

class Bid(val amount: Int, val bidder: String)

fun auctionPrice(bid: Bid?, minimumPrice: Int): Int {
    return bid?.amount ?: minimumPrice
}

10. 附加练习

要了解更多关于Kotlin语言的实践,请查看JetBrains Academy提供的Kotlin核心课程。要跳转到特定主题,请访问知识地图查看课程涵盖的主题列表。