1.对象
Java 的写法
MainActivity.this
Kotlin 的写法
this@MainActivity
2.类
Java 的写法
MainActivity.class
Kotlin 的写法
MainActivity::class.java
3.继承
Java 的写法
public class MainActivity extends AppCompatActivity {
}
Kotlin 的写法(在 Kotlin 中被继承类必须被 open 关键字修饰)
class MainActivity : AppCompatActivity() {
}
4. 变量
Java 的写法
Intent intent = new Intent();
Kotlin 的写法
var intent = Intent()
5. 常量
Java 的写法
final String text = "";
Kotlin 的写法
val text = ""
6. 静态常量
Java 的写法
public class MainActivity extends AppCompatActivity {
static final String text = "";
}
Kotlin 的写法(需要注意的是要把静态变量定义在类上方)
const val text = ""
class MainActivity : AppCompatActivity() {
}
7. 定义方法
Java 的写法
public void test(String message) {
}
Kotlin 的写法(Unit 跟 void 一样效果)
fun test(message : String) : Unit {
}
// 在 Kotlin 可以省略 Unit 这种返回值
fun test(message : String) {
}
8. 重载方法
Java 的写法
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
}
}
Kotlin 的写法
class MainActivity : AppCompatActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
}
}
9. 基本数据类型
Java 的写法
int i = 1;
long l = 2;
boolean b = true;
float f = 0;
double d = 0;
char c = 'A';
String s = "text";
Kotlin 的写法
var i : Int = 1
var l : Long = 2
var b : Boolean = true
var f : Float = 0F
var d : Double = 0.0
var c : Char = 'A'
var s : String = "text"
// 更简洁点可以这样,自动推倒类型
var i = 1
var l = 2
var b = true
var f = 0F
var d = 0.0
var c = 'A'
var s = "text"
10. 比较类型
Java 的写法
if ("" instanceof String) {
}
Kotlin 的写法
if ("" is String) {
}
11. 转换符
Java 的写法
int number = 100;
System.out.println(String.format("数量有%d", number));
Kotlin 的写法
var number = 100
println("数量有${number}")
// 换种简洁的写法
var number = 100
println("数量有$number")
// 如果不想字符串被转义可以使用\$
var number = 100
println("数量有\$number")
12. 字符串比较
Java 的写法
String s1 = "text";
String s2 = "text";
if (s1.equals(s2)) {
}
Kotlin 的写法(Kotlin 对字符串比较的写法进行优化了,其他类型对象对比还是要用 equals 方法)
var s1 = "text"
var s2 = "text"
if (s1 == s2) {
}
13. 数组
Java 的写法
int[] array1 = {1, 2, 3};
float[] array2 = {1f, 2f, 3f};
String[] array3 = {"1", "2", "3"};
Kotlin 的写法
val array1 = intArrayOf(1, 2, 3)
val array2 = floatArrayOf(1f, 2f, 3f)
val array3 = arrayListOf("1", "2", "3")
14. 循环
Java 的写法
String[] array = {"1", "2", "3"};
for (int i = 0; i < array.length; i++) {
System.out.println(array[i]);
}
Kotlin 的写法
val array = arrayListOf("1", "2", "3")
for (i in array.indices) {
println(array[i])
}
15. 下标循环
Java 的写法
String[] array = {"1", "2", "3"};
for (int i = 1; i < array.length; i++) {
System.out.println(array[i]);
}
Kotlin 的写法(这种写法在 Kotlin 中称之为区间)
val array = arrayListOf("1", "2", "3")
for (i in IntRange(1, array.size - 1)) {
println(array[i])
}
// 换种更简洁的写法
val array = arrayListOf("1", "2", "3")
for (i in 1..array.size - 1) {
println(array[i])
}
// 编译器提示要我们换种写法
val array = arrayListOf("1", "2", "3")
for (i in 1 until array.size) {
println(array[i])
}
16. 高级循环
Java 的写法
String[] array = {"1", "2", "3"};
for (String text : array) {
System.out.println(text);
}
Kotlin 的写法
val array = arrayListOf("1", "2", "3")
for (text in array) {
println(text)
}
17. 判断器
Java 的写法
int count = 1;
switch (count) {
case 0:
System.out.println(count);
break;
case 1:
case 2:
System.out.println(count);
break;
default:
System.out.println(count);
break;
}
Kotlin 的写法
var count = 1
when (count) {
0 -> {
println(count)
}
in 1..2 -> {
println(count)
}
else -> {
println(count)
}
}
var count = 1
// 换种更简洁的写法
when (count) {
0 -> println(count)
in 1..2 -> println(count)
else -> println(count)
}
19. 构造函数
Java 的写法
public class MyView extends View {
public MyView(Context context) {
this(context, null);
}
public MyView(Context context, @Nullable AttributeSet attrs) {
this(context, attrs, 0);
}
public MyView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
}
}
Kotlin 的写法(只有一种构造函数直接在对象后面加,多个函数次函数 用关键字 constructor )
class MyView : View {
constructor(context : Context) : this(context, null) {
}
constructor(context : Context, attrs : AttributeSet?) : this(context, attrs, 0) {
}
constructor(context : Context, attrs : AttributeSet?, defStyleAttr : Int) : super(context, attrs, defStyleAttr) {
}
}
// 换种更简洁的写法
class MyView : View {
constructor(context : Context) : this(context, null)
constructor(context : Context, attrs : AttributeSet?) : this(context, attrs, 0)
constructor(context : Context, attrs : AttributeSet?, defStyleAttr : Int) : super(context, attrs, defStyleAttr)
}
// 只有一种构造函数的还可以这样写
class MyView(context: Context?) : View(context) {
}
19. 类创建
Java 的写法
public class Person {
String name;
int age;
public Person(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
}
Person person = new Person("Zhangsan", 100);
person.setName("ZS");
person.setAge(18);
System.out.println("name: " + person.getName() + ", age: " + person.getAge());
Kotlin 的写法(如果不想暴露成员变量的set方法,可以将 var 改成 val )
class Person {
var name : String? = null
get() = field
set(value) {field = value}
var age : Int = 0
get() = field
set(value) {field = value}
}
// 换种更简洁的写法
class Person(var name : String, var age : Int)
var person = Person("Zhangsan", 100)
person.name = "ZS"
person.age = 18
println("name: {$person.name}, age: {$person.age}")
20. 接口
Java 的写法
public interface Callback {
void onSuccess();
void onFail();
}
Kotlin 的写法(Kotlin接口方法里面是可以自己实现,这里就不再演示了)
interface Callback {
fun onSuccess()
fun onFail()
}
21. 匿名内部类
Java 的写法
new Callback() {
@Override
public void onSuccess() {
}
@Override
public void onFail() {
}
};
Kotlin 的写法
object:Callback {
override fun onSuccess() {
}
override fun onFail() {
}
}
22. 内部类
Java 的写法
public class MainActivity extends AppCompatActivity {
public class MyTask {
}
}
Kotlin 的写法
class MainActivity : AppCompatActivity() {
inner class MyTask {
}
}
23. 内部类访问外部类同名变量
Java 的写法
String name = "Zhangsan";
public class MyTask {
String name = "ZS";
public void show() {
System.out.println(name + "---" + MainActivity.this.name);
}
}
Kotlin 的写法
var name = "Zhangsan"
inner class MyTask {
var name = "ZS"
fun show() {
println(name + "---" + this@MainActivity.name)
}
}
24. 抽象类
Java 的写法
public abstract class BaseActivity extends AppCompatActivity implements Runnable {
abstract void init();
}
Kotlin 的写法
abstract class BaseActivity : AppCompatActivity(), Runnable {
abstract fun init()
}
25. 静态变量和方法
Java 的写法
public class ToastUtils {
public static Toast sToast;
public static void show() {
sToast.show();
}
}
Kotlin 的写法(在 Kotlin 将这种方式称之为伴生对象 companion object)
companion object ToastUtils {
var sToast : Toast? = null
fun show() {
sToast!!.show()
}
}
26. 可变参数
Java 的写法
public int add(int... array) {
int count = 0;
for (int i : array) {
count += i;
}
return count;
}
Kotlin 的写法
fun add(vararg array: Int) : Int {
var count = 0
//for (i in array) {
// count += i
//}
array.forEach {
count += it
}
return count
}
27.泛型
Java 的写法
public class Bean<T extends String> {
T data;
public Bean(T t) {
this.data = t;
}
}
Bean<String> bean = new Bean<>("666666");
Kotlin 的写法
class Bean<T : Comparable<String>>(t: T) {
var data = t
}
var bean = Bean<String>("666666")
// 换种更简洁的写法
var bean = Bean("666666")
在Kotlin中,你不能直接使用 <T: String>
这样的语法来限制泛型类型参数。Kotlin的泛型类型参数的上界约束语法略有不同。在Java中,你可以这样声明一个泛型类:
java
public class Bean<T extends String> { private T value; public Bean(T value) { this.value = value; } // ... }
但是在Kotlin中,你不能直接使用 class Bean<T: String>(...)
的形式,因为String
是一个具体的类,而不是一个类型参数。在Kotlin中,你应该使用 Comparable<String>
或者其他更通用的类型作为上界,这是因为泛型类型参数的约束通常需要是一个类型参数或者一个接口。这是因为String
类本身并不接受类型参数,所以你不能直接将String
用作类型参数的上界。
如果你想要一个泛型类只接受String
或其子类作为类型参数,你实际上只需要将泛型参数限制为String
即可,但是由于String
没有子类(在标准的JVM环境中),这实际上限制了泛型参数只能是String
类型。在Kotlin中,你可以这样声明:
kotlin
class Bean<T : String>(t: T) { val value: T = t }
或者,如果你想让泛型参数可以是任何可比较的类型,包括String
,你可以这样声明:
kotlin
class Bean<T : Comparable<T>>(t: T) { val value: T = t }
在上面的例子中,Bean
类的构造函数接收一个类型为T
的参数t
,并且T
必须是Comparable<T>
的子类型。这意味着T
必须能够与其他T
类型的对象进行比较,这包括String
,因为String
实现了Comparable<String>
接口。
然而,如果你只是想让Bean
类的类型参数只能是String
,最直接的方式是在构造函数中直接声明类型:
kotlin
class Bean(t: String) { val value: String = t }
这种方式虽然失去了泛型的好处,但它确保了value
成员变量的类型始终是String
。
28. Lambda {参数名1: 参数类型, 参数名2:参数类型 -> 函数体}
Lambda 表达式虽然在 Java JDK 已经加上了,但是没有普及开来,现在搭配 Kotlin 是一个不错的选择
tv_content.setOnClickListener(View.OnClickListener(
fun(v : View) {
v.visibility = View.GONE
}
))
现在可以用 Lambda 表达式进行简化
tv_content.setOnClickListener { v -> v.visibility = View.GONE } (省略写法,只有一个参数 去除 括号, 把lambda 提到外面 )
29. 空安全
在 Java 不强制我们处理空对象,所以常常会导致 NullPointerException 空指针出现,现在 Kotlin 对空对象进行了限定,必须在编译时处理对象是否为空的情况,不然会编译不通过
在对象不可空的情况下,可以直接使用这个对象
fun getText() : String {
return "text"
}
val text = getText()
print(text.length)
在对象可空的情况下,必须要判断对象是否为空
fun getText() : String? {
return null
}
val text = getText()
if (text != null) {
print(text.length)
}
// 如果不想判断是否为空,可以直接这样,如果 text 对象为空,则会报空指针异常,一般情况下不推荐这样使用
val text = getText()
print(text!!.length)
// 还有一种更好的处理方式,如果 text 对象为空则不会报错,但是 text.length 的结果会等于 null
val text = getText()
print(text?.length)
30. 方法支持添加默认参数
在 Java 上,我们可能会为了扩展某个方法而进行多次重载
public void toast(String text) {
toast(this, text, Toast.LENGTH_SHORT);
}
public void toast(Context context, String text) {
toast(context, text, Toast.LENGTH_SHORT);
}
public void toast(Context context, String text, int time) {
Toast.makeText(context, text, time).show();
}
toast("弹个窗");
toast(this, "弹个窗");
toast(this, "弹个窗", Toast.LENGTH_LONG);
但是在 Kotlin 上面,我们无需进行重载,可以直接在方法上面直接定义参数的默认值
fun toast(context : Context = this, text : String, time : Int = Toast.LENGTH_SHORT) {
Toast.makeText(context, text, time).show()
}
toast(text = "弹个窗")
toast(this, "弹个窗")
toast(this, "弹个窗", Toast.LENGTH_LONG)
31. 扩展函数
扩展函数是 Kotlin 用于简化一些代码的书写产生的,其中有 let、with、run、apply、also 五个函数
(1) let 函数
在函数块内可以通过 it 指代该对象。返回值为函数块的最后一行或指定return表达式
一般写法
fun main() {
val text = "CurvedBowZhang"
println(text.length)
val result = 1000
println(result)
}
let 写法
fun main() {
val result = "CurvedBowZhang".let {
println(it.length)
1000
}
println(result)
}
最常用的场景就是使用let函数处理需要针对一个可null的对象统一做判空处理
mVideoPlayer?.setVideoView(activity.course_video_view)
mVideoPlayer?.setControllerView(activity.course_video_controller_view)
mVideoPlayer?.setCurtainView(activity.course_video_curtain_view)
mVideoPlayer?.let {
it.setVideoView(activity.course_video_view)
it.setControllerView(activity.course_video_controller_view)
it.setCurtainView(activity.course_video_curtain_view)
}
又或者是需要去明确一个变量所处特定的作用域范围内可以使用
(2) with 函数
前面的几个函数使用方式略有不同,因为它不是以扩展的形式存在的。它是将某对象作为函数的参数,在函数块内可以通过 this 指代该对象。返回值为函数块的最后一行或指定return表达式
定义 Person 类
class Person(var name : String, var age : Int)
一般写法
fun main() {
var person = Person("CurvedBowZhang", 100)
println(person.name + person.age)
var result = 1000
println(result)
}
with 写法
fun main() {
var result = with(Person("CurvedBowZhang", 100)) {
println(name + age)
1000
}
println(result)
}
适用于调用同一个类的多个方法时,可以省去类名重复,直接调用类的方法即可,经常用于 Android 中RecyclerView中onBinderViewHolder中,数据 model 的属性映射到 UI 上
override fun onBindViewHolder(holder: ViewHolder, position: Int){
val item = getItem(position)?: return
holder.nameView.text = "姓名:${item.name}"
holder.ageView.text = "年龄:${item.age}"
}
override fun onBindViewHolder(holder: ViewHolder, position: Int){
val item = getItem(position)?: return
with(item){
holder.nameView.text = "姓名:$name"
holder.ageView.text = "年龄:$age"
}
}
(3) run 函数
实际上可以说是let和with两个函数的结合体,run函数只接收一个lambda函数为参数,以闭包形式返回,返回值为最后一行的值或者指定的return的表达式
一般写法
var person = Person("CurvedBowZhang", 100)
println(person.name + "+" + person.age)
var result = 1000
println(result)
run 写法
var person = Person("CurvedBowZhang", 100)
var result = person.run {
println("$name + $age")
1000
}
println(result)
适用于let,with函数任何场景。
因为run函数是let,with两个函数结合体,准确来说它弥补了let函数在函数体内必须使用it参数替代对象,在run函数中可以像with函数一样可以省略,直接访问实例的公有属性和方法;
另一方面它弥补了with函数传入对象判空问题,在run函数中可以像let函数一样做判空处理,这里还是借助 onBindViewHolder 案例进行简化
override fun onBindViewHolder(holder: ViewHolder, position: Int){
val item = getItem(position)?: return
holder.nameView.text = "姓名:${item.name}"
holder.ageView.text = "年龄:${item.age}"
}
override fun onBindViewHolder(holder: ViewHolder, position: Int){
val item = getItem(position)?: return
item?.run {
holder.nameView.text = "姓名:$name"
holder.ageView.text = "年龄:$age"
}
}
(4) apply 函数
从结构上来看apply函数和run函数很像,唯一不同点就是它们各自返回的值不一样,run函数是以闭包形式返回最后一行代码的值,而apply函数的返回的是传入对象的本身
一般写法
val person = Person("CurvedBowZhang", 100)
person.name = "ZJX"
person.age = 50
apply 写法
val person = Person("CurvedBowZhang", 100).apply {
name = "ZJX"
age = 50
}
整体作用功能和run函数很像,唯一不同点就是它返回的值是对象本身,而run函数是一个闭包形式返回,返回的是最后一行的值。正是基于这一点差异它的适用场景稍微与run函数有点不一样。
apply一般用于一个对象实例初始化的时候,需要对对象中的属性进行赋值。或者动态inflate出一个XML的View的时候需要给View绑定数据也会用到,这种情景非常常见。特别是在我们开发中会有一些数据model向View model转化实例化的过程中需要用到
mRootView = View.inflate(activity, R.layout.example_view, null)
mRootView.tv_cancel.paint.isFakeBoldText = true
mRootView.tv_confirm.paint.isFakeBoldText = true
mRootView.seek_bar.max = 10
mRootView.seek_bar.progress = 0
使用 apply 函数后的代码是这样的
mRootView = View.inflate(activity, R.layout.example_view, null).apply {
tv_cancel.paint.isFakeBoldText = true
tv_confirm.paint.isFakeBoldText = true
seek_bar.max = 10
seek_bar.progress = 0
}
多层级判空问题
if (mSectionMetaData == null || mSectionMetaData.questionnaire == null || mSectionMetaData.section == null) {
return;
}
if (mSectionMetaData.questionnaire.userProject != null) {
renderAnalysis();
return;
}
if (mSectionMetaData.section != null && !mSectionMetaData.section.sectionArticles.isEmpty()) {
fetchQuestionData();
return;
}
kotlin的apply函数优化
mSectionMetaData?.apply {
//mSectionMetaData不为空的时候操作mSectionMetaData
}?.questionnaire?.apply {
//questionnaire不为空的时候操作questionnaire
}?.section?.apply {
//section不为空的时候操作section
}?.sectionArticle?.apply {
//sectionArticle不为空的时候操作sectionArticle
}
(5) also 函数
also函数的结构实际上和let很像唯一的区别就是返回值的不一样,let是以闭包的形式返回,返回函数体内最后一行的值,如果最后一行为空就返回一个Unit类型的默认值。而also函数返回的则是传入对象的本身
fun main() {
val result = "CurvedBowZhang".let {
println(it.length)
1000
}
println(result) // 打印:1000
}
fun main() {
val result = "CurvedBowZhang".also {
println(it.length)
}
println(result) // 打印:CurvedBowZhang
}
适用于let函数的任何场景,also函数和let很像,只是唯一的不同点就是let函数最后的返回值是最后一行的返回值而also函数的返回值是返回当前的这个对象。一般可用于多个扩展函数链式调用
(6) 小结
通过以上几种函数的介绍,可以很方便优化 Kotlin 中代码编写,整体看起来几个函数的作用很相似,但是各自又存在着不同。使用的场景也有相同的地方,比如:run函数就是let和with的结合体。