List subList()的一个demo

 

只要保证
toIndex不大于size
并且
fromIndex不大于toIndex即可
(谁会传一个负数的下标呢?)

    public List<E> subList(int fromIndex, int toIndex) {
        subListRangeCheck(fromIndex, toIndex, size);
        return new SubList(this, 0, fromIndex, toIndex);
    }

    static void subListRangeCheck(int fromIndex, int toIndex, int size) {
        if (fromIndex < 0)
            throw new IndexOutOfBoundsException("fromIndex = " + fromIndex);
        if (toIndex > size)
            throw new IndexOutOfBoundsException("toIndex = " + toIndex);
        if (fromIndex > toIndex)
            throw new IllegalArgumentException("fromIndex(" + fromIndex +
                                               ") > toIndex(" + toIndex + ")");
    }

 

 

    public static void main(String[] args) {
        ArrayList<String> list = new ArrayList<>();
        list.add("1");
        int size = list.size();
        System.out.println(size);
        System.out.println(list.subList(0, size));
        System.out.println(list.subList(size, size));
    }
1
[1]
[]

 

 

import org.junit.Test;

import java.util.ArrayList;
import java.util.List;

import static org.hamcrest.MatcherAssert.assertThat;
import static org.hamcrest.core.Is.is;

/**
 * Created by MyWorld on 2016/1/21.
 */
public class SubListDemoTest {

    @Test
    public void ListSizeShouldEquals1_WhenSourceList9Step5() {
        List<Integer> sourceList = new ArrayList<Integer>();
        for (int i = 0; i < 9; i++) {
            sourceList.add(i);
        }
        int step = 5;
        List<String> targetList = SubListDemo.split(sourceList, step);
        assertThat(targetList.size(), is(2));
    }

    @Test
    public void ListSizeShouldEquals2_WhenSourceList10Step5() {
        List<Integer> sourceList = new ArrayList<Integer>();
        for (int i = 0; i < 10; i++) {
            sourceList.add(i);
        }
        int step = 5;
        List<String> targetList = SubListDemo.split(sourceList, step);
        assertThat(targetList.size(), is(2));
    }

    @Test
    public void ListSizeShouldEquals3_WhenSourceList11Step5() {
        List<Integer> sourceList = new ArrayList<Integer>();
        for (int i = 0; i < 11; i++) {
            sourceList.add(i);
        }
        int step = 5;
        List<String> targetList = SubListDemo.split(sourceList, step);
        assertThat(targetList.size(), is(3));
    }

}

 

import java.util.ArrayList;
import java.util.List;

/**
 * Created by MyWorld on 2016/1/21.
 */
public class SubListDemo {
    public static List<String> split(List<Integer> list, int step) {
        List<String> listStr = new ArrayList<String>();
        int size = list.size();
        int times = size / step + (size % step == 0 ? 0 : 1);
        for (int i = 0; i < times; i++) {
            int fromIndex = i * step;
            int toIndex = (i + 1) * step;
            listStr.add(list.subList(fromIndex, (toIndex < size ? toIndex : size)).toString());
        }
        return listStr;
    }
}

 
tips:

java.util.List中有一个subList方法,用来返回一个list的一部分的视图。

List<E> subList(int fromIndex, int toIndex);

它返回原来list的从[fromIndex, toIndex)之间这一部分的视图,之所以说是视图,是因为实际上,返回的list是靠原来的list支持的。

所以,你对原来的list和返回的list做的“非结构性修改”(non-structural changes),都会影响到彼此对方。

所谓的“非结构性修改”,是指不涉及到list的大小改变的修改。相反,结构性修改,指改变了list大小的修改。

 

那么,如果涉及到结构性修改会怎么样呢?

如果发生结构性修改的是返回的子list,那么原来的list的大小也会发生变化;

而如果发生结构性修改的是原来的list(不包括由于返回的子list导致的改变),那么返回的子list语义上将会是undefined。在AbstractList(ArrayList的父类)中,undefined的具体表现形式是抛出一个ConcurrentModificationException。

因此,如果你在调用了sublist返回了子list之后,如果修改了原list的大小,那么之前产生的子list将会失效,变得不可使用。

 

tips: 如何删除一个list的某个区段,比如删除list的第2-5个元素?

方法是: 可以利用sublist的幕后还是原来的list的这个特性,比如

list.subList(from, to).clear();

这样就可以了。

 

示例代码:

public static void main(String[] args) {
        List<String> parentList = new ArrayList<String>();

        for(int i = 0; i < 5; i++){
            parentList.add(String.valueOf(i));
        }

        List<String> subList = parentList.subList(1, 3);
        for(String s : subList){
            System.out.println(s);//output: 1, 2
        }

        //non-structural modification by sublist, reflect parentList
        subList.set(0, "new 1");
        for(String s : parentList){
            System.out.println(s);//output: 0, new 1, 2, 3, 4
        }

        //structural modification by sublist, reflect parentList
        subList.add(String.valueOf(2.5));
        for(String s : parentList){
            System.out.println(s);//output:0, new 1, 2,    2.5, 3,    4
        }

        //non-structural modification by parentList, reflect sublist
        parentList.set(2, "new 2");
        for(String s : subList){
            System.out.println(s);//output: new 1, new 2
        }

        //structural modification by parentList, sublist becomes undefined(throw exception)
        parentList.add("undefine");
//        for(String s : subList){
//            System.out.println(s);
//        }
//        subList.get(0);
    }

一个很有趣的思考:如何最高效的实现一个list的split方法?

参见:http://stackoverflow.com/questions/379551/java-split-a-list-into-two-sub-lists

http://www.cnblogs.com/gaojing/archive/2012/06/17/java-list-sublist-caution.html

 

时间: 2024-09-22 13:36:26

List subList()的一个demo的相关文章

一个Demo让你掌握Android所有控件

原文:一个Demo让你掌握Android所有控件 本文是转载收藏,侵删,出处:"安卓巴士"       下面给出实现各个组件的源代码: 1.下拉框实现--Spinner [java] view plaincopyprint? package com.cellcom;      import java.util.ArrayList;   import java.util.List;      import android.app.Activity;   import android.os

请求网络数据-怎么请求聚合数据的数据,求一个demo,谢谢各位大神

问题描述 怎么请求聚合数据的数据,求一个demo,谢谢各位大神 我自己做的请求不出来数据,下面上一个类的代码.请大神指点指点 package com.action; import java.io.BufferedReader; import java.io.DataOutputStream; import java.io.IOException; import java.io.InputStream; import java.io.InputStreamReader; import java.i

cocos2d x-cocos如何将事件监听封装到自定义的精灵类中,求一个demo

问题描述 cocos如何将事件监听封装到自定义的精灵类中,求一个demo cocos如何将事件监听封装到自定义的精灵类中,求一个demo.每次创建新的精灵时就会添加触摸监听.在其他层中可以拖拽移动这些添加到层中的精灵 解决方案 把你的方法放进init()里面,继承layer或者node,create()后就会调用.具体的点击事件,你可以百度,很多 解决方案二: 把你的方法放进init()里面,继承layer或者node,create()后就会调用.具体的点击事件,你可以百度,很多 解决方案三:

javascript-深夜了还有前端的大神吗,我写了一个Demo但是没调明白,求大神帮看下?

问题描述 深夜了还有前端的大神吗,我写了一个Demo但是没调明白,求大神帮看下? <!DOCTYPE html> <html> <head> <meta charset="utf-8"/> <script src="http://libs.baidu.com/jquery/1.9.1/jquery.min.js"></script> <style type="text/css&q

谢谢-急需Java 供外部调用的api的一个demo实例

问题描述 急需Java 供外部调用的api的一个demo实例 写接口把json格式封装起来 用post请求的接口的方式 框架用springmvc 解决方案 参考: http://my.oschina.net/haopeng/blog/324934 解决方案二: 就是用springmvc返回一个json就可以

求一个Demo,用户在微信中访问页面时,页面输出用户的微信用户名

问题描述 高分求一个显示微信登录用户名的DEMO,最好能显示用户名和当前所在的地址,输出到页面上 解决方案 解决方案二:要完成这个功能不是只有简单的网页能实现的,这涉及到微信生态系统,是需要有公众号帐号配合的.解决方案三:额,公众号是有的...解决方案四:引用1楼tcmakebest的回复: 要完成这个功能不是只有简单的网页能实现的,这涉及到微信生态系统,是需要有公众号帐号配合的. 我是这样想的,所有东西配置全后,我其实是想得到对方微信的一个永久id,这个id作为我开发的网站的用户名,写入ses

求助IO流这块的一个demo里面的问题

问题描述 packagecom.lk.example;importjava.io.BufferedInputStream;importjava.io.BufferedOutputStream;importjava.io.File;importjava.io.FileInputStream;importjava.io.FileNotFoundException;importjava.io.FileOutputStream;importjava.io.IOException;importjava.i

Android调用Jni,非常简单的一个Demo

step1:创建一个android项目      Project Name:jnitest      Build Target: Android 1.6      Application Name: JNI Test      Package Name: org.eshock.jnitest      Create Activity: JNITest JNITest.java类代码: import android.app.Activity;   import android.os.Bundle;

给小弟一个demo学习吧-谁有九宫格lockpattern解锁的例码或者知道怎么用?

问题描述 谁有九宫格lockpattern解锁的例码或者知道怎么用? 请先看完我的需求 网上的九宫格都存在一个问题,三个点连接时可以跳过当中一个点直接连接头尾两端.这样有隐患不好. Github上有开源的,但是只有最新版本基于stadio平台的,我想要eclipse平台的. csdn上有个3.0的老版本,但始终用不了(或许是我用法不对,希望有大神指点一下) 总之希望有大神能发一份简单能运行的例码并包含相应库文件,或者能指点一下最新版本或3.0版本怎么用,先谢谢了 解决方案 google官方都快要