为什么不能使用相同的Span对象上面两次setSpan?

Jud*_*udy 4 android

为什么不能使用相同的Span对象上面两次setSpan?

SpannableString ss = new SpannableString("aaaaa [1] bbbb [1] cccc [1]");

我需要用图像替换所有[1].如果我使用以下代码,则只有最后一个代码替换为图像:

etShow = (EditText) findViewById(R.id.show);
SpannableString ss = new SpannableString("aaaaa[1]bbbb[1]cccc[1]");
int[] starts = new int[3];
int[] ends = new int[3];
int h = 0;
int k = 0;
for (int i = 0; i < ss.length(); i++) {
    if (ss.charAt(i) == '[') {
    starts[h] = i;
    h++;
    } else if (ss.charAt(i) == ']') {
    ends[k] = i;
    k++;
    }
    }

Drawable d = getResources().getDrawable(R.drawable.ic_launcher);
        d.setBounds(0, 0, 50, 50);
        ImageSpan im = new ImageSpan(d);

for(int i=0;i<3;i++){
        ss.setSpan(im, starts[i], ends[i]+1, Spannable.SPAN_INCLUSIVE_EXCLUSIVE);   
        }
etShow.getText().insert(0, ss);
Run Code Online (Sandbox Code Playgroud)

If change to the following code, all the [1] are replaced by the image.

Drawable d = getResources().getDrawable(R.drawable.ic_launcher);
        d.setBounds(0, 0, 50, 50);
        ImageSpan im = new ImageSpan(d);
        ImageSpan im1 = new ImageSpan(d);
        ImageSpan im2 = new ImageSpan(d);
        //for(int i=0;i<3;i++){
//      ss.setSpan(im, starts[i], ends[i]+1, Spannable.SPAN_INCLUSIVE_EXCLUSIVE);
        ss.setSpan(im, starts[0], ends[0]+1, Spannable.SPAN_INCLUSIVE_EXCLUSIVE);
        ss.setSpan(im1, starts[1], ends[1]+1, Spannable.SPAN_INCLUSIVE_EXCLUSIVE);
        ss.setSpan(im2, starts[2], ends[2]+1, Spannable.SPAN_INCLUSIVE_EXCLUSIVE);
    //  }
Run Code Online (Sandbox Code Playgroud)

How can explaint this?

Com*_*are 10

我怀疑跨度对象HashMapSpanned表示内部作为a的键.因此,重用相同的span对象具有将其先前使用替换为新用途的效果.


小智 5

这几天读了Spannable源码,通过Google发现了这个问题,所以想贴一些源码来回答这个问题。

SpannableString通过SpannableStringInternal和setSpan方法实现如下。

/* package */ void setSpan(Object what, int start, int end, int flags) {
    ...
    int count = mSpanCount;
    Object[] spans = mSpans;
    int[] data = mSpanData;
    for (int i = 0; i < count; i++) {
        if (spans[i] == what) {
            int ostart = data[i * COLUMNS + START];
            int oend = data[i * COLUMNS + END];
            data[i * COLUMNS + START] = start;
            data[i * COLUMNS + END] = end;
            data[i * COLUMNS + FLAGS] = flags;
            sendSpanChanged(what, ostart, oend, nstart, nend);
            return;
        }
    }
    ...
}
Run Code Online (Sandbox Code Playgroud)

当您通过 setSpan 方法传递相同的跨度时,它将检查跨度数组是否具有相同的跨度,并用消息替换旧的开始值和结束值。