Kotlin memory optimization from a compilation perspective

Kotlin memory optimization from a compilation perspective

Author: Yan Yongjun, Unit: China Mobile Smart Home Operation Center

Labs Guide

Today we are going to talk about Kotlin, a static programming language developed by JetBrains for modern multi-platform applications.

Kotlin can be compiled into Java bytecode or JavaScript, which is convenient for running on devices without JVM. In addition, Kotlin can also be compiled into binary code and run directly on the machine.

At Google I/O 2017, Google announced first-class support for Kotlin on Android. Currently, Kotlin has become the preferred language for Android application development.

Kotlin has many advantages over Java, such as null safety, easier-to-use Lambda expressions, support for extensions, numerous syntax sugars, etc. However, few people mention Kotlin's memory optimization for Java from a compilation perspective. Here we take a peek through decompilation.

Part 01 Java inner class holds reference to outer class

There is a common perception in Java that inner classes in Java hold references to outer classes, and improper use can easily cause memory leaks. See the following example.

We write the following code to verify.

We first create a parent class to observe whether the subclass will call the finalize method.

 public class BaseActivity extends AppCompatActivity {
@Override
protected void finalize() throws Throwable {
Log.e("yanlog", "BaseActivity finalize:" + this);
super.finalize();
}
}

We create a subclass, and in the subclass we create a Thread that will not terminate. ​

 public class TmpJavaActivity extends BaseActivity{
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
Thread thread = new Thread(new Runnable() {
@Override
public void run() {
while(true){
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
});
thread.start();
}
}

Through testing, we can find that the finalize method of BaseActivity can never be called, that is, the external class TmpJavaActivity can never be recycled. We use the decompilation tool jadx to observe the decompiled smail code.


From the above decompiled smail code, we can see that Java will pass the outer class object as a parameter to the inner class object. Once the inner class cannot be released, the outer class will not be released, thus causing a memory leak.

Part 02 Kotlin inner classes do not necessarily hold references to outer classes

We can write the same code above in Kotlin as follows:

 class TmpActivity : BaseActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_tmp)
val thread = Thread {
while (true) {
Thread.sleep(1000)
}
}
thread.start()
}
}

By observing the logs, we found that the external class TmpActivity can be recycled normally. Let's take a look at the smail source code.

From the above smail source code, we can see that, unlike Java, inner classes in Kotlin will be compiled into a normal class. Because the actual operation of inner classes does not depend on outer classes, after compilation, the outer class will not be passed to the inner class as a parameter of the inner class constructor, that is, the inner class will not hold the application of the outer class, so it will not cause memory leaks.

But what if the inner class actually needs to hold a reference to the outer class? Let's look at the following code.

 class TmpActivity : BaseActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_tmp)
val thread = Thread {
while (true) {
Log.e("yanlog","thread"+this@TmpActivity)
java.lang.Thread.sleep(1000)
}
}
thread.start()
}
}

Observe the decompiled smail source code as follows

From the above source code, we can see that when constructing an object of an inner class, a reference to the outer class is passed to the inner class, causing a memory leak.

From the above two examples, we can see that in Kotlin, inner classes do not hold references to outer classes unless necessary, which reduces the possibility of memory leaks compared to Java.

<<:  Application of workflow engine in vivo marketing automation

>>:  Why do you advise everyone to wait for the official version of iOS 15.6? There are four reasons

Recommend

Get these six points right, attracting seed users won’t be a problem (Part 1)

Probably, the first and most important operationa...

8 ways to attract traffic from Douyin private domain

The number of daily active users of Tik Tok has e...

"Space Express" is delivered! The launch of Tianzhou-7 was a complete success

At 22:27 on January 17, the Long March 7 Yao-8 ca...

Seth Klarman is a truly great book

I took a lot of notes while reading this book, es...

How can you make more money than others by copying others?

Preface Wrong, let me ask you a question, how do ...

6 good ways to clean up mobile phone garbage, 99% of people don’t know!

The phone memory is always insufficient, and clea...

Taobao Double Eleven event, how to blow up your screen?

The Taobao Double Eleven event is getting closer ...

2 cases to teach you the correct approach to user operations!

Everyone knows about user operation , but most pe...