使用Intent传递数据大家都知道,但是如果你使用Intent传递大于1Mb的数据时,就一定会报如下的错误
2022-06-25 11:37:16.601 5901-5901/com.openld.senior E/JavaBinder: !!! FAILED BINDER TRANSACTION !!! (parcel size = 1049112)
2022-06-25 11:37:16.602 5901-5901/com.openld.senior E/AndroidRuntime: FATAL EXCEPTION: main
Process: com.openld.senior, PID: 5901
java.lang.RuntimeException: Failure from system
at android.app.Instrumentation.execStartActivity(Instrumentation.java:1711)
at android.app.Activity.startActivityForResult(Activity.java:5192)
at androidx.activity.ComponentActivity.startActivityForResult(ComponentActivity.java:574)
at android.app.Activity.startActivityForResult(Activity.java:5150)
at androidx.activity.ComponentActivity.startActivityForResult(ComponentActivity.java:560)
at android.app.Activity.startActivity(Activity.java:5521)
at android.app.Activity.startActivity(Activity.java:5489)
at com.openld.seniorstructure.main.testintentbigdata.TestIntentTransBIgDataActivity.addListeners$lambda-0(TestIntentTransBIgDataActivity.kt:39)
at com.openld.seniorstructure.main.testintentbigdata.TestIntentTransBIgDataActivity.$r8$lambda$57fJAR8O7Q8Thsg13Hl8D6OIbjo(Unknown Source:0)
at com.openld.seniorstructure.main.testintentbigdata.TestIntentTransBIgDataActivity$$ExternalSyntheticLambda0.onClick(Unknown Source:2)
at android.view.View.performClick(View.java:7125)
at android.view.View.performClickInternal(View.java:7102)
at android.view.View.access$3500(View.java:801)
at android.view.View$PerformClick.run(View.java:27336)
at android.os.Handler.handleCallback(Handler.java:883)
at android.os.Handler.dispatchMessage(Handler.java:100)
at android.os.Looper.loop(Looper.java:214)
at android.app.ActivityThread.main(ActivityThread.java:7356)
at java.lang.reflect.Method.invoke(Native Method)
at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:492)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:930)
Caused by: android.os.TransactionTooLargeException: data parcel size 1049112 bytes
at android.os.BinderProxy.transactNative(Native Method)
at android.os.BinderProxy.transact(BinderProxy.java:510)
at android.app.IActivityTaskManager$Stub$Proxy.startActivity(IActivityTaskManager.java:3847)
at android.app.Instrumentation.execStartActivity(Instrumentation.java:1705)
at android.app.Activity.startActivityForResult(Activity.java:5192)
at androidx.activity.ComponentActivity.startActivityForResult(ComponentActivity.java:574)
at android.app.Activity.startActivityForResult(Activity.java:5150)
at androidx.activity.ComponentActivity.startActivityForResult(ComponentActivity.java:560)
at android.app.Activity.startActivity(Activity.java:5521)
at android.app.Activity.startActivity(Activity.java:5489)
at com.openld.seniorstructure.main.testintentbigdata.TestIntentTransBIgDataActivity.addListeners$lambda-0(TestIntentTransBIgDataActivity.kt:39)
at com.openld.seniorstructure.main.testintentbigdata.TestIntentTransBIgDataActivity.$r8$lambda$57fJAR8O7Q8Thsg13Hl8D6OIbjo(Unknown Source:0)
at com.openld.seniorstructure.main.testintentbigdata.TestIntentTransBIgDataActivity$$ExternalSyntheticLambda0.onClick(Unknown Source:2)
at android.view.View.performClick(View.java:7125)
at android.view.View.performClickInternal(View.java:7102)
at android.view.View.access$3500(View.java:801)
at android.view.View$PerformClick.run(View.java:27336)
at android.os.Handler.handleCallback(Handler.java:883)
at android.os.Handler.dispatchMessage(Handler.java:100)
at android.os.Looper.loop(Looper.java:214)
at android.app.ActivityThread.main(ActivityThread.java:7356)
at java.lang.reflect.Method.invoke(Native Method)
at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:492)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:930)
2022-06-25 11:38:39.289 6145-6145/? E/m.openld.senio: Unknown bits set in runtime_flags: 0x8000
看最关键的点其实很清楚
Caused by: android.os.TransactionTooLargeException: data parcel size 1049112 bytes
就是说你的传输数据太大了,当前的大小达到了1049112 bytes。
那我们进到TransactionTooLargeException的定义中去看看
/*
* Copyright (C) 2011 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package android.os;
import android.os.RemoteException;
/**
* The Binder transaction failed because it was too large.
* <p>
* During a remote procedure call, the arguments and the return value of the call
* are transferred as {@link Parcel} objects stored in the Binder transaction buffer.
* If the arguments or the return value are too large to fit in the transaction buffer,
* then the call will fail. {@link TransactionTooLargeException} is thrown as a
* heuristic when a transaction is large, and it fails, since these are the transactions
* which are most likely to overfill the transaction buffer.
* </p><p>
* The Binder transaction buffer has a limited fixed size, currently 1MB, which
* is shared by all transactions in progress for the process. Consequently this
* exception can be thrown when there are many transactions in progress even when
* most of the individual transactions are of moderate size.
* </p><p>
* There are two possible outcomes when a remote procedure call throws
* {@link TransactionTooLargeException}. Either the client was unable to send
* its request to the service (most likely if the arguments were too large to fit in
* the transaction buffer), or the service was unable to send its response back
* to the client (most likely if the return value was too large to fit
* in the transaction buffer). It is not possible to tell which of these outcomes
* actually occurred. The client should assume that a partial failure occurred.
* </p><p>
* The key to avoiding {@link TransactionTooLargeException} is to keep all
* transactions relatively small. Try to minimize the amount of memory needed to create
* a {@link Parcel} for the arguments and the return value of the remote procedure call.
* Avoid transferring huge arrays of strings or large bitmaps.
* If possible, try to break up big requests into smaller pieces.
* </p><p>
* If you are implementing a service, it may help to impose size or complexity
* contraints on the queries that clients can perform. For example, if the result set
* could become large, then don't allow the client to request more than a few records
* at a time. Alternately, instead of returning all of the available data all at once,
* return the essential information first and make the client ask for additional information
* later as needed.
* </p>
*/
public class TransactionTooLargeException extends RemoteException {
public TransactionTooLargeException() {
super();
}
public TransactionTooLargeException(String msg) {
super(msg);
}
}
英语好的话看下注释就明白了了,就是Binder数据传输的时候数据太大了。
大意是说Binder传输依赖的是Bimder传输缓存,该缓存有1Mb的大小限制。而且该1Mb的缓存限制是被应用进程中的所有传输过程所共用的(言下之意是说你单个一次传输限制小于1Mb未必就不会出现上述的崩溃,别的传输过程也要占用部分空间的)。
因此不崩溃的度在哪里?
其实这里给的结论就是说你传输的时候尽量保证数据很小,当然正常情况下你用Intent传递数据的时候就是一些常用数据类型的key-value,本来就没多大,也不用担心触发该崩溃。
但是,当你在页面间利用Intent传输图片的时候呢?是不是就很容易因为传输了较大格式的图片而导致该崩溃呢。此外延伸一下,假如onSaveInstanceState()方法和onRestoreInstanceState()方法触发的时候里面也要有Bundle的,那些存储在其中的数据也是受到1Mb的限制的。
下面就给出解决办法,使用putBinder()方法传递大数据,此时使用的是共享内存而不是Binder传输缓存,因此不受1Mb的限制,但是使用该方法也有要注意的点。
看下注释。
/**
* Inserts an {@link IBinder} value into the mapping of this Bundle, replacing
* any existing value for the given key. Either key or value may be null.
*
* <p class="note">You should be very careful when using this function. In many
* places where Bundles are used (such as inside of Intent objects), the Bundle
* can live longer inside of another process than the process that had originally
* created it. In that case, the IBinder you supply here will become invalid
* when your process goes away, and no longer usable, even if a new process is
* created for you later on.</p>
*
* @param key a String, or null
* @param value an IBinder object, or null
*/
public void putBinder(@Nullable String key, @Nullable IBinder value) {
unparcel();
mMap.put(key, value);
}
意思是说一般情况在进程中利用Bundle传递数据时,Bundle在使用该数据的进程中存活的时间比创建该Bundle的进程中存活的时间要久。但是如果使用putBinder()方式传递数据时,数据在自定义的Binder中,创建Binder的进程一旦不存在,那Binder在使用它的进程中就会变为不可用。这一点在数据流转与取数据的时候一定要小心。
下面直接给出我测试的核心代码,通过对比第一个点击事件一定会崩,第二个点击事件跳到了结果页且结果页正确地拿到了大数据。
触发页面代码如下:
package com.openld.seniorstructure.main.testintentbigdata
import android.content.Intent
import android.os.Binder
import android.os.Bundle
import androidx.appcompat.app.AppCompatActivity
import androidx.appcompat.widget.AppCompatButton
import com.openld.seniorstructure.R
class TestIntentTransBIgDataActivity : AppCompatActivity() {
private lateinit var mBtnFail: AppCompatButton
private lateinit var mBtnSuccess: AppCompatButton
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_test_intent_trans_big_data)
initWidgets()
addListeners()
}
private fun initWidgets() {
mBtnFail = findViewById(R.id.btn_fail)
mBtnSuccess = findViewById(R.id.btn_success)
}
private fun addListeners() {
// 必崩溃
mBtnFail.setOnClickListener {
val intent = Intent(
this,
TestIntentTransBigDataResultActivity::class.java
)
val data = ByteArray(1024 * 1024)
val bundle = Bundle()
bundle.putByteArray("bigData", data)
intent.putExtra("bundle", bundle)
startActivity(intent)
}
// 可以传递大数据
mBtnSuccess.setOnClickListener {
val intent = Intent(
this,
TestIntentTransBigDataResultActivity::class.java
)
val data = ByteArray(1024 * 1024)
val bundle = Bundle()
val bigData = BigData(ByteArray(1024 * 1024))
bundle.putBinder("bigData", bigData)
intent.putExtra("bundle", bundle)
startActivity(intent)
}
}
}
data class BigData(val byteArray: ByteArray) : Binder() {
}
对应布局xml:
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".main.testintentbigdata.TestIntentTransBIgDataActivity">
<androidx.appcompat.widget.AppCompatButton
android:id="@+id/btn_fail"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_marginStart="8dp"
android:layout_marginTop="8dp"
android:layout_marginEnd="8dp"
android:background="@color/red"
android:text="使用常规方式传递大数据"
android:textAllCaps="false"
android:textColor="@color/white"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent"
tools:ignore="HardcodedText" />
<androidx.appcompat.widget.AppCompatButton
android:id="@+id/btn_success"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_marginStart="8dp"
android:layout_marginTop="8dp"
android:layout_marginEnd="8dp"
android:background="@color/red"
android:text="使用Binder方式传递大数据"
android:textAllCaps="false"
android:textColor="@color/white"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@+id/btn_fail"
tools:ignore="HardcodedText" />
</androidx.constraintlayout.widget.ConstraintLayout>
结果页代码如下:
package com.openld.seniorstructure.main.testintentbigdata
import android.os.Bundle
import android.widget.Toast
import androidx.appcompat.app.AppCompatActivity
import com.openld.seniorstructure.R
class TestIntentTransBigDataResultActivity : AppCompatActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_test_intent_trans_big_data_result)
val bundle = intent.getBundleExtra("bundle")
val ba = bundle?.getBinder("bigData") as BigData
Toast.makeText(this, "" + ba.byteArray.size / 1024, Toast.LENGTH_SHORT).show()
}
}
页面布局xml:
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".main.testintentbigdata.TestIntentTransBigDataResultActivity">
<androidx.appcompat.widget.AppCompatTextView
android:layout_width="0dp"
android:layout_height="0dp"
android:layout_margin="16dp"
android:autoSizeMaxTextSize="48sp"
android:autoSizeMinTextSize="28sp"
android:autoSizeTextType="uniform"
android:gravity="center"
android:lines="1"
android:text="Intent传递大数据成功了呀"
android:textColor="@color/red"
android:textSize="36sp"
android:textStyle="bold"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent"
tools:ignore="HardcodedText"
tools:targetApi="o" />
</androidx.constraintlayout.widget.ConstraintLayout>
最终大数据传递成功的结果截图
接到了1048576 Byte的数据