Kotlin的擴(kuò)展(Extension)特性,你了解了嗎?
Kotlin擴(kuò)展(Extension)特性允許為現(xiàn)有的類添加新的函數(shù)和屬性,而無(wú)需繼承該類或使用裝飾器模式。可以在不修改原始類的情況下,為它添加新的行為。
在實(shí)際編程當(dāng)中是非常有用的功能,具體場(chǎng)景如:我們想修改JDK中的String,想在它的基礎(chǔ)上增加一個(gè)方法"lastElement() "來(lái)獲取末尾元素,如果使用Java,我們是無(wú)法通過常規(guī)手段實(shí)現(xiàn)的,因?yàn)槲覀儫o(wú)法修改JDK的源碼。
擴(kuò)展函數(shù)
fun ClassName.functionName(parameters) {
    // 函數(shù)體
}ClassName是要添加函數(shù)的類名,functionName是新函數(shù)的名稱,parameters是函數(shù)的參數(shù)列表,函數(shù)體是函數(shù)的實(shí)際實(shí)現(xiàn)。
例如,我們可以向String類添加一個(gè)名為lastElement的擴(kuò)展函數(shù),用于來(lái)獲取末尾元素:
fun String.lastElement(): Char? {
    if (this.isEmpty()) {
        return null
    }
    return this[length - 1]
}
// 使用擴(kuò)展函數(shù)
fun main() {
    val msg = "Hello Wolrd"
    // lastElement就像String的成員方法一樣可以直接調(diào)用
    val last = msg.lastElement() // last = d
}lastElement函數(shù)就會(huì)在所有String對(duì)象上可用,而不需要修改String類的源代碼。
擴(kuò)展函數(shù)實(shí)現(xiàn)原理,反編譯示例代碼:
public final class ExtKt {
   public static final Character lastElement(String $this) {
      CharSequence var1 = (CharSequence)$this;
      if (var1.length() == 0) {
        return null
      }
      return  var1.charAt(var1.length() - 1);
   }
}
public static final void main() {
  String msg = "Hello Wolrd";
  Character last = ExtKt.lastElement(msg);
}原本定義在String類型上面的擴(kuò)展函數(shù)lastElement(),變成了一個(gè)普通的靜態(tài)方法。另外,之前定義的擴(kuò)展函數(shù)lastElement()是沒有參數(shù)的,但反編譯后的Java代碼中,lastElement(String $this)多了一個(gè)String類型的參數(shù)。原本msg.lastElement()的地方變成了ExtKt.lastElement(msg),這說明,Kotlin編寫的擴(kuò)展函數(shù)調(diào)用代碼,最終會(huì)變成靜態(tài)方法的調(diào)用。
擴(kuò)展屬性
Kotlin中的擴(kuò)展屬性允許我們向現(xiàn)有的類添加新的屬性,而無(wú)需繼承該類或使用裝飾者模式。擴(kuò)展屬性的語(yǔ)法與擴(kuò)展函數(shù)類似,但是在屬性名之前需要指定接收者類型。
還是以lastElement為例,以擴(kuò)展屬性的方式實(shí)現(xiàn):
// 接收者類型
val String.lastElement: Char?
    get() = if (isEmpty()) {
            null
        } else {
            get(length - 1)
        }
fun main() {
    val msg = "Hello Wolrd"
    // lastElement就像String的成員屬性一樣可以直接調(diào)用
    val last = msg.lastElement // last = d
}需要注意的是,擴(kuò)展屬性并不會(huì)真正地向類中添加新的屬性,它只是提供了一種便捷的方式來(lái)訪問現(xiàn)有類的屬性或計(jì)算新的屬性值。不管是擴(kuò)展函數(shù)還是擴(kuò)展屬性,它本質(zhì)上都會(huì)變成一個(gè)靜態(tài)的方法。
應(yīng)用場(chǎng)景
Kotlin的擴(kuò)展特性允許開發(fā)者向現(xiàn)有的類添加新的方法和屬性,而無(wú)需繼承該類或使用裝飾者模式。
- 擴(kuò)展第三方庫(kù):可以使用擴(kuò)展函數(shù)為第三方庫(kù)中的類添加額外的功能,而無(wú)需修改原始類的源代碼。
 - 使代碼更具可讀性:可以通過為常用的類添加自定義方法,使代碼更易讀、更易維護(hù)。
 - 適配特定平臺(tái):可以使用擴(kuò)展函數(shù)為特定平臺(tái)(如Android或iOS)上的類添加平臺(tái)特定的功能,而無(wú)需在通用代碼中添加平臺(tái)相關(guān)的邏輯。
 - 減少重復(fù)代碼:可以通過擴(kuò)展函數(shù)將一些重復(fù)的操作封裝成新的方法,從而減少代碼重復(fù)性。
 
當(dāng)然,擴(kuò)展特性有一些使用限制:
- 擴(kuò)展函數(shù)不能訪問私有或受保護(hù)的成員:擴(kuò)展函數(shù)可以在類的外部定義,但不能訪問類的私有或受保護(hù)成員。
 - 擴(kuò)展函數(shù)不能被重寫:由于擴(kuò)展函數(shù)是靜態(tài)解析的,因此不能被子類重寫。
 - 不能在擴(kuò)展函數(shù)中添加新的屬性:擴(kuò)展函數(shù)可以為現(xiàn)有類添加新的函數(shù),但不能添加新的屬性。
 - 不能在擴(kuò)展函數(shù)中訪問super關(guān)鍵字:擴(kuò)展函數(shù)中無(wú)法使用super關(guān)鍵字來(lái)調(diào)用基類的函數(shù)。
 - 作用域限制:擴(kuò)展函數(shù)的作用域是在導(dǎo)入它的包內(nèi),因此在其他包中無(wú)法直接使用。
 
Kotlin的擴(kuò)展特性使用限制主要是為了保證代碼的可靠性和可維護(hù)性。















 
 
 

















 
 
 
 