注解声明

  注解是将元数据附加到代码的方法。要声明注解,请将 annotation 修饰符放在类的前面

annotation class Fancy

  注解的附加属性可以通过用元注解标注注解类来指定

    — @Target指定可以用该注解标注的元素的可能的类型(类、函数、属性、表达式等);

    — @Retention指定该注解是否存储在编译后的class文件中,以及它在运行时能否通过反射可⻅(默认都是 true);

    — @Repeatable允许在单个元素上多次使用相同的该注解;

    — @MustBeDocumented指定该注解是公有API的一部分,并且应该包含在生成的API文档中显示的类或方法的签名中。

@Target(AnnotationTarget.CLASS, AnnotationTarget.FUNCTION,   
AnnotationTarget.VALUE_PARAMETER, AnnotationTarget.EXPRESSION)
@Retention(AnnotationRetention.SOURCE)
@MustBeDocumented
annotation class Fancy

  

用法

@Fancy class Foo {
@Fancy fun baz(@Fancy foo: Int): Int {
return (@Fancy 1)
}
}

  如果需要对类的主构造函数进行标注,则需要在构造函数声明中添加 constructor 关键字 ,并将注解添加到 其前面

class Foo @Inject constructor(dependency: MyDependency) { ...... }

  你也可以标注属性访问器

class Foo {
var x: MyDependency? = null
@Inject set
}

  

构造函数

  注解可以有接受参数的构造函数

annotation class Special(val why: String) 

@Special("example") class Foo {}

  允许的参数类型有

    — 对应于Java原生类型的类型(Int、Long等); — 字符串;

    — 类(Foo::class);

    — 枚举;

    — 其他注解;

    — 上面已列类型的数组。

  注解参数不能有可空类型,因为 JVM 不支持将 null 作为注解属性的值存储。

  如果注解用作另一个注解的参数,则其名称不以 @ 字符为前缀

annotation class ReplaceWith(val expression: String)

annotation class Deprecated( 
    val message: String,
    val replaceWith: ReplaceWith = ReplaceWith("")
) @Deprecated("This function is deprecated, use === instead", ReplaceWith("this === other"))

  如果需要将一个类指定为注解的参数,请使用 Kotlin 类(KClass)。Kotlin 编译器会自动将其转换为 Java 类,以 便 Java 代码能够正常访问该注解与参数

import kotlin.reflect.KClass

annotation class Ann(val arg1: KClass<*>, val arg2: KClass<out Any>) 

@Ann(String::class, Int::class) class MyClass

  

Lambda 表达式

  注解也可以用于 lambda 表达式。它们会被应用于生成 lambda 表达式体的 invoke() 方法上。这对于像Quasar 这样的框架很有用,该框架使用注解进行并发控制

annotation class Suspendable

val f = @Suspendable { Fiber.sleep(10) }

  

注解使用处目标

  当对属性或主构造函数参数进行标注时,从相应的 Kotlin 元素生成的 Java 元素会有多个,因此在生成的 Java 字节码中该注解有多个可能位置 。如果要指定精确地指定应该如何生成该注解,请使用以下语法

class Example(
  @field:Ann val foo,// 标注 Java 字段
  @get:Ann val bar,// 标注 Java getter
  @param:Ann val quux// 标注 Java 构造函数参数
)

  可以使用相同的语法来标注整个文件。要做到这一点,把带有目标 file 的注解放在文件的顶层、package 指 令之前或者在所有导入之前(如果文件在默认包中的话)

@file:JvmName("Foo") 

package org.jetbrains.demo

  如果你对同一目标有多个注解,那么可以这样来避免目标重复——在目标后面添加方括号并将所有注解放在方 括号内

class Example {
@set:[Inject VisibleForTesting]
var collaborator: Collaborator
}

  支持的使用处目标的完整列表为

    —  file;

    —  property(具有此目标的注解对 Java 不可⻅);

    —  field ;

    —  get(属性 getter);

    —  set(属性 setter);

    —  receiver(扩展函数或属性的接收者参数);

    —  param(构造函数参数);

    —  setparam(属性 setter 参数);

    —  delegate(为委托属性存储其委托实例的字段)

  

  要标注扩展函数的接收者参数,请使用以下语法

fun @receiver:Fancy String.myExtension() { ... }

  如果不指定使用处目标,则根据正在使用的注解的 @Target 注解来选择目标 。如果有多个适用的目标,则使用以下列表中的第一个适用目标

    —  param ;

    —  property ;

    —  field

Java注解

  Java 注解与 Kotlin 100% 兼容

import org.junit.Test
import org.junit.Assert.*
import org.junit.Rule
import org.junit.rules.*
class Tests {
// 将 @Rule 注解应用于属性 getter
@get:Rule val tempFolder = TemporaryFolder()
@Test fun simple() {
val f = tempFolder.newFile()
assertEquals(42, getTheAnswer())
}
}

  因为 Java 编写的注解没有定义参数顺序,所以不能使用常规函数调用语法来传递参数。相反,你需要使用具名参 数语法

// Java
public @interface Ann {
int intValue();
String stringValue();
}

  

 // Kotlin
@Ann(intValue = 1, stringValue = "abc") class C

  就像在 Java 中一样,一个特殊的情况是 value 参数;它的值无需显式名称指定

// Java
public @interface AnnWithValue {
String value();
}

  

 // Kotlin
@AnnWithValue("abc") class C

  

数组作为注解参数

  如果 Java 中的 value 参数具有数组类型,它会成为 Kotlin 中的一个 vararg 参数

// Java
public @interface AnnWithArrayValue {
String[] value();
}

  

 // Kotlin
@AnnWithArrayValue("abc", "foo", "bar") class C

  对于具有数组类型的其他参数,你需要显式使用数组字面值语法(自 Kotlin 1.2 起)或者 arrayOf(......)

// Java
public @interface AnnWithArrayMethod {
String[] names();
}

  

/ Kotlin 1.2+:
@AnnWithArrayMethod(names = ["abc", "foo", "bar"])
class C // 旧版本 Kotlin:
@AnnWithArrayMethod(names = arrayOf("abc", "foo", "bar"))
class D

  

访问注解实例的属性

  注解实例的值会作为属性暴露给 Kotlin 代码

// Java
public @interface Ann {
int value();
}

  

// Kotlin
fun foo(ann: Ann) {
val i = ann.value
}

  

kotlin更多语言结构——>注解的更多相关文章

  1. Java 语言结构【转】

    Java 语言结构 基础:包(Package).类(Class)和对象(Object) 了解 Java 的包(Package).类(Class)和对象(Object)这些基础术语是非常重要的,这部分内 ...

  2. KOTLIN开发语言文档(官方文档) -- 2.基本概念

    网页链接:https://kotlinlang.org/docs/reference/basic-types.html 2.   基本概念 2.1.  基本类型 从可以在任何变量处理调用成员函数和属性 ...

  3. 06. Go 语言结构体

    Go语言结构体(struct) Go 语言通过用自定义的方式形成新的类型,结构体是类型中带有成员的复合类型.Go 语言使用结构体和结构体成员来描述真实世界的实体和实体对应的各种属性. Go 语言中的类 ...

  4. KOTLIN开发语言文档(官方文档) -- 入门

    网页链接:https://kotlinlang.org/docs/reference/basic-syntax.html 1.   入门 1.1.  基本语法 1.1.1.   定义包 包说明应该在源 ...

  5. 漫谈C语言结构体struct、公用体union空间占用

    先用代码说话: #include<stdio.h> union union_data0{ int a ;//本身占用4个字节 char b ;//本身占用1个字节 int c ; }; u ...

  6. 解析C语言结构体对齐(内存对齐问题)

    C语言结构体对齐也是老生常谈的话题了.基本上是面试题的必考题.内容虽然很基础,但一不小心就会弄错.写出一个struct,然后sizeof,你会不会经常对结果感到奇怪?sizeof的结果往往都比你声明的 ...

  7. 不可或缺 Windows Native (8) - C 语言: 结构体,共用体,枚举,类型定义符

    [源码下载] 不可或缺 Windows Native (8) - C 语言: 结构体,共用体,枚举,类型定义符 作者:webabcd 介绍不可或缺 Windows Native 之 C 语言 结构体 ...

  8. (转)PHP的语言结构和函数的区别

    相信大家经常看到对比一些PHP应用中,说用isset() 替换 strlen(),isset比strlen执行速度快等. 例子: if ( isset($user) ) { //do some thi ...

  9. php入门 数据类型 运算符 语言结构语句 函数 类与面向对象

    php PHP-enabled web pages are treated just like regular HTML pages and you can create and edit them ...

  10. Go语言结构体(struct)

    Go 语言结构体 Go 语言中数组可以存储同一类型的数据,但在结构体中我们可以为不同项定义不同的数据类型. 结构体是由一系列具有相同类型或不同类型的数据构成的数据集合. 结构体表示一项记录,比如保存图 ...

随机推荐

  1. 使用AI模型替代工业仿真过程

    引自: https://www.zhihu.com/question/641951284/answer/3384531468 使用AI模型替代工业仿真,如:CAE,等等,进行仿真环境的求解运算.

  2. 查看并添加python中库的搜索路径

    根据前文  pip install --user 使用方法和注意事项--python中安装module库到用户packages路径中 我们知道python中除了自身的全局module lib路径以外还 ...

  3. 【转载】AI的剥削:肯尼亚工人训练ChatGPT,看大量有害内容心理受伤——AI新时代下剥削

    原文地址: https://mbd.baidu.com/newspage/data/landingsuper?context=%7B%22nid%22%3A%22news_90623597686514 ...

  4. 使用 Apache DolphinScheduler 进行 EMR 任务调度

    By AWS Team 前言 随着企业规模的扩大,业务数据的激增,我们会使用 Hadoop/Spark 框架来处理大量数据的 ETL/聚合分析作业,⽽这些作业将需要由统一的作业调度平台去定时调度. 在 ...

  5. Linux内核如何判断地址是否位于用户空间?

    一. 问题描述 access_ok函数是什么原理? 二.问题分析 我们在内核空间和用户空间进行数据拷贝的时候必须判断用户空间地址是否合法. 主要通过偶函数access_ok来判断. 1. Linux用 ...

  6. Vite+Vue3 项目 华仔待办

    此"华仔",不是彼"华仔",你懂的! 先来了个截图 紧跟着,实现步骤也来了 1. 安装 Node.js,终端运行 npm create vue@latest,项 ...

  7. SRE 必备知识 - Kafka 探秘之零拷贝技术

    如果你了解过 Kafka,那么它用到的一个性能优化技术可能会引起你的注意 -- 操作系统的零拷贝(zero-copy)优化. 零拷贝操作可以避免对数据的非必要拷贝,当然,并非是说完全没有拷贝. 在 K ...

  8. 【YashanDB知识库】应用绑定参数的慢查询,慢日志抓取不到

    [问题分类]功能使用 [关键字]慢日志.绑定参数 [问题描述]应用下发的绑定参数的SQL,慢日志功能抓取不到查询耗时超过阈值的执行SQL [问题原因分析]慢日志的功能没有考虑绑定参数的情况 [解决/规 ...

  9. 一文轻松搞定 tarjan 算法(二)(附带 tarjan 题单)

    完结篇:tarjan 求割点.点双连通分量.割边(桥)(附 40 道很好的 tarjan 题目). 上一篇(tarjan 求强连通分量,缩点,求边双) tarjan 求割点 还是求强联通分量的大致思路 ...

  10. Angular 学习笔记 language service

    尝试 v10 rc 的时候, 突然 language service 不 work 了. ctrl + shift + p -> Show logs... 这样可以检查和 report issu ...