kotlin中主构造函数是什么

Kotlin 中的主构造函数

主构造函数(Primary Constructor)是 Kotlin 类声明的一部分,用于在 创建对象时初始化类的属性。它不像 Java 那样是一个函数体,而是紧跟在类名后面。

主构造函数的基本定义
class Person(val name: String, val age: Int)

上面这段代码中:

  • nameage类的属性,它们直接在主构造函数中声明并初始化。
  • val 关键字表示这些属性是 只读的(不可变),如果用 var,则是可变的。

等价于 Java 代码:

class Person {private final String name;private final int age;public Person(String name, int age) {this.name = name;this.age = age;}
}

如何初始化主构造函数中的类属性?

在主构造函数中,类的属性可以通过以下几种方式初始化:

1. 直接在主构造函数中定义并赋值
class Person(val name: String, val age: Int)

这样 nameage 直接成为 Person 类的属性,无需额外赋值。


2. 使用 init 代码块

如果初始化逻辑较复杂,可以在 init 代码块中进行操作:

class Person(val name: String, val age: Int) {init {println("Person created: Name = $name, Age = $age")}
}

init 代码块会在 对象创建时 立即执行,并且按代码顺序先于次构造函数执行


3. 主构造函数 + 默认参数

Kotlin 允许给主构造函数的参数设置默认值:

class Person(val name: String = "Unknown", val age: Int = 0)

这样在实例化对象时,可以省略某些参数:

val p1 = Person()          // name = "Unknown", age = 0
val p2 = Person("Alice")   // name = "Alice", age = 0
val p3 = Person("Bob", 25) // name = "Bob", age = 25

主构造函数与次构造函数

如果需要提供额外的构造方式,可以添加 次构造函数(Secondary Constructor):

class Person(val name: String, val age: Int) {constructor(name: String) : this(name, 0) {println("Secondary constructor called")}
}

执行顺序:

  1. 先执行 主构造函数(包括 init 代码块)
  2. 再执行 次构造函数 的额外逻辑

示例:

val p = Person("John")

输出:

Person created: Name = John, Age = 0
Secondary constructor called

总结

  1. 主构造函数 是类定义的一部分,直接在类名后面声明:
    class Person(val name: String, val age: Int)
    
  2. 属性的初始化 通常有三种方式:
    • 直接赋值(最常见)
    • 使用 init 代码块
    • 默认参数
  3. 次构造函数 只能调用主构造函数,并在其基础上提供额外的初始化逻辑。
  4. 优先使用主构造函数,次构造函数只在特殊情况下使用(如不同初始化需求)。

这样可以使代码更简洁、可读性更强,同时遵循 Kotlin 的惯用法(idiomatic Kotlin)。

二 主构造函数是如何直接赋值的?

在 Kotlin 中,主构造函数可以直接将参数赋值给类的属性,而无需显式地编写额外的代码。这是通过在主构造函数中声明属性并使用 valvar 关键字实现的。

主构造函数直接赋值的机制

  1. 定义属性

    • 在主构造函数中,使用 valvar 声明参数时,Kotlin 会自动将这些参数转换为类的属性。
    • val 表示只读属性(生成 getter 方法)。
    • var 表示可变属性(生成 getter 和 setter 方法)。
  2. 自动赋值

    • 当创建类的实例时,传递给主构造函数的参数会自动赋值给对应的属性
    • 不需要手动编写赋值代码。
  3. 简化代码

    • 这种方式可以显著减少样板代码,使代码更加简洁。

示例代码

示例 1:基本用法
class Person(val name: String, var age: Int)fun main() {val person = Person("Alice", 25)println(person.name) // 输出:Aliceprintln(person.age)  // 输出:25person.age = 26      // 可以修改 var 属性println(person.age)  // 输出:26
}
  • name 是一个只读属性(val),只能读取。
  • age 是一个可变属性(var),可以读取和修改。

示例 2:带默认值的主构造函数
class Person(val name: String = "Unknown", var age: Int = 0)fun main() {val person1 = Person() // 使用默认值println(person1.name) // 输出:Unknownprintln(person1.age)  // 输出:0val person2 = Person("Bob", 30) // 指定值println(person2.name) // 输出:Bobprintln(person2.age)  // 输出:30
}
  • 如果主构造函数的参数有默认值,则在创建对象时可以省略这些参数。

示例 3:主构造函数与初始化块结合
class Person(val name: String, var age: Int) {init {println("Person created with name: $name and age: $age")}
}fun main() {val person = Person("Charlie", 28)
}
  • init 块会在对象创建时执行,通常用于执行一些初始化逻辑。
  • 输出:
    Person created with name: Charlie and age: 28
    

注意事项

  1. 主构造函数必须调用

    • 如果类有主构造函数,则所有次构造函数(如果有)都必须直接或间接调用主构造函数。
  2. 没有 valvar 的参数

    • 如果主构造函数中的参数没有使用 valvar 声明,则该参数不会成为类的属性,而是仅在初始化过程中可用。
    class Person(name: String, age: Int) {// name 和 age 不是类的属性
    }
    
  3. 继承时的主构造函数

    • 子类必须通过 super 调用父类的主构造函数。
    open class Animal(val name: String)
    class Dog(name: String, val breed: String) : Animal(name)
    

通过这种方式,Kotlin 提供了一种简洁的方式来处理主构造函数的参数赋值操作,既方便又高效。

三 kotlin中 主构造函数和类属性的关系

在 Kotlin 中,主构造函数和类属性之间有着紧密的关系。Kotlin 提供了一种简洁的方式来定义类的属性,并直接通过主构造函数初始化这些属性。以下是它们之间的关系及其实现方式:

主构造函数与类属性

  1. 声明属性:可以在类头(即主构造函数)中声明属性。这意味着不需要在类体内显式地编写额外的代码来初始化这些属性。

  2. 使用 valvar:当在主构造函数中声明一个参数时,可以使用 valvar 来指定该参数是否作为只读或可变属性暴露给外部。这将自动为该属性生成相应的 getter(对于 val)或 getter/setter(对于 var)方法。

  3. 自动赋值:当创建类的实例时,传递给主构造函数的参数会自动赋值给对应的属性。

示例

class Person(val name: String, var age: Int)
  • 在这个例子中,name 是一个只读属性(因为使用了 val),而 age 是一个可变属性(因为使用了 var)。这两个属性都是直接在主构造函数中声明并初始化的。
  • 创建 Person 类的对象时,需要提供 nameage 的值,例如 val person = Person("John", 30)

初始化块

Kotlin 还允许在类中定义初始化块(init 块),这些块会在对象被创建时执行,且在属性初始化之后、次构造函数之前执行。这对于执行一些基于主构造函数参数的初始化逻辑非常有用。

class Person(val name: String, var age: Int) {init {println("Person $name is $age years old.")}
}

没有 valvar 的情况

如果在主构造函数中声明参数时没有使用 valvar,那么这些参数仅在初始化过程中可用,不会成为类的属性。

class Person(name: String, age: Int) {// name 和 age 不是类的属性
}

在这种情况下,如果想让这些参数成为类的属性,需要手动将它们赋值给类中的属性。

总结

主构造函数与类属性的关系在于,Kotlin 允许直接在主构造函数中声明并初始化类的属性,从而简化代码并提高可读性。通过使用 valvar 关键字,可以控制属性的可访问性和修改性,同时利用初始化块实现更复杂的初始化逻辑。这种设计使得 Kotlin 在处理类的初始化过程时既灵活又强大。

四 类属性和局部变量

在 Kotlin 中,主构造函数中的属性与方法中的变量有不同的作用域和生命周期。

主构造函数中的属性

当在主构造函数中使用 valvar 声明参数时,这些参数会成为类的属性。这意味着它们可以在整个类的范围内访问,并且根据是使用了 val 还是 var,它们可以是只读(只能通过 getter 访问)或可变(可以通过 getter 和 setter 访问)的。这种做法简化了代码,减少了样板代码的数量,因为不需要在类体内部显式地声明和初始化这些属性。

示例:

class Person(val name: String, var age: Int)

在这个例子中,name 是一个只读属性(因为它前面有 val),而 age 是一个可变属性(因为它前面有 var)。这两个都是类属性,可以从类的任何地方访问(考虑到可见性修饰符的情况下)。

方法中的变量

相反,在方法内的变量是局部变量。它们的作用域仅限于声明它们的方法内,并且不会成为类的一部分。这意味着它们既不是类属性也不是对象状态的一部分,仅仅是用来存储方法执行期间临时数据的容器

示例:

class Example {fun doSomething() {val localVar = 10 // 局部变量println(localVar)}
}

在这个例子中,localVar 只能在 doSomething 方法内访问。一旦方法执行完毕,localVar 就会被销毁,因为它是一个局部变量。

总结

  • 主构造函数中的属性(当使用 valvar 声明时)是类属性,具有类范围的可见性和生命周期
  • 方法中的变量是局部变量,其作用域和生命周期被限制在声明它们的方法内,不作为类属性存在。

五 Kotlin 中的 set 和 get 方法

在 Kotlin 中,setget 方法是用于访问和修改类属性的特殊方法。Kotlin 提供了非常简洁的语法来处理这些方法,并且会根据属性的声明自动生成它们(除非显式地自定义)。


1. Kotlin 中的 setget 方法是什么?

  • Getter (get):用于获取属性的值。
  • Setter (set):用于设置属性的值。

在 Kotlin 中,这些方法是隐式的,默认情况下不需要显式编写代码。当使用 valvar 声明一个属性时,Kotlin 会自动生成对应的 getset 方法(如果适用)。

示例:
class Person(var name: String, var age: Int)
  • 在这个例子中:
    • name 是一个可变属性(var),所以 Kotlin 会为它生成 gettersetter 方法。
    • age 同样是一个可变属性(var),也会生成 gettersetter 方法。

可以通过 Java 的反射或调试工具看到这些方法的实际存在形式。


2. 默认类属性是否会自动生成 setget 方法?

是的,Kotlin 会自动为类属性生成 setget 方法,具体取决于属性的声明方式:

  • 对于 val 声明的属性

    • 只会生成 getter 方法,因为 val 是只读属性,不能被重新赋值。
  • 对于 var 声明的属性

    • 会生成 gettersetter 方法,因为 var 是可变属性,可以被读取和修改。
自动生成的示例:
class Person(val name: String, var age: Int)fun main() {val person = Person("Alice", 25)// 调用 getterprintln(person.name) // 自动调用 name 的 getter 方法println(person.age)  // 自动调用 age 的 getter 方法// 调用 setterperson.age = 26      // 自动调用 age 的 setter 方法
}

在这个例子中:

  • name 是只读属性(val),所以只有 getter 方法。
  • age 是可变属性(var),所以有 gettersetter 方法。

3. 局部变量会自动生成 setget 方法吗?

不会!局部变量不会自动生成 setget 方法。

局部变量的作用域仅限于声明它们的方法、代码块或表达式内,它们既不是类的成员,也没有状态管理的需求,因此没有必要生成 setget 方法。

示例:
fun exampleFunction() {val localVar = 10println(localVar) // 直接访问局部变量,没有 getter 或 setter
}
  • localVar 是一个局部变量,它的作用域仅限于 exampleFunction 方法内。
  • Kotlin 不会为局部变量生成任何 gettersetter 方法,因为它们与类的状态无关

4. 如何自定义 setget 方法?

虽然 Kotlin 默认会为类属性生成 gettersetter 方法,但可以通过自定义来改变其行为。

示例:
class Person {var name: String = "Unknown"get() = field.capitalize() // 自定义 getter,将名字首字母大写set(value) {field = value.trim()   // 自定义 setter,去除输入值的前后空格}
}fun main() {val person = Person()person.name = "  alice  "println(person.name) // 输出:Alice
}
  • field 是一个特殊的关键字,表示属性的实际存储位置(即后备字段)。
  • 在自定义的 gettersetter 中,必须通过 field 来访问或修改属性的值。

总结

  1. 类属性

    • 如果使用 valvar 声明,Kotlin 会自动生成 gettersetter 方法(val 只生成 getter)。
    • 可以通过自定义 getset 方法来改变默认行为。
  2. 局部变量

    • 局部变量不会生成 gettersetter 方法,因为它们的作用域仅限于声明它们的方法或代码块内。

六 代码示例

1 kotlin代码

package test.fclass Test5 {
}class Person(var name: String, var age: Int){fun exampleFunction() {val localVar = 10println(localVar) // 直接访问局部变量,没有 getter 或 setter}
}class Person2 {var name: String = "Unknown"get() = field.capitalize() // 自定义 getter,将名字首字母大写set(value) {field = value.trim()   // 自定义 setter,去除输入值的前后空格}
}

2 转java

// Test5.java
package test.f;import kotlin.Metadata;@Metadata(mv = {2, 0, 0},k = 1,xi = 48,d1 = {"\u0000\f\n\u0002\u0018\u0002\n\u0002\u0010\u0000\n\u0002\b\u0002\u0018\u00002\u00020\u0001B\u0005¢\u0006\u0002\u0010\u0002¨\u0006\u0003"},d2 = {"Ltest/f/Test5;", "", "()V", "untitled"}
)
public final class Test5 {
}
// Person.java
package test.f;import kotlin.Metadata;
import kotlin.jvm.internal.Intrinsics;
import org.jetbrains.annotations.NotNull;@Metadata(mv = {2, 0, 0},k = 1,xi = 48,d1 = {"\u0000\u001e\n\u0002\u0018\u0002\n\u0002\u0010\u0000\n\u0000\n\u0002\u0010\u000e\n\u0000\n\u0002\u0010\b\n\u0002\b\n\n\u0002\u0010\u0002\n\u0000\u0018\u00002\u00020\u0001B\u0015\u0012\u0006\u0010\u0002\u001a\u00020\u0003\u0012\u0006\u0010\u0004\u001a\u00020\u0005¢\u0006\u0002\u0010\u0006J\u0006\u0010\u000f\u001a\u00020\u0010R\u001a\u0010\u0004\u001a\u00020\u0005X\u0086\u000e¢\u0006\u000e\n\u0000\u001a\u0004\b\u0007\u0010\b\"\u0004\b\t\u0010\nR\u001a\u0010\u0002\u001a\u00020\u0003X\u0086\u000e¢\u0006\u000e\n\u0000\u001a\u0004\b\u000b\u0010\f\"\u0004\b\r\u0010\u000e¨\u0006\u0011"},d2 = {"Ltest/f/Person;", "", "name", "", "age", "", "(Ljava/lang/String;I)V", "getAge", "()I", "setAge", "(I)V", "getName", "()Ljava/lang/String;", "setName", "(Ljava/lang/String;)V", "exampleFunction", "", "untitled"}
)
public final class Person {@NotNullprivate String name;private int age;public Person(@NotNull String name, int age) {Intrinsics.checkNotNullParameter(name, "name");super();this.name = name;this.age = age;}@NotNullpublic final String getName() {return this.name;}public final void setName(@NotNull String var1) {Intrinsics.checkNotNullParameter(var1, "<set-?>");this.name = var1;}public final int getAge() {return this.age;}public final void setAge(int var1) {this.age = var1;}public final void exampleFunction() {int localVar = 10;System.out.println(localVar);}
}
// Person2.java
package test.f;import kotlin.Metadata;
import kotlin.jvm.internal.Intrinsics;
import kotlin.text.StringsKt;
import org.jetbrains.annotations.NotNull;@Metadata(mv = {2, 0, 0},k = 1,xi = 48,d1 = {"\u0000\u0014\n\u0002\u0018\u0002\n\u0002\u0010\u0000\n\u0002\b\u0002\n\u0002\u0010\u000e\n\u0002\b\u0006\u0018\u00002\u00020\u0001B\u0005¢\u0006\u0002\u0010\u0002R&\u0010\u0005\u001a\u00020\u00042\u0006\u0010\u0003\u001a\u00020\u00048F@FX\u0086\u000e¢\u0006\u000e\n\u0000\u001a\u0004\b\u0006\u0010\u0007\"\u0004\b\b\u0010\t¨\u0006\n"},d2 = {"Ltest/f/Person2;", "", "()V", "value", "", "name", "getName", "()Ljava/lang/String;", "setName", "(Ljava/lang/String;)V", "untitled"}
)
public final class Person2 {@NotNullprivate String name = "Unknown";@NotNullpublic final String getName() {return StringsKt.capitalize(this.name);}public final void setName(@NotNull String value) {Intrinsics.checkNotNullParameter(value, "value");this.name = StringsKt.trim((CharSequence)value).toString();}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/900697.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

PHP 过滤器

PHP 过滤器 引言 PHP作为一种广泛使用的服务器端脚本语言&#xff0c;提供了强大的数据处理能力。在处理数据时&#xff0c;确保数据的安全性和准确性至关重要。PHP过滤器&#xff08;Filters&#xff09;就是用来对数据进行预处理和后处理的工具。本文将详细介绍PHP过滤器的…

【WebRTC】开源项目Webrtc-streamer介绍

WebRTC-Streamer 这是一个用于通过简单的信令机制&#xff08;参见 api&#xff09;流式传输 WebRTC 媒体源的实验项目&#xff0c;支持以下媒体源&#xff1a; 捕获设备 屏幕捕获 mkv 文件 RMTP/RTSP 源 同时该项目也兼容 WHEP 接口。 注意 * 在线演示已停止&#xff0c…

【Java设计模式】第9章 原型模式讲解

9. 原型模式 9.1 原型模式讲解 定义:通过拷贝原型实例创建新对象,无需调用构造函数。特点: 创建型模式无需了解创建细节适用场景: 类初始化消耗资源多对象创建过程繁琐(如属性赋值复杂)循环体中需创建大量对象优点: 性能优于直接new简化创建流程缺点: 必须实现clone()…

【Java集合】LinkedList源码深度分析

参考笔记&#xff1a;java LinkedList 源码分析&#xff08;通俗易懂)_linkedlist源码分析-CSDN博客 目录 1.前言 2.LinkedList简介 3.LinkedList的底层实现 4.LinkedList 与 ArrayList 的对比 4.1 如何选择 4.2 对比图 5.LinkedList 源码Debug 5.1 add(E e) &#xff…

openssl源码分析之加密模式(modes)

openssl实现分组加密模式&#xff08;例如AES128-CBC的CBC部分&#xff09;的模块名字叫做modes&#xff0c;源代码位于 https://gitee.com/gh_mirrors/openssl/tree/master/crypto/modes 博主又打不开github了TT&#xff0c;只能找个gitee镜像 头文件是modes.h。 该模块目前…

Java 搭建 MC 1.18.2 Forge 开发环境

推荐使用 IDEA 插件 Minecraft Development 进行创建项目 创建完成后即可进行 MOD 开发。 但是关于 1.18.2 的开发教程太少&#xff0c;因此自己研究了一套写法&#xff0c;写法并非是最优的但是是探索开发MOD中的一次笔记和记录 GITHUB: https://github.com/zimoyin/zhenfa…

nginx如何实现负载均衡?

Nginx 是一款高性能的 Web 服务器和反向代理服务器&#xff0c;它可以通过配置实现负载均衡功能。以下是实现负载均衡的详细步骤和方法&#xff1a; 1. 基本概念 负载均衡是将客户端请求分发到多个后端服务器上&#xff0c;以提高系统的可用性和性能。Nginx 支持多种负载均衡策…

深度学习天崩开局

李沐大神的d2l包导入&#xff0c; 这玩意需要python311版本&#xff0c;我现在版本已经313了&#xff0c;作为一个天生要强的男人&#xff0c;我是坚决不向低版本低头的。 然后我就研究啊&#xff0c;各种翻资料啊&#xff0c;然后deepseek加豆包都翻烂了&#xff0c; 最终所…

docker部署jenkins并成功自动化部署微服务

一、环境版本清单&#xff1a; docker 26.1.4JDK 17.0.28Mysql 8.0.27Redis 6.0.5nacos 2.5.1maven 3.8.8jenkins 2.492.2 二、服务架构&#xff1a;有gateway&#xff0c;archives&#xff0c;system这三个服务 三、部署步骤 四、安装linux 五、在linux上安装redis&#…

MPDrive:利用基于标记的提示学习提高自动驾驶的空间理解能力

25年4月来自南方科技大学、百度、英国 KCL和琶洲实验室&#xff08;广东 AI 和数字经济实验室&#xff09;的论文“MPDrive: Improving Spatial Understanding with Marker-Based Prompt Learning for Autonomous Driving”。 自动驾驶视觉问答&#xff08;AD-VQA&#xff09;…

Halcon图像采集

Halcon是一款强大的机器视觉软件&#xff0c;结合C#可以开发出功能完善的视觉应用程序。 基本设置 确保已经安装了Halcon和Halcon的.NET库&#xff08;HalconDotNet&#xff09;。 1. 添加引用 在C#项目中&#xff0c;需要添加对HalconDotNet.dll的引用&#xff1a; 右键点…

Win10定时任务计划无法显示要执行的EXE任务程序界面,问题解决办法

用C#开发的一款WINFORM程序&#xff0c;在电脑测试一切顺利&#xff0c;运行结果正确。但用电脑的定时任务执行时&#xff0c;程序界面不显示&#xff0c;重启电脑、各种试都不行&#xff0c;最终问题解决。 解决办法&#xff1a; 要选“只在用户登陆时运行”&#xff0c;才能执…

Navicat和PLSQL在oracle 使用语句报ORA-00911: 无效字符

后面我发现可能是在复制SQL语句中有中文&#xff0c;但是环境变量未配置中文环境。 因为Oracle的语法解析器特别严格&#xff0c;就会报出以上的错误出来。 SQL语句错误&#xff0c;存在中文字符或者sql语句空格导致&#xff0c;去掉即可解决。 我重新写语句&#xff0c;发现…

[ctfshow web入门] web30

信息收集 题目将flag system php不区分大小写地过滤了 解题 前置知识 print_r&#xff1a;php中用于打印数组 scandir&#xff1a;php中用于获取指点目录下的所以文件目录名 getcwd&#xff1a;获取当前目录 目录获取 这里提供两种方法 print_r(scandir(getcwd())); pri…

linux下MMC_TEST的使用

一:打开如下配置,将相关文件编译到内核里: CONFIG_MMC_TEST CONFIG_MMC_DEBUG CONFIG_DEBUG_FS二:将mmc设备和mmc_test驱动进行绑定 2.1查看mmc设备编号 ls /sys/bus/mmc/drivers/mmcblk/mmc0:aaaa2.2将mmc设备与原先驱动进行解绑 echo mmc0:aaaa >

《深度解析LightGBM与MySQL数据集成:高效机器学习的新范式》

在机器学习工程实践中&#xff0c;数据与模型的高效交互一直是制约算法性能发挥的关键瓶颈。LightGBM作为梯度提升决策树框架的杰出代表&#xff0c;其与关系型数据库MySQL的深度集成能力&#xff0c;为数据科学家提供了从原始数据到预测结果的完整解决方案。这种集成不是简单的…

处理Excel的python库openpyxl、xlrd、xlwt、pandas有什么区别,搞懂它

openpyxl、xlrd、xlwt、pandas 都能处理 Excel 表格&#xff0c;但用途和适合的场景不同。今天做个总结&#xff1a; 库名功能支持格式读写支持样式备注openpyxl全面的.xlsx处理库.xlsx&#xff08;Excel2007&#xff09;✅✅✅首选xlrd读取.xls文件的老牌工具.xls&#xff08…

EasyExcel-一款好用的excel生成工具

EasyExcel是一款处理excel的工具类&#xff0c;主要特点如下&#xff08;官方&#xff09;&#xff1a; 特点 高性能读写&#xff1a;FastExcel 专注于性能优化&#xff0c;能够高效处理大规模的 Excel 数据。相比一些传统的 Excel 处理库&#xff0c;它能显著降低内存占用。…

视频分析设备平台EasyCVR携手高空抛物AI智能分析技术,打造住宅小区头顶安全智能防线

一、背景介绍 随着城市化进程的高速推进&#xff0c;城市天际线不断被刷新&#xff0c;高楼大厦密密麻麻。然而&#xff0c;高空抛物问题也逐渐显现&#xff0c;这一行为不仅严重影响城市文明的形象&#xff0c;更带来很多安全隐患&#xff0c;威胁居民的生命财产安全&#xf…

Spring MVC 操作会话属性详解(@SessionAttributes 与 @SessionAttribute)

Spring MVC 操作会话属性详解&#xff08;SessionAttributes 与 SessionAttribute&#xff09; 1. 核心注解对比 注解作用范围功能SessionAttributes类级别声明控制器中需要持久化的模型属性&#xff08;存入 HttpSession&#xff09;SessionAttribute方法参数/返回值显式绑定…