热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

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

这篇文章主要给大家介绍了关于Kotlin中的一些技巧与迂回操作的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用kotlin具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

Kotlin概述

科特林岛(Котлин)是一座俄罗斯的岛屿,位于圣彼得堡以西约30公里处,形状狭长,东西长度约14公里,南北宽度约2公里,面积有16平方公里,扼守俄国进入芬兰湾的水道。科特林岛上建有喀琅施塔得市,为圣彼得堡下辖的城市。

我们这里讲的Kotlin,就是一门以这个Котлин岛命名的现代程序设计语言。它是一门静态类型编程语言,支持JVM平台,Android平台,浏览器JS运行环境,本地机器码等。支持与Java,Android 100% 完全互操作。

本文将给大家详细关于Kotlin技巧与迂回操作的一些内容,下面话不多说了,来一起看看详细的介绍吧

不需要 import 就能使用的顶层函数

一个顶层函数,除非你在同一个文件里使用,否则就需要 import 或者使用完全限定名。问题是有些人就是嫌烦,想要所谓的“全局函数”,就像 Kotlin 标准库里的 println 一样。其实很简单,只需要写得跟 println 一样就行了:

package kotlin

fun fuck() {}

因为 kotlin 包下的东西都是自动导入的,也就不需要自己动手导入啦。

需要传入编译器参数 -Xallow-kotlin-package 来允许使用 kotlin 开头的包名。

递归的 Lambda 表达式

刚才在某个 Kotlin 裙里看到有人在问:

是不是lambda无法递归

举个例子,我们可以写一个简单的递归函数:

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

如果要写成 Lambda 呢?这样的代码会报错:

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

我们自然是不能直接写这样的代码的,它会说 a 没有定义。解决方法当然是使用 lateinit:

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

更进一步:匿名 Lambda 表达式的递归

正统的「Lambda演算」里面的函数全部都是匿名函数,需要使用「不动点组合子」实现递归:

// 这是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 函数:

fun main(args: Array) {}

经过编译后,再反编译成Java:

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的情况:

// build.gradle.kts
tasks.withType {
 // 加上下面这行
 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

data class SomeClass(var name: String)

众所周知 Kotlin 不允许给声明在主构造器里面的属性写自定义getter、setter,主要是为了防止有好事者乱写,破坏规则就不好了。所以迂回操作如下:

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 生成的字符串会比较丑。

流的读取

普通青年:

// java 代码
void someFunc(InputStream in, OutputStream out) throws IOException {
 int read;
 while ((read = in.read()) != -1) {
 out.write(read);
 }
}

文艺青年:

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

二逼青年:

fun someFunc(`in`: InputStream, out: OutputStream) {
 var read: Int = `in`.read()
 while (read != -1) {
 out.write(read)
 read = `in`.read()
 }
}

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

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

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

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

// 把扩展丢进一个object里面
object StringExtension {
 @JvmStatic fun String.fuck() = println("fuck $this")
}
// 使用说明
fun test() {
 // 下面这行被注释掉的代码不能通过编译
 // "kotlin".fuck()

 // 你要这么用,将MyExtentions塞进上下文(即this)
 with(StringExtention) {
 "kotlin".fuck()
 }
}
// 或者手动引入
import StringExtension.fuck

fun test() {
 "kotlin".fuck()
}
// 以下是夏姬八写,别模仿
interface Extension

inline fun  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() }
}

链式调用时输出中间值

inline fun  T.println(): T = printlnBy { it }

inline fun  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) }。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对的支持。


推荐阅读
  • HDU 2372 El Dorado(DP)的最长上升子序列长度求解方法
    本文介绍了解决HDU 2372 El Dorado问题的一种动态规划方法,通过循环k的方式求解最长上升子序列的长度。具体实现过程包括初始化dp数组、读取数列、计算最长上升子序列长度等步骤。 ... [详细]
  • Android中高级面试必知必会,积累总结
    本文介绍了Android中高级面试的必知必会内容,并总结了相关经验。文章指出,如今的Android市场对开发人员的要求更高,需要更专业的人才。同时,文章还给出了针对Android岗位的职责和要求,并提供了简历突出的建议。 ... [详细]
  • 在说Hibernate映射前,我们先来了解下对象关系映射ORM。ORM的实现思想就是将关系数据库中表的数据映射成对象,以对象的形式展现。这样开发人员就可以把对数据库的操作转化为对 ... [详细]
  • baresip android编译、运行教程1语音通话
    本文介绍了如何在安卓平台上编译和运行baresip android,包括下载相关的sdk和ndk,修改ndk路径和输出目录,以及创建一个c++的安卓工程并将目录考到cpp下。详细步骤可参考给出的链接和文档。 ... [详细]
  • 【Windows】实现微信双开或多开的方法及步骤详解
    本文介绍了在Windows系统下实现微信双开或多开的方法,通过安装微信电脑版、复制微信程序启动路径、修改文本文件为bat文件等步骤,实现同时登录两个或多个微信的效果。相比于使用虚拟机的方法,本方法更简单易行,适用于任何电脑,并且不会消耗过多系统资源。详细步骤和原理解释请参考本文内容。 ... [详细]
  • Android Studio Bumblebee | 2021.1.1(大黄蜂版本使用介绍)
    本文介绍了Android Studio Bumblebee | 2021.1.1(大黄蜂版本)的使用方法和相关知识,包括Gradle的介绍、设备管理器的配置、无线调试、新版本问题等内容。同时还提供了更新版本的下载地址和启动页面截图。 ... [详细]
  • 本文介绍了在SpringBoot中集成thymeleaf前端模版的配置步骤,包括在application.properties配置文件中添加thymeleaf的配置信息,引入thymeleaf的jar包,以及创建PageController并添加index方法。 ... [详细]
  • 知识图谱——机器大脑中的知识库
    本文介绍了知识图谱在机器大脑中的应用,以及搜索引擎在知识图谱方面的发展。以谷歌知识图谱为例,说明了知识图谱的智能化特点。通过搜索引擎用户可以获取更加智能化的答案,如搜索关键词"Marie Curie",会得到居里夫人的详细信息以及与之相关的历史人物。知识图谱的出现引起了搜索引擎行业的变革,不仅美国的微软必应,中国的百度、搜狗等搜索引擎公司也纷纷推出了自己的知识图谱。 ... [详细]
  • 本文讲述了作者通过点火测试男友的性格和承受能力,以考验婚姻问题。作者故意不安慰男友并再次点火,观察他的反应。这个行为是善意的玩人,旨在了解男友的性格和避免婚姻问题。 ... [详细]
  • 安卓select模态框样式改变_微软Office风格的多端(Web、安卓、iOS)组件库——Fabric UI...
    介绍FabricUI是微软开源的一套Office风格的多端组件库,共有三套针对性的组件,分别适用于web、android以及iOS,Fab ... [详细]
  • 本文详细介绍了Linux中进程控制块PCBtask_struct结构体的结构和作用,包括进程状态、进程号、待处理信号、进程地址空间、调度标志、锁深度、基本时间片、调度策略以及内存管理信息等方面的内容。阅读本文可以更加深入地了解Linux进程管理的原理和机制。 ... [详细]
  • 1,关于死锁的理解死锁,我们可以简单的理解为是两个线程同时使用同一资源,两个线程又得不到相应的资源而造成永无相互等待的情况。 2,模拟死锁背景介绍:我们创建一个朋友 ... [详细]
  • 后台获取视图对应的字符串
    1.帮助类后台获取视图对应的字符串publicclassViewHelper{将View输出为字符串(注:不会执行对应的ac ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • Google Play推出全新的应用内评价API,帮助开发者获取更多优质用户反馈。用户每天在Google Play上发表数百万条评论,这有助于开发者了解用户喜好和改进需求。开发者可以选择在适当的时间请求用户撰写评论,以获得全面而有用的反馈。全新应用内评价功能让用户无需返回应用详情页面即可发表评论,提升用户体验。 ... [详细]
author-avatar
好人cuiyin_550
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有