textwatcher 上的 IndexOutOfBoundsException

IndexOutOfBoundsException on a textwatcher

我正在使用 textwatcher 检查用户输入的有效性。我的一些用户因此 textwacher 导致崩溃。这是 google 给出的堆栈跟踪:

java.lang.IndexOutOfBoundsException: Invalid index 1, size is 1

at java.util.ArrayList.throwIndexOutOfBoundsException(ArrayList.java:255)
at java.util.ArrayList.get(ArrayList.java:308)
at android.widget.TextView.sendAfterTextChanged(TextView.java:7997)
at android.widget.TextView$ChangeWatcher.afterTextChanged(TextView.java:10043)
at android.text.SpannableStringBuilder.sendAfterTextChanged(SpannableStringBuilder.java:970)
at android.text.SpannableStringBuilder.replace(SpannableStringBuilder.java:497)
at android.text.SpannableStringBuilder.replace(SpannableStringBuilder.java:435)
at android.text.SpannableStringBuilder.replace(SpannableStringBuilder.java:30)
at android.view.inputmethod.BaseInputConnection.replaceText(BaseInputConnection.java:679)
at android.view.inputmethod.BaseInputConnection.setComposingText(BaseInputConnection.java:437)
at com.android.internal.view.IInputConnectionWrapper.executeMessage(IInputConnectionWrapper.java:333)
at com.android.internal.view.IInputConnectionWrapper$MyHandler.handleMessage(IInputConnectionWrapper.java:77)
at android.os.Handler.dispatchMessage(Handler.java:102)
at android.os.Looper.loop(Looper.java:136)
at android.app.ActivityThread.main(ActivityThread.java:5584)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:515)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1268)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1084)
at dalvik.system.NativeStart.main(Native Method)

还有我的代码:

        Zipcode.addTextChangedListener(new TextWatcher() {
            @Override
            public void beforeTextChanged(CharSequence charSequence, int i, int i1, int i2) {
            }

            @Override
            public void onTextChanged(CharSequence charSequence, int i, int i1, int i2) {
            }

            @Override
            public void afterTextChanged(Editable editable) {
                if (editable.toString().equals(zc)) return;
                zc = editable.toString();
                isZipcodeChecked = false;
                Log.d(TAG, "Text changed : " + editable.toString());
                if (editable.toString().length() != 5) {
                    Zipcode.setTextColor(Color.RED);
                    return;
                }
                checkZipcode(editable.toString());
            }
        });

我没有更多关于这是如何发生的信息。大多数时候它工作正常,我无法重现这个错误。知道发生了什么事吗?

答案已经在评论中给出了,但很难看出你是否像我一样只是浏览。所以真正的答案是:

避免添加 TextWatcher 两次或更多次,因为第一次将编辑内容,随后的将导致异常。

由于无法检查 EditText 是否已设置 TextWatcher,因此我使用布尔值构建了自己的解决方案,用于检查是否已添加 TextWatcher。

private EditText mEditText;
private TextWatcher mTextWatcher;
private boolean mTextWatcherIsEnabled = false;

public void setTextWatcher(TextWatcher textWatcher) {
    mTextWatcher = textWatcher;
}

public void enableTextWatcher() {
    if (!mTextWatcherIsEnabled) {
        mEditText.addTextChangedListener(mTextWatcher);
    }
    mTextWatcherIsEnabled = true;
}

public void disableTextWatcher() {
    if (mTextWatcherIsEnabled) {
        mEditText.removeTextChangedListener(mTextWatcher);
    }
    mTextWatcherIsEnabled = false;
}

对我来说,这是因为我在同一个视图中添加了两个不同的 TextWatcher,而我首先添加的那个正在删除自己:

view.addTextChangedListener(new ListenerA());
view.addTextChangedListener(new ListenerB());
class ListenerA extends TextWatcher {

    @Override
    public void beforeTextChanged(CharSequence s, int i, int i1, int i2) {
        view.removeTextChangedListener(this);
        // ...
    }
}

因为第一个删除了自己,这里会导致索引越界异常:

private void sendBeforeTextChanged(CharSequence text, int start, int before, int after) {
    if (mListeners != null) {
        final ArrayList<TextWatcher> list = mListeners;
        final int count = list.size();
        for (int i = 0; i < count; i++) {
            list.get(i).beforeTextChanged(text, start, before, after);
        }
    }
    // ...
}

count 被查询,列表的大小为 2。所以它调用 get(0) 然后删除自己。然后它调用 get(1),但列表大小现在只有 1,因此抛出异常。

我解决这个问题的方法是一开始并没有真正删除我的 TextWatcher,而是只是禁用了该行为。您也可以通过保证只有列表中的最后一个观察者删除自己来解决这个问题。