国产片侵犯亲女视频播放_亚洲精品二区_在线免费国产视频_欧美精品一区二区三区在线_少妇久久久_在线观看av不卡

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - Android - Kotlin中的一些技巧與迂回操作分享

Kotlin中的一些技巧與迂回操作分享

2022-03-08 16:02Dexlind Android

這篇文章主要給大家介紹了關于Kotlin中的一些技巧與迂回操作的相關資料,文中通過示例代碼介紹的非常詳細,對大家學習或者使用kotlin具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧

Kotlin概述

科特林島(Котлин)是一座俄羅斯的島嶼,位于圣彼得堡以西約30公里處,形狀狹長,東西長度約14公里,南北寬度約2公里,面積有16平方公里,扼守俄國進入芬蘭灣的水道。科特林島上建有喀瑯施塔得市,為圣彼得堡下轄的城市。

我們這里講的Kotlin,就是一門以這個Котлин島命名的現代程序設計語言。它是一門靜態類型編程語言,支持JVM平臺,Android平臺,瀏覽器JS運行環境,本地機器碼等。支持與Java,Android 100% 完全互操作。

Kotlin中的一些技巧與迂回操作分享

本文將給大家詳細關于Kotlin技巧與迂回操作的一些內容,下面話不多說了,來一起看看詳細的介紹吧

不需要 import 就能使用的頂層函數

一個頂層函數,除非你在同一個文件里使用,否則就需要 import 或者使用完全限定名。問題是有些人就是嫌煩,想要所謂的“全局函數”,就像 Kotlin 標準庫里的 println 一樣。其實很簡單,只需要寫得跟 println 一樣就行了:

?
1
2
3
package kotlin
 
fun fuck() {}

因為 kotlin 包下的東西都是自動導入的,也就不需要自己動手導入啦。

需要傳入編譯器參數 -Xallow-kotlin-package 來允許使用 kotlin 開頭的包名。

遞歸的 Lambda 表達式

剛才在某個 Kotlin 裙里看到有人在問:

是不是lambda無法遞歸

舉個例子,我們可以寫一個簡單的遞歸函數:

?
1
2
fun a() { println("1551"); a() }
a() // 打印出很多1551

如果要寫成 Lambda 呢?這樣的代碼會報錯:

?
1
val a: () -> Unit = { println("1551"); a() }

我們自然是不能直接寫這樣的代碼的,它會說 a 沒有定義。解決方法當然是使用 lateinit:

?
1
2
3
lateinit var a: () -> Unit
a = { println("1551"); a() }
a() // 打印出很多1551

更進一步:匿名 Lambda 表達式的遞歸

正統的「Lambda演算」里面的函數全部都是匿名函數,需要使用「不動點組合子」實現遞歸:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
// 這是kotlin-js
val z = { f: dynamic ->
 { g: dynamic -> g(g) }
 { x: dynamic -> f { y: dynamic -> x(x)(y) } }
}
val a = z { f: () -> Unit ->
 {
 println("1551"); f()
 }
}
// 求斐波那契數列第n項的函數
val fib: (Int) -> Int = z { f: (Int) -> Int ->
 { x: Int ->
 if (x <= 2) 1
 else f(x - 1) + f(x - 2)
 }
}
// 輸出斐波那契數列前10項
println((1.rangeTo(10).map(fib)))

上面的那一坨 val z 即是「Z組合子」。(讀者可以思考一下為什么這里我給了 Kotlin-js 的例子是而不是 Kotlin-jvm(逃

阻止編譯器添加對非空類型的函數參數的 NullCheck

總所周知,當一個函數的參數是非空類型時,Kotlin編譯器會在方法入口處加一行檢查入參是否為空的代碼。比如說 main 函數:

?
1
fun main(args: Array<String>) {}

經過編譯后,再反編譯成Java:

?
1
2
3
public static final void main(@NotNull String[] args) {
 Intrinsics.checkParameterIsNotNull(args, "args");
}

可惡!辣雞編譯器自作主張!我不想要這行代碼!

如果不想編譯器生成這些代碼,把這幾個編譯器參數 -Xno-call-assertions、-Xno-param-assertions、-Xno-receiver-assertions 傳給Kotlin編譯器即可。

傳遞編譯器參數的方法:

使用IDEA調用編譯器的情況:

Project 設置:File -> Settings -> 找到 Kotlin Compiler -> Additional command line parameters

Module 設置:File -> Project Structure -> Module -> 找到你的Module里面的Kotlin設置 -> Additional command line parameters

使用Gradle Kotlin DSL的情況:

?
1
2
3
4
5
// build.gradle.kts
tasks.withType<KotlinCompile> {
 // 加上下面這行
 kotlinOptions.freeCompilerArgs = listOf("-Xno-call-assertions", "-Xno-param-assertions", "-Xno-receiver-assertions")
}

PS:注意IDEA的 Delegate IDE build/run actions to gradle 這個選項是否勾選的區別。

給data class自定義getter和setter

?
1
data class SomeClass(var name: String)

眾所周知 Kotlin 不允許給聲明在主構造器里面的屬性寫自定義getter、setter,主要是為了防止有好事者亂寫,破壞規則就不好了。所以迂回操作如下:

?
1
2
3
4
5
data class SomeClass(private var _name: String) {
 var name: String
 get() = _name
 set(value) { _name = value }
}

解釋:private的_name不會生成getter和setter,你再把你想寫的getter和setter添上就好。這樣SomeClass里面就有3樣東西:String _name,String getName()和void setName(String)(以及data class根據_name自動生成的那些)。

缺點很明顯,toString 生成的字符串會比較丑。

流的讀取

普通青年:

?
1
2
3
4
5
6
7
// java 代碼
void someFunc(InputStream in, OutputStream out) throws IOException {
 int read;
 while ((read = in.read()) != -1) {
 out.write(read);
 }
}

文藝青年:

?
1
2
3
4
5
6
fun someFunc(`in`: InputStream, out: OutputStream) {
 var read: Int = -1
 while ({ read = `in`.read();read }() != -1) {
 out.write(read)
 }
}

二逼青年:

?
1
2
3
4
5
6
7
fun someFunc(`in`: InputStream, out: OutputStream) {
 var read: Int = `in`.read()
 while (read != -1) {
 out.write(read)
 read = `in`.read()
 }
}

群里的優秀的青年(不是我):

?
1
2
3
4
5
6
fun someFunc(`in`: InputStream, out: OutputStream) {
 var read: Int = -1
 while (`in`.read().also { read = it } != -1) {
 out.write(read)
 }
}

限制擴展的作用域(防止污染命名空間)

注意:此技巧并不穩定,可能在未來被官方干掉。

?
1
2
3
4
// 把擴展丟進一個object里面
object StringExtension {
 @JvmStatic fun String.fuck() = println("fuck $this")
}
?
1
2
3
4
5
6
7
8
9
10
// 使用說明
fun test() {
 // 下面這行被注釋掉的代碼不能通過編譯
 // "kotlin".fuck()
 
 // 你要這么用,將MyExtentions塞進上下文(即this)
 with(StringExtention) {
 "kotlin".fuck()
 }
}
?
1
2
3
4
5
6
// 或者手動引入
import StringExtension.fuck
 
fun test() {
 "kotlin".fuck()
}
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
// 以下是夏姬八寫,別模仿
interface Extension
 
inline fun <T : Extension, R> T.use(block: T.() -> R) = this.block()
 
object StringExtension : Extension {
 @JvmStatic fun String.fuck() = println("fuck $this")
}
 
object IntExtension : Extension {
 @JvmStatic fun Int.love() = println("I love $this")
}
 
fun test() {
 StringExtension.use { "kotlin".fuck() }
 
 IntExtension.use { 1551.love() }
}

鏈式調用時輸出中間值

?
1
2
3
4
5
6
7
8
9
10
11
12
13
inline fun <T> T.println(): T = printlnBy { it }
 
inline fun <T, U> T.printlnBy(selector: (T) -> U): T = this.also { println(selector(it)) }
 
fun test() {
 listOf(1, 2, 3).asSequence()
  .map { it * 3 }.printlnBy { it.sum() } // <==這里
  .filter { it and 1 == 0 }
  .sum().println() // <==還有這里
}
// 輸出:
// 18
// 6

注意副作用,別夏姬八用!

如果是集合操作,可以考慮使用 onEach 這個高階函數,例如onEach { println(it) }。

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對服務器之家的支持。

原文鏈接:https://aisia.moe/2018/01/07/kotlin-jiqiao/

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 午夜在线视频 | 精品动漫一区 | 国产一区二区av | 青青草免费在线视频 | 黄色网免费看 | 一级爱 | 一本色道久久综合狠狠躁的推荐 | 国产99久久 | 欧美日韩中文在线观看 | av电影一区二区 | 日韩在线中文字幕 | 亚洲国产精品成人女人久久久 | 亚洲精品1区2区 | 丝袜+亚洲+另类+欧美+变态 | 免费久久99精品国产婷婷六月 | 成年人视频在线观看免费 | 亚洲精品区 | 国产成人一区二区啪在线观看 | 国产日产精品一区二区三区四区 | av不卡在线播放 | 夜夜av | 中文字幕在线免费观看 | 日韩欧美在线不卡 | 色偷偷偷 | 国产综合视频在线观看 | 成人免费毛片高清视频 | 国产精品99久久免费观看 | 亚洲午夜视频在线 | 国产一区二区免费视频 | 蜜桃臀一区二区三区 | 亚洲一区视频 | 欧洲亚洲精品久久久久 | 日韩免费视频 | 艹逼网 | 亚洲精品久久久久久国产精华液 | 亚洲一区观看 | 一级性视频| 欧美日韩久久久 | 久久精品国产99国产精品 | 精品久久久久久久久久久下田 | 国产精品国色综合久久 |