Due to Android's memory management scheme, you will often find yourself needing to communicate with different components of your application, system components, or other applications installed on the phone. Parcelable
will help you pass data between these components.
Android uses Binder to facilitate such communication in a highly optimized way. The Binder communicates with Parcels, which is a message container. The Binder marshals the Parcel to be sent, sends and receives it, and then unmarshals it on the other side to reconstruct a copy of the original Parcel.
With the update to kotlin, you can use the plugin kotlin-parcelize Add
apply plugin: 'kotlin-parcelize'
to the top of your app's build.gradle
.
import kotlinx.parcelize.Parcelize
@Parcelize
class User(val firstName: String, val lastName: String, val age: Int): Parcelable
See kotlin-parcelize for more examples of how to use @Parcelize annotation
The simplest way to create a Parcelable in Java is to use a third-party library called Parceler. See this guide on Parceler to see how to use it. By annotating your Java classes that you intend to use, the library is able to create much of the boilerplate code needed as discussed in the manual steps below.
To allow for your class instances to be sent as a Parcel you must implement the Parcelable
interface along with a static field called CREATOR
, which itself requires a special constructor in your class.
Here is a typical implementation:
public class MyParcelable implements Parcelable {
// You can include parcel data types
private int mData;
private String mName;
// We can also include child Parcelable objects. Assume MySubParcel is such a Parcelable:
private MySubParcelable mInfo;
// This is where you write the values you want to save to the `Parcel`.
// The `Parcel` class has methods defined to help you save all of your values.
// Note that there are only methods defined for simple values, lists, and other Parcelable objects.
// You may need to make several classes Parcelable to send the data you want.
@Override
public void writeToParcel(Parcel out, int flags) {
out.writeInt(mData);
out.writeString(mName);
out.writeParcelable(mInfo, flags);
}
// Using the `in` variable, we can retrieve the values that
// we originally wrote into the `Parcel`. This constructor is usually
// private so that only the `CREATOR` field can access.
private MyParcelable(Parcel in) {
mData = in.readInt();
mName = in.readString();
mInfo = in.readParcelable(MySubParcelable.class.getClassLoader());
}
public MyParcelable() {
// Normal actions performed by class, since this is still a normal object!
}
// In the vast majority of cases you can simply return 0 for this.
// There are cases where you need to use the constant `CONTENTS_FILE_DESCRIPTOR`
// But this is out of scope of this tutorial
@Override
public int describeContents() {
return 0;
}
// After implementing the `Parcelable` interface, we need to create the
// `Parcelable.Creator<MyParcelable> CREATOR` constant for our class;
// Notice how it has our class specified as its type.
public static final Parcelable.Creator<MyParcelable> CREATOR
= new Parcelable.Creator<MyParcelable>() {
// This simply calls our new constructor (typically private) and
// passes along the unmarshalled `Parcel`, and then returns the new object!
@Override
public MyParcelable createFromParcel(Parcel in) {
return new MyParcelable(in);
}
// We just need to copy this and change the type to match our class.
@Override
public MyParcelable[] newArray(int size) {
return new MyParcelable[size];
}
};
}
Note that the Parcelable
interface has two methods defined: int describeContents()
and void writeToParcel(Parcel dest, int flags)
. After implementing the Parcelable
interface, we need to create the Parcelable.Creator<MyParcelable> CREATOR
constant for our class which requires us to define createFromParcel
, newArray
.
We can now pass the parcelable data between activities within an intent:
// somewhere inside an Activity
MyParcelable dataToSend = new MyParcelable();
Intent i = new Intent(this, NewActivity.class);
i.putExtra("myDataKey", dataToSend); // using the (String name, Parcelable value) overload!
startActivity(i); // dataToSend is now passed to the new Activity
and then access the data in the NewActivity
that was launched using:
public class NewActivity extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
MyParcelable object = (MyParcelable) getIntent().getParcelableExtra("myDataKey");
}
}
Now we can access the parcelable data from within the launched activity!
If a launched activity is returning a data result back to the parent activity, the onActivityResult()
method in the parent activity is invoked:
// ActivityOne.java, time to handle the result of the sub-activity
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
// REQUEST_CODE is defined above
if (resultCode == RESULT_OK && requestCode == REQUEST_CODE) {
// Extract object from result extras
// Make sure the key here matches the one specified in the result passed from ActivityTwo.java
MyParcelable object = data.getParcelableExtra("myDataKey");
}
}
Instead of using getIntent()
to retrieve the passed object in this case, we access the Intent
object via the parameter
representing the result (in this example, "data").
You may notice some similarities between Parcelable
and Serializable
. DO NOT, I repeat, DO NOT attempt to persist Parcel
data. It is meant for high-performance transport and you could lose data by trying to persist it.
Using Parcelable
compared to Serializable
can achieve up to 10x performance increase in many cases for transport which is why it's the Android preferred method.
There are a few common gotchas associated to Parcelable to consider below:
One very important thing to pay close attention to is the order that you write and read your values to and from the Parcel. They need to match up in both cases. In my example, I write the int
and then the String
to the Parcel. Afterwards, I read them in that same exact order. The mechanism that Android uses to read the Parcel is blind and completely trusts you to get the order correct, or else you will run into run-time crashes.
Another problem I have encountered is with ClassNotFound
exceptions. This is an issue with the Classloader not finding your class. To fix this you can manually set the Classloader to use. If nothing is set, then it will try the default Classloader which leads to the exception.
As mentioned before you can only put primitives, lists and arrays, Strings, and other Parcelable objects into a Parcel. This means that you cannot store framework dependent objects that are not Parcelable. For example, you could not write a Drawable
to a Parcel. To work around this problem, you can instead do something like writing the resource ID of the Drawable as an integer to the Parcel. On the receiving side you can try to rebuild the Drawable using that. Remember, Parcel is supposed to be fast and lightweight! (though it is interesting to see Bitmap
implementing Parcelable)
Where is boolean
!? For whatever odd reason there is no simple way to write a boolean to a Parcel. To do so, you can instead write a byte
with the corresponding value with out.writeByte((byte) (myBoolean ? 1 : 0));
and retrieve it similarly with myBoolean = in.readByte() != 0;
There is a Parcelable plugin that can be imported directly into IntelliJ or Android Studio, which enables you to generate the boilerplate code for creating Parcelables. You can install this plugin by going to Android Studio
-> File
-> Settings
-> Plugins
-> Browse repositories
:
Here are all the Java types it supports:
Serializable
/Parcelable
implementation) for: Date
, Bundle
Parcelable
objectslong
, int
, float
, double
, boolean
, byte
, String
Parcel.writeValue(Object)
): Integer
, Long
, Float
, Double
, Boolean
, Byte
boolean[]
, byte[]
, char[]
, double[]
, float[]
, int[]
, long[]