改不完的 Bug,写不完的矫情。公众号 杨正友 现在专注移动基础平台开发 ,涵盖音视频, APM和信息安全等各个知识领域;只做全网最 Geek 的公众号,欢迎您的关注!精彩内容不容错过~
Kotlin 拥有强大的生态,因为 Kotlin 是一门跑在 Java 虚拟机上的函数式语言,完全符合 JVM 的设计规范,如: 类型擦除,装箱和拆箱等等。所以也可以像 Java 一样支持 Android 原生环境开发,服务端,甚至大前端方向。那么 Kotlin 和 Java 有什么共同点呢?
kotlin 和 java8 的相同点我总结了三个,第一个是: 它们都是面向对象和函数式编程语言,第二个是: 它们都是强类型静态语言,最后一个就是它们都符合 JVM 设计规范
说完共同点我们再聊聊他们的不同点吧~
kotlin 和 java8 的不同挺多的,我简单的归纳了一下:
选中导航栏: 右键 -> Tools -> Kotlin -> Show Kotlin Bytecode -> Decompile
选中文件标签: 右键 -> Convert Java File To Kotlin File
查看 Kotlin 字节码
Kotlin 定义方式比较简单,括号中是参数,格式: "函数名(参数名称: 参数): 返回类型",返回类型定义在括号外 如:T
fun <T> reflectField(instance: Any?, name: String): T? {
if (instance == null) return null
try {
val field = instance.javaClass.getDeclaredField(name)
field.isAccessible = true
return field.get(instance) as T
} catch (e: Exception) {
e.printStackTrace()
}
return null
}
Kotlin 可以在 构造方法或者相关函数大括号里面设置默认参数,这样传参就可以使用默认参数了,我们具体看看怎么使用的吧~
第二个参数可以省略,此时会使用默认值
不省略时跟 Java 无异
named argument 的中文含义是 可以根据 参数名 进行传递参数,传参顺序不需要严格按照函数定义顺序,这样就可以避免参数误传的问题了
varible(变量) 通过 var 关键字来声明,格式为 var 变量名[:类型] = [初始值]
private var mConfig = RabbitConfig()
lateinit var application: Application
private var isInit = false
当然 kotlin 也有自动拆箱和装箱将 Int 翻译成 Java 的 int 或 Integer 以提高性能
private var Int = 8
Kotlin 会自动给属性加上 setter 和 getter 方法
class RabbitConfig(
var enable: Boolean = true,
var enableLog: Boolean = true,
@Transient var uiConfig: RabbitUiConfig = RabbitUiConfig(),
var storageConfig: RabbitStorageConfig = RabbitStorageConfig(),
var monitorConfig: RabbitMonitorConfig = RabbitMonitorConfig(),
var reportConfig: RabbitReportConfig = RabbitReportConfig()
)
用 Show Kotlin Bytecode 翻译一下结果如下:
如果想重写 set() 或 get() 方法 可以这么做
data class RabbitConfig(
var enable: Boolean = true,
var enableLog: Boolean = true,
@Transient var uiConfig: RabbitUiConfig = RabbitUiConfig(),
var storageConfig: RabbitStorageConfig = RabbitStorageConfig(),
var monitorConfig: RabbitMonitorConfig = RabbitMonitorConfig(),
var reportConfig: RabbitReportConfig = RabbitReportConfig()
) {
var age: Int = 10
set(value) {
println("setter $value")
field = value
}
val createTimeShow: () -> String
get() = { this.toString() }
}
以上声明的变量皆为非空变量,即不允许为空值。如果需要声明一个变量可为空,仅需在变量类型后面加上?
val n: String?=null
声明一个变量可为空,仅需在变量类型后面加上?
// 在使用n的时候,由于n可能为null,直接使用编译器会报错
n.length
// 正确的使用方式是,在使用前,做判空处理
// 如果变量为空,只需在变量类型后面加上?,这样就不会访问length属性了
n?.length
// 如果确定一个可空变量不为空
// 可以加!!告诉编译器它不为空
n!!.length
val i=10
val s="i=$i"//求值结果为:“i=10”
val s="adb"
val str="$s.length is ${s.length}"//执行结果为“adb.length is 3”
双冒号操作符 表示把一个方法(变量)当做一个参数,传递到另一个方法(变量)中进行使用,和 Java8 类似
/**
* 把功能的名字转换为对应存储数据对象
* */
fun nameToInfoClass(name: String): Class<*> {
return when (name) {
RabbitMonitorProtocol.BLOCK.name -> RabbitBlockFrameInfo::class.java
RabbitMonitorProtocol.APP_SPEED.name -> RabbitAppStartSpeedInfo::class.java
RabbitMonitorProtocol.FPS.name -> RabbitFPSInfo::class.java
RabbitMonitorProtocol.MEMORY.name -> RabbitMemoryInfo::class.java
RabbitMonitorProtocol.EXCEPTION.name -> RabbitExceptionInfo::class.java
RabbitMonitorProtocol.NET.name -> RabbitHttpLogInfo::class.java
RabbitMonitorProtocol.SLOW_METHOD.name -> RabbitSlowMethodInfo::class.java
RabbitMonitorProtocol.BLOCK_CALL.name -> RabbitBlockFrameInfo::class.java
RabbitMonitorProtocol.GLOBAL_MONITOR.name -> RabbitAppPerformanceInfo::class.java
RabbitMonitorProtocol.ANR.name -> RabbitAnrInfo::class.java
else -> RabbitFPSInfo::class.java
}
}
kotlin 正常函数
//---------------------案例一--------------------------
a(fun b(param: Int): String {
return param.toString()
});
val d = fun b(param: Int): String {
return param.toString()
}
//---------------------案例二--------------------------
view.setOnClickListener(new OnClickListener() {
@Override
void onClick(View v) {
switchToNextPage();
}
});
kotlin 匿名函数
//-----------------------案例一------------------------
a(fun(param: Int): String {
return param.toString()
});
val d = fun(param: Int): String {
return param.toString()
}
//-----------------------案例二------------------------
fun setOnClickListener(onClick: (View) -> Unit) {
this.onClick = onClick
}
view.setOnClickListener(fun(v: View): Unit) {
switchToNextPage()
})
view.setOnClickListener() { v: View ->
switchToNextPage()
}
//-----------------------------------------------
view.setOnClickListener { v: View ->
switchToNextPage()
}
//-----------------------------------------------
mSimpleListRv.setOnClickListener {
startActivity(Intent(this, SimpleListActivity::class.java))
}
//-----------------------------------------------
view.setOnClickListener {
switchToNextPage()
it.setVisibility(GONE)
}
// -----------------------正序遍历-----------------------
for (index in 1..100){
print(index)
}
// -----------------------倒序遍历-----------------------
for (index in 100 downTo 1){
print(index)
}
// -----------------------不使用1作为遍历的步长-----------------------
for (index in 1..100 step 2){
print(index)//会输出1..3..5......
}
// -----------------------创建一个不包含末尾元素的区间-----------------------
for (index in 1 until 10){
println(index)//输出0..9
}
// -----------------------遍历一个数组/列表,想同时取出下标和元素-----------------------
val array = arrayOf("a", "b", "c")
for ((index,e) in array.withIndex()){
println("下标=$index----元素=$e")
}
// -----------------------遍历一个数组/列表,只取出下标-----------------------
val array = arrayOf("a", "b", "c")
for (index in array.indices){
println("index=$index")//输出0,1,2
}
// -----------------------遍历取元素-----------------------
val array = arrayOf("a", "b", "c")
for (element in array){
println("element=$element")//输出a,b,c
}
Kotlin 中,枚举类型以类的形式存在,因此叫做枚举类,详细代码如下:
enum class Color {
RED, GREEN, BLUE
}
//---------------使用-------------------
var color: Color = Color.BLUE
var color2 = Color.GREEN
// 比较两个枚举类型变量
var bool: Boolean = color == color2
//---------------为枚举值指定数值-------------------
enum class Direction private constructor(var value: Int) {
NORTH(1), WEST(2), EAST(3), SOUTH(4)
}
when 表达式 可以支持自动转型(Auto-casting),详细代码如下:
when (view) {
is TextView -> toast(view.text)
is RecyclerView -> toast("Item count = ${view.adapter.itemCount}")
is SearchView -> toast("Current query: ${view.query}")
else -> toast("View type not supported")
}
kotlin 中缀表达式只有一个参数,且用 infix 修饰的函数,这就是我们所说的自定义运算符的中缀表达式,详细代码如下:
// 书
class Book{
// 传入任意类型,返回一个Boolean类型的参数
infix fun on(any: Any): Boolean{
return true
}
}
// 桌子
class Desk
fun main(args: Array<String>) {
if(Book() on Desk()){
println("书在桌上")
}
}
?.表示,如果 foo 这时候是一个空的,则返回 null,否则就返回 foo.bar(),详细代码如下:
if (foo != null){
return foo.bar()
}else{
return null
}
?: 作用是当数据非空时,直接返回数据,而当数据为空时,返回合并到的数据。利用该运算符,可以很容易的把可空类型转换为非空类型,详细代码如下:
if(foo!=null)
{
foo
}
else
{
bar
}
foo?.length?:-1
Kotlin 中 !! 表示非空断言运算符,详细代码如下:
if(foo!=null)
{
foo
}else
{
return NullPointerException
}
Kotlin 可以使用安全转换操作符 as?,它可以在失败时返回 null,详细代码如下:
foo as? type
if(foo is Stype)
{
foo as Type
}else
{
null
}
?表示可空类型与非空类型,Kotlin 的类型系统旨在消除来自代码空引用的危险,详细代码如下:
foo?
var foo?="abc"
foo=null
// 编译成功 foo可为空
var foo="abc"
foo=null
// 编译失败 foo不可以为空
kotlin 中继承全部使用“:”,不区分 extend 或 implement,但是有几个规则需要注意一下:
object 全局声明的对象只有一个,所以他是天生的单例模型
在 Kotlin 里面声明一个 data 类需要满足以下条件:
嵌套类的用法: 如果要在 Kotlin 中嵌套一个类,需要在该内部类加 inner 修饰
//嵌套类属于静态类和外部类没任何关系
fun main(args : Array<String>){
var ot = OutClass().innerClass()
ot.hello()
}
class OutClass{
var name ="李武"
//inner表示内部类
inner class innerClass{
var name = "张三"
fun hello(){
println("你好 $name")
//内部类使用this,访问外部类的变量
println("你好 ${this@OutClass.name}")
}
}
}
本文先从 kotlin 概念带大家进入 kotlin 语言,然后说了一下 kotlin 和 java8 的异同以及相互转换方式,最后对 Kotlin 基础语法进行一一阐述,如: kotlin 方法和变量使用,kotlin 字符串模板, lambada 表达式,以及各种表达式的实际运用。好了今天的 kotlin 就说到这里了,如果大家对此有什么疑问欢迎到评论区留言~
各位看官,看都看了,帮忙点个赞呗~
原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。
如有侵权,请联系 cloudcommunity@tencent.com 删除。
原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。
如有侵权,请联系 cloudcommunity@tencent.com 删除。