赞
踩
目录
线性表是n个具有相同特性的数据元素的有限序列。线性表是一种在实际中广泛使用的数据结构,常见的线性表:顺序表、链表、栈、队列……
线性表在逻辑上时线性结构,也就是说是连续的一条直线,但是在物理结构上不一定是连续的,线性表在物理上存储时,通常以数组和链式结构的形式存储。
顺序表是用一段物理地址连续的存储单元依次存储数据元素的线性结构,一般情况下采用数组存储。在数组上完成数据的增删改查。
- public class SeqList {
- private int[] array;
- private int size;
- // 默认构造方法
- SeqList(){ }
- // 将顺序表的底层容量设置为initcapacity
- SeqList(int initcapacity){ }
- // 新增元素,默认在数组最后新增
- public void add(int data) { }
- // 在 pos 位置新增元素
- public void add(int pos, int data) { }
- // 判定是否包含某个元素
- public boolean contains(int toFind) { return true; }
- // 查找某个元素对应的位置
- public int indexOf(int toFind) { return -1; }
- // 获取 pos 位置的元素
- public int get(int pos) { return -1; }
- // 给 pos 位置的元素设为 value
- public void set(int pos, int value) { }
- //删除第一次出现的关键字key
- public void remove(int toRemove) { }
- // 获取顺序表长度
- public int size() { return 0; }
- // 清空顺序表
- public void clear() { }
- // 打印顺序表,注意:该方法并不是顺序表中的方法,为了方便看测试结果给出的
- public void display() { }
- }
在集合框架中,ArrayList是一个普通的类,实现了List接口。
【说明】:
方式 | 解释 |
---|---|
ArrayList() | 无参构造 |
ArrayList(Collection<?exrtends E>c) | 利用其他Collection构建ArrayList |
ArrayList(int initialCapacity) | 指定顺序表初始容量 |
- public static void main(String[] args) {
- // ArrayList创建,推荐写法
- // 构造一个空的列表
- List<Integer> list1 = new ArrayList<>();
- // 构造一个具有10个容量的列表
- List<Integer> list2 = new ArrayList<>(10);
- list2.add(1);
- list2.add(2);
- list2.add(3);
- // list2.add("hello"); // 编译失败,List<Integer>已经限定了,list2中只能存储整形元素
- // list3构造好之后,与list中的元素一致
- ArrayList<Integer> list3 = new ArrayList<>(list2);
- // 避免省略类型,否则:任意类型的元素都可以存放,使用时将是一场灾难
- List list4 = new ArrayList();
- list4.add("111");
- list4.add(100);
- }
方法 | 解释 |
---|---|
boolean add(E e) | 尾插e |
void add(int index, E element) | 将e插入到index位置 |
Boolean addAll(Collection<? extends E>c) | 尾插c中元素 |
E remove(int index) | 删除index位置 |
boolean remove(Object o) | 删除遇到的第一个o |
E get(int index) | 获取下标index位置元素 |
E set(int index, E element) | 将下标index位置元素设置为element |
void clear( ) | 清空 |
boolean contains (Object o) | 判断o是否在线性表中 |
int indexOf(Object o) | 返回第一个o所在下标 |
int lastIndexOf(Object o) | 返回最后一个o的下标 |
List<E>subList(int fromIndex, int toIndex) | 截取部分list |
- public static void main(String[] args) {
- List<String> list = new ArrayList<>();
- list.add("JavaSE");
- list.add("JavaWeb");
- list.add("JavaEE");
- list.add("JVM");
- list.add("测试课程");
- System.out.println(list);
- // 获取list中有效元素个数
- System.out.println(list.size());
- // 获取和设置index位置上的元素,注意index必须介于[0, size)间
- System.out.println(list.get(1));
- list.set(1, "JavaWEB");
- System.out.println(list.get(1));
- // 在list的index位置插入指定元素,index及后续的元素统一往后搬移一个位置
- list.add(1, "Java数据结构");
- System.out.println(list);
- // 删除指定元素,找到了就删除,该元素之后的元素统一往前搬移一个位置
- list.remove("JVM");
- System.out.println(list);
- // 删除list中index位置上的元素,注意index不要超过list中有效元素个数,否则会抛出下标越界异常
- list.remove(list.size()-1);
- System.out.println(list);
- // 检测list中是否包含指定元素,包含返回true,否则返回false
- if(list.contains("测试课程")){
- list.add("测试课程");
- }
- // 查找指定元素第一次出现的位置:indexOf从前往后找,lastIndexOf从后往前找
- list.add("JavaSE");
- System.out.println(list.indexOf("JavaSE"));
- System.out.println(list.lastIndexOf("JavaSE"));
- // 使用list中[0, 4)之间的元素构成一个新的SubList返回,但是和ArrayList共用一个elementData数组
- List<String> ret = list.subList(0, 4);
- System.out.println(ret);
- list.clear();
- System.out.println(list.size());
- }
ArrayList可以使用三种方式遍历:for循环+下标、foreach、使用迭代器
- public static void main(String[] args) {
- List<Integer> list = new ArrayList<>();
- list.add(1);
- list.add(2);
- list.add(3);
- list.add(4);
- list.add(5);
- // 使用下标+for遍历
- for (int i = 0; i < list.size(); i++) {
- System.out.print(list.get(i) + " ");
- }
- System.out.println();
- // 借助foreach遍历
- for (Integer integer : list) {
- System.out.print(integer + " ");
- }
- System.out.println();
- Iterator<Integer> it = list.listIterator();
- while(it.hasNext()){
- System.out.print(it.next() + " ");
- }
- System.out.println();
- }
注意:
这个代码有什么缺陷吗?为什么?
- public static void main(String[] args) {
- List<Integer> list = new ArrayList<>();
- for (int i = 0; i < 100; i++) {
- list.add(i);
- }
- }
ArrayList是一个动态类型的顺序表,即:在插入元素的过程中会自动扩容。以下是ArrayList源码中扩容方式:
- Object[] elementData; // 存放元素的空间
- private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {}; // 默认空间
- private static final int DEFAULT_CAPACITY = 10; // 默认容量大小
- public boolean add(E e) {
- ensureCapacityInternal(size + 1); // Increments modCount!!
- elementData[size++] = e;
- return true;
- }
- private void ensureCapacityInternal(int minCapacity) {
- ensureExplicitCapacity(calculateCapacity(elementData, minCapacity));
- }
- private static int calculateCapacity(Object[] elementData, int minCapacity) {
- if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
- return Math.max(DEFAULT_CAPACITY, minCapacity);
- }
- return minCapacity;
- }
- private void ensureExplicitCapacity(int minCapacity) {
- modCount++;
- // overflow-conscious code
- if (minCapacity - elementData.length > 0)
- grow(minCapacity);
- }
- private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8;
- private void grow(int minCapacity) {
- // 获取旧空间大小
- int oldCapacity = elementData.length;
- // 预计按照1.5倍方式扩容
- int newCapacity = oldCapacity + (oldCapacity >> 1);
- // 如果用户需要扩容大小 超过 原空间1.5倍,按照用户所需大小扩容
- if (newCapacity - minCapacity < 0)
- newCapacity = minCapacity;
- // 如果需要扩容大小超过MAX_ARRAY_SIZE,重新计算容量大小
- if (newCapacity - MAX_ARRAY_SIZE > 0)
- newCapacity = hugeCapacity(minCapacity);
- // 调用copyOf扩容
- elementData = Arrays.copyOf(elementData, newCapacity);
- }
- private static int hugeCapacity(int minCapacity) {
- // 如果minCapacity小于0,抛出OutOfMemoryError异常
- if (minCapacity < 0)
- throw new OutOfMemoryError();
- return (minCapacity > MAX_ARRAY_SIZE) ? Integer.MAX_VALUE : MAX_ARRAY_SIZE;
- }
【总结】
- public class Card {
- public int rank; // 牌面值
- public String suit; // 花色
- @Override
- public String toString() {
- return String.format("[%s %d]", suit, rank);
- }
- }
- import java.util.List;
- import java.util.ArrayList;
- import java.util.Random;
- public class CardDemo {
- public static final String[] SUITS = {"♠", "♥", "♣", "♦"};
- // 买一副牌
- private static List<Card> buyDeck() {
- List<Card> deck = new ArrayList<>(52);
- for (int i = 0; i < 4; i++) {
- for (int j = 1; j <= 13; j++) {
- String suit = SUITS[i];
- int rank = j;
- Card card = new Card();
- card.rank = rank;
- card.suit = suit;
- deck.add(card);
- }
- }
- return deck;
- }
- private static void swap(List<Card> deck, int i, int j) {
- Card t = deck.get(i);
- deck.set(i, deck.get(j));
- deck.set(j, t);
- }
- private static void shuffle(List<Card> deck) {
- Random random = new Random(20190905);
- for (int i = deck.size() - 1; i > 0; i--) {
- int r = random.nextInt(i);
- swap(deck, i, r);
- }
- }
- public static void main(String[] args) {
- List<Card> deck = buyDeck();
- System.out.println("刚买回来的牌:");
- System.out.println(deck);
- shuffle(deck);
- System.out.println("洗过的牌:");
- System.out.println(deck);
- // 三个人,每个人轮流抓 5 张牌
- List<List<Card>> hands = new ArrayList<>();
- hands.add(new ArrayList<>());
- hands.add(new ArrayList<>());
- hands.add(new ArrayList<>());
- for (int i = 0; i < 5; i++) {
- for (int j = 0; j < 3; j++) {
- hands.get(j).add(deck.remove(0));
- }
- }
- System.out.println("剩余的牌:");
- System.out.println(deck);
- System.out.println("A 手中的牌:");
- System.out.println(hands.get(0));
- System.out.println("B 手中的牌:");
- System.out.println(hands.get(1));
- System.out.println("C 手中的牌:");
- System.out.println(hands.get(2));
- }
- }
运行结果
- 刚买回来的牌:
- [[♠ 1], [♠ 2], [♠ 3], [♠ 4], [♠ 5], [♠ 6], [♠ 7], [♠ 8], [♠ 9], [♠ 10], [♠ 11], [♠ 12], [♠ 13], [♥ 1], [♥ 2], [♥ 3], [♥ 4], [♥ 5], [♥ 6], [♥ 7],
- [♥ 8], [♥ 9], [♥ 10], [♥ 11], [♥ 12], [♥ 13], [♣ 1], [♣ 2], [♣ 3], [♣ 4], [♣ 5], [♣ 6], [♣ 7], [♣ 8], [♣ 9], [♣ 10], [♣ 11], [♣ 12], [♣
- 13], [♦ 1], [♦ 2], [♦ 3], [♦ 4], [♦ 5], [♦ 6], [♦ 7], [♦ 8], [♦ 9], [♦ 10], [♦ 11], [♦ 12], [♦ 13]]
- 洗过的牌:
- [[♥ 11], [♥ 6], [♣ 13], [♣ 10], [♥ 13], [♠ 2], [♦ 1], [♥ 9], [♥ 12], [♦ 5], [♥ 8], [♠ 6], [♠ 3], [♥ 5], [♥ 1], [♦ 6], [♦ 13], [♣ 12], [♦ 12],
- [♣ 5], [♠ 4], [♣ 3], [♥ 7], [♦ 3], [♣ 2], [♠ 1], [♦ 2], [♥ 4], [♦ 8], [♠ 10], [♦ 11], [♥ 10], [♦ 7], [♣ 9], [♦ 4], [♣ 8], [♣ 7], [♠ 8], [♦ 9], [♠
- 12], [♠ 11], [♣ 11], [♦ 10], [♠ 5], [♠ 13], [♠ 9], [♠ 7], [♣ 6], [♣ 4], [♥ 2], [♣ 1], [♥ 3]]
- 剩余的牌:
- [[♦ 6], [♦ 13], [♣ 12], [♦ 12], [♣ 5], [♠ 4], [♣ 3], [♥ 7], [♦ 3], [♣ 2], [♠ 1], [♦ 2], [♥ 4], [♦ 8], [♠ 10], [♦ 11], [♥ 10], [♦ 7], [♣ 9], [♦
- 4], [♣ 8], [♣ 7], [♠ 8], [♦ 9], [♠ 12], [♠ 11], [♣ 11], [♦ 10], [♠ 5], [♠ 13], [♠ 9], [♠ 7], [♣ 6], [♣ 4], [♥ 2], [♣ 1], [♥ 3]]
- A 手中的牌:
- [[♥ 11], [♣ 10], [♦ 1], [♦ 5], [♠ 3]]
- B 手中的牌:
- [[♥ 6], [♥ 13], [♥ 9], [♥ 8], [♥ 5]]
- C 手中的牌:
- [[♣ 13], [♠ 2], [♥ 12], [♠ 6], [♥ 1]]
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。