Skip to content

How Android Apps are Built and Run

David Griffiths edited this page Jan 28, 2017 · 12 revisions

Android Studio will take care of the details how to build and run your apps. But have you ever wondered what happens under the covers when you press that Run button?

This is how they’re built

Java compilation

Your code is written in Java. But is that Java code compiled and run the same way as, say, a web application?

The compilation process for Android apps is very different from other Java applications. But it begins in the same way: your Java source code files are compiled into .class files using the javac command:

This converts Java code like this:

public MainActivity() {
  super();
  currentPosition = 0;
}

into Java byte-codes representing Java assembly that looks something like this:

public com.hfad.bitsandpizzas.MainActivity();
  Code:
   0:	aload_0
   1:	invokespecial	#5; //Method android/app/Activity."<init>":()V
   4:	aload_0
   5:	iconst_0
   6:	putfield	#3; //Field currentPosition:I
   9:	return

Conversion to Dalvik bytecodes

The .class files contain standard Oracle JVM Java byte-codes. But Android devices don’t use this byte-code format. Instead, Android has its own distinct byte-code format called Dalvik. Dalvik byte-codes, like Oracle JVM byte-codes, are machine-code instructions for a theoretical processor.

The compilation process needs to convert the .class files, and any .jar libraries into a single classes.dex file containing Dalvik byte-codes. This is done with the dx command:

The dx command stitches all of the .class and .jar files together into a single classes.dex file written in Dalvik byte-code format.

The Dalvik assembly of our original Java code will look something like this:

0x0000: iput-object v1, v0, Lcom/hfad/bitsandpizzas/MainActivity; com.hfad.bitsandpizzas.MainActivity$2.this$0 // field@4869
0x0002: invoke-direct {v0}, void java.lang.Object.<init>() // method@13682
0x0005: return-void

Put classes.dex and resources into a package file

The classes.dex file and the resources from your application, such as images and layouts, are then compressed into a zip-like file called an Android Package or .apk file. This is done with the Android Asset Packaging Tool or aapt:

The .apk file is the application package that you can distribute. However, there’s one more step that you might need to do…

You might then also sign the .apk file

If you’re going to distribute your app through the Google Play Store, you will need to sign it. Signing an app package means that you store an additional file in the .apk that is based on a checksum of the contents of the .apk and a separately generated private key.

The .apk file uses the standard jarsigner tool that comes as part of Oracle’s Java Development Kit. The jarsigner tool was created to sign .jar files, but it will also work with .apk files as they are also zip-compressed files.

If you sign the .apk file, you will then also need to run it through a tool called zipalign, which will make sure that the compressed parts of the file are lined up on byte-boundaries. Android wants them byte aligned so it can read them easily without needing to uncompress the file.

This is how they’re deployed

The adb server starts if it’s not already running

The app will be deployed to an Android device using the Android Debug Bridge. This involves running an adb server process on your development client and a similar adb inside the Android device. If the adb process is not running on your machine, the adb command will start it.

The adb process will open a network socket, and listen for commands on port 5037. Every adb command you enter will send its instructions to this port.

The .apk file is transferred to the device

The adb command is used to transfer the .apk file into the file system on the Android device. The location is defined by the package-name of the app. So, for example, if the package is com.hfad.bitsandpizzas, then the .apk file will be placed in /data/app/com.hfad.bitsandpizzas.

This is how they run

The way that Android apps are run has changed fairly recently. Since API level 21, the older Dalvik virtual machine has been replaced with a new Android Runtime. Let’s look at what happens, step-by-step, when an app is run.

A user asks for an app to be launched

A process called Zygote is used to launch the app. Zygote is an incomplete version of an Android process–its memory space contains all the core libraries that are needed by any app, but it doesn’t yet include any of the code that’s specific to a particular app.

Zygote creates a copy of itself using the fork system call. Android is a Linux system, and the fork call can duplicate a process like Zygote very quickly. This is the reason Zygote process is used: it’s a lot faster to duplicate an half-started process like Zygote, than it is to load a new process from the main system files. Zygote means your app launches faster.

Android converts the .dex code to native OAT format

The new app process now needs to load the code that’s specific to your app. Remember, your app code is stored in the classes.dex file inside your .apk package. So the classes.dex file is extracted from the .apk and placed into a separate directory. But rather than simply place a copy of the classes.dex file, Android will convert the Dalvik byte-codes in classes.dex into native machine code.

That’s right. All of that code that starts life as Java code is now converted into a piece of native compiled code. Technically, the classes.dex will be converted into an ELF shared object. Android calls this library format OAT and the tool that converts the classes.dex file is called dex2oat.

The converted file is stored into a directory with a name like this:

/data/dalvik-cache/x86/data@[email protected]@[email protected]

The path will include the package-name of the app, to ensure that it won’t overwrite any other app.

The converted code will be in machine code that’s specific to the CPU of the Android device. For example, if the Android device is an x86, the OAT file will look like this:

0x001db888:         85842400E0FFFF    	test    eax, [esp + -8192]
suspend point dex PC: 0x0000
GC map objects:  v0 (r5), v1 (r6)
0x001db88f:                 83EC1C    	sub     esp, 28
0x001db892:               896C2410    	mov     [esp + 16], ebp
0x001db896:               89742414    	mov     [esp + 20], esi
0x001db89a:               897C2418    	mov     [esp + 24], edi
0x001db89e:                   8BF8    	mov     edi, eax
0x001db8a0:                 890424    	mov     [esp], eax
...

The app loads the native library

This native library is then mapped directly into the memory of the app process.

And from there, the app will start the initial activity and the app will appear on the screen.