java-List
创始人
2024-05-16 23:36:14
0

java-List

  • 1. 预备知识-泛型(Generic)
    • 1.1 泛型的引入
    • 1.2 泛型的分类
    • 1.3 泛型类的定义的简单演示
    • 1.4 泛型背后作用时期和背后的简单原理
    • 1.5 泛型类的使用
    • 1.6 泛型总结
  • 2. 预备知识-包装类(Wrapper Class)
    • 2.1 基本数据类型和包装类直接的对应关系
    • 2.2 包装类的使用,装箱(boxing)和拆箱(unboxing)
    • 2.3 自动装箱(autoboxing)和自动拆箱(autounboxing)
    • 2.4 javap 反编译工具
  • 3. List 的使用
    • 3.1 常见方法
    • 3.2 示例
    • 3.3 练习-扑克牌

大家好,我是晓星航。今天为大家带来的是 java数据结构中List知识的讲解!😀

1. 预备知识-泛型(Generic)

1.1 泛型的引入

**问题:**我们之前实现过的顺序表,只能保存 int 类型的元素,如果现在需要保存 指向 Person 类型对象的引用的顺序表,请问应该如何解决?如果又需要保存指向 Book 对象类型的引用呢?

回答:

  1. 首先,我们在学习多态过程中已知一个前提,基类的引用可以指向子类的对象。
  2. 其次,我们也已知 Object 是 java 中所有类的祖先类。

那么,要解决上述问题,我们很自然的想到一个解决办法,将我们的顺序表的元素类型定义成 Object 类型,这样我们的 Object 类型的引用可以指向 Person 类型的对象或者指向 Book 类型的对象了。

示例代码:

public class MyArrayList {private Object[] array; // 保存顺序表的元素,即 Object 类型的引用private int size; // 保存顺序表内数据个数public void add(Object o) { 尾插 }public Object get(int index) {获取 index 位置的元素 }...
}

这样,我们可以就可以很自由的存储指向任意类型对象的引用到我们的顺序表了。

示例代码:

MyArrayList books = new MyArrayList();
for (int i = 0; i < 10; i++) {books.add(new Book()); // 尾插 10 本书到顺序表
}
MyArrayList people = new MyArrayList();
for (int i = 0; i < 10; i++) {people.add(new Person()); // 尾插 10 个人到顺序表
}

遗留问题:现在的 MyArrayList 虽然可以做到添加任意类型的引用到其中了,但遇到以下代码就会产生问题。

MyArrayList books = new MyArrayList();
books.add(new Book);
// 将 Object 类型转换为 Person 类型,需要类型转换才能成功
// 这里编译正确,但运行时会抛出异常 ClassCastException
Person person = (Person)books.get(0);

提示:问题暴露的越早,影响越小。编译期间的问题只会让开发者感觉到,运行期间的错误会让所有的软件使用者承受错误风险。

所以我们需要一种机制,可以 1. 增加编译期间的类型检查 2. 取消类型转换的使用 泛型就此诞生!

泛型的意义:

  1. 自动对类型进行检查
  2. 自动对类型进行强制类型的转换

1.2 泛型的分类

  1. 泛型类
  2. 泛型方法

1.3 泛型类的定义的简单演示

关于泛型类的定义,这里只是了解即可,我们重点学习泛型类的使用。

// 1. 尖括号 <> 是泛型的标志
// 2. E 是类型变量(Type Variable),变量名一般要大写
// 3. E 在定义时是形参,代表的意思是 MyArrayList 最终传入的类型,但现在还不知道public class MyArrayList {private E[] array;private int size;...}

注意: 泛型类可以一次有多个类型变量,用逗号分割。

1.4 泛型背后作用时期和背后的简单原理

  1. 泛型是作用在编译期间的一种机制,即运行期间没有泛型的概念。
  2. 泛型代码在运行期间,就是我们上面提到的,利用 Object 达到的效果(这里不是很准确,以后会做说明)。

1.5 泛型类的使用

// 定义了一个元素是 Book 引用的 MyArrayList
MyArrayList books = new MyArrayList();
books.add(new Book());// 会产生编译错误,Person 类型无法转换为 Book 类型
books.add(new Person());// 不需要做类型转换
Book book = book.get(0);// 不需要做类型转换
// 会产生编译错误,Book 类型无法转换为 Person 类型
Person person = book.get(0);

通过以上代码,我们可以看到泛型类的一个使用方式:只需要在所有类型后边跟尖括号,并且尖括号内是真正的类型,即 E 可以看作的最后的类型。

注意: Book 只能想象成 E 的类型,但实际上 E 的类型还是 Object。

1.6 泛型总结

  1. 泛型是为了解决某些容器、算法等代码的通用性而引入,并且能在编译期间做类型检查。
  2. 泛型利用的是 Object 是所有类的祖先类,并且父类的引用可以指向子类对象的特定而工作。
  3. 泛型是一种编译期间的机制,即 MyArrayList 和 MyArrayList

在运行期间是一个类型

  1. 泛型是 java 中的一种合法语法,标志就是尖括号 <>

2. 预备知识-包装类(Wrapper Class)

Object 引用可以指向任意类型的对象,但有例外出现了,8 种基本数据类型不是对象,那岂不是刚才的泛型机制要失效了?

实际上也确实如此,为了解决这个问题,java 引入了一类特殊的类,即这 8 种基本数据类型的包装类,在使用过程中,会将类似 int 这样的值包装到一个对象中去。

2.1 基本数据类型和包装类直接的对应关系

基本就是类型的首字母大写,除了 IntegerCharacter

例如我们将String(字符串)转为int(整形)类型:

String str = "123";
int ret = Integer.valueOf(str);
System.out.println(ret + 1);

2.2 包装类的使用,装箱(boxing)和拆箱(unboxing)

装箱、装包:把简单类型–>包装类类型

拆箱、拆包:把包装类类型–>简单数据类型

int i = 10;// 装箱操作,新建一个 Integer 类型对象,将 i 的值放入对象的某个属性中
Integer ii = Integer.valueOf(i);
Integer ij = new Integer(i);// 拆箱操作,将 Integer 对象中的值取出,放到一个基本数据类型中
int j = ii.intValue();

2.3 自动装箱(autoboxing)和自动拆箱(autounboxing)

可以看到在使用过程中,装箱和拆箱带来不少的代码量,所以为了减少开发者的负担,java 提供了自动机制。

int i = 10;Integer ii = i; // 自动装箱
Integer ij = (Integer)i; // 自动装箱int j = ii; // 自动拆箱
int k = (int)ii; // 自动拆箱

注意:自动装箱和自动拆箱是工作在编译期间的一种机制。

小问题-为什么Interger类型的a与b赋值为129时结果是false,但是a与b赋值为123时结果是true?

底层方法代码:

答:因为Integer value0f中如果范围-128——127时就调用high方法,运行结果就是true。如果范围不在-128——127时那么就会调用catch中的low方法,结果就为false

2.4 javap 反编译工具

这里我们刚好学习一个 jdk 中一个反编译工具来查看下自动装箱和自动拆箱过程,并且看到这个过程是发生在编译期间的。

javap -c类名称Compiled from "Main.java"public class Main {public Main();Code:0: aload_01: invokespecial #1 // Method java/lang/Object."":()V4: returnpublic static void main(java.lang.String[]);Code:0: bipush 102: istore_13: iload_14: invokestatic #2 // Method java/lang/Integer.valueOf:
(I)Ljava/lang/Integer;7: astore_28: iload_19: invokestatic #2 // Method java/lang/Integer.valueOf:
(I)Ljava/lang/Integer;12: astore_313: aload_214: invokevirtual #3 // Method java/lang/Integer.intValue:()I17: istore 419: aload_220: invokevirtual #3 // Method java/lang/Integer.intValue:()I23: istore 525: return
}

3. List 的使用

List的官方文档

ArrayList的官方文档

LinkedList的官方文档

3.1 常见方法

List(线性表):

上述图片方法举例可以参考下方超链接中的String中6.1–6.6的方法举例。

String方法博客

subList方法演示:

ArrayList list = new ArrayList<>();
list.add("hello");
list.add("xxh");
list.add("xyh");
list.add("lol");
list.add("coc");
System.out.println(list.subList(1, 3));
System.out.println(list);

这里subList截取了三个元素,但是只输出了两个元素是因为我们在使用list的方法中采取的是左闭右开的模式。

ArrayList(顺序表):

如果ArrayList调用,不带参数的构造方法,那么顺序表的大小是0.当第一次add的时候,整个顺序表才变为了10;当这10个放满了,开始扩容,以1.5倍的方式进行扩容。

如果调用的是给定容量的构造方法,那么你的顺序表大小,就是你给定的容量,放满了还是1.5倍扩充。

此时ArrayList链表的初始大小即为13。

LinkedList(链表):

3.2 示例

import java.util.List;
import java.util.ArrayList;
import java.util.LinkedList;public class TestDemo {public static void main(String[] args) {List courses = new ArrayList<>();courses.add("C 语言");courses.add("Java SE");courses.add("Java Web");courses.add("Java EE");
// 和数组一样,允许添加重复元素courses.add("C 语言");
// 按照添加顺序打印System.out.println(courses);
// 类似数组下标的方式访问System.out.println(courses.get(0));System.out.println(courses);courses.set(0, "计算机基础");System.out.println(courses);
// 截取部分 [1, 3)List subCourses = courses.subList(1, 3);System.out.println(subCourses);
// 重新构造List courses2 = new ArrayList<>(courses);System.out.println(courses2);List courses3 = new LinkedList<>(courses);System.out.println(courses3);
// 引用的转换ArrayList courses4 = (ArrayList)courses2;System.out.println(courses4);
// LinkedList c = (LinkedList)course2; 错误的类型LinkedList courses5 = (LinkedList)courses3;System.out.println(courses5);
// ArrayList c = (ArrayList)course3; 错误的类型}
}

运行结果:

3.3 练习-扑克牌

import java.util.List;
import java.util.ArrayList;
import java.util.Random;public class TestDemo {public class Card {public int rank; // 牌面值public String suit; // 花色@Overridepublic String toString() {return String.format("[%s %d]", suit, rank);}}public class CardDemo {public static final String[] SUITS = {"♠", "♥", "♣", "♦"};// 买一副牌private static List buyDeck() {List 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 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 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 deck = buyDeck();System.out.println("刚买回来的牌:");System.out.println(deck);shuffle(deck);System.out.println("洗过的牌:");System.out.println(deck);
// 三个人,每个人轮流抓 5 张牌List> 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));}}

运行结果:

3.4 面试题练习

杨辉三角

感谢各位读者的阅读,本文章有任何错误都可以在评论区发表你们的意见,我会对文章进行改正的。如果本文章对你有帮助请动一动你们敏捷的小手点一点赞,你的每一次鼓励都是作者创作的动力哦!😘

相关内容

热门资讯

安卓系统苹果手机识别,跨界融合... 你知道吗?在科技飞速发展的今天,手机已经成为了我们生活中不可或缺的一部分。而说到手机,安卓系统和苹果...
harmonyos系统是不是安... 亲爱的读者,你是否曾好奇过HarmonyOS系统与安卓系统之间的关系?是不是安卓的“亲戚”?今天,就...
手机怎么装系统安卓,安卓系统安... 手机卡顿了?想给安卓系统来个大变身?别急,跟着我一步步来,保证让你的手机焕然一新!一、准备工作在开始...
安卓Linux系统内网穿透,A... 你有没有想过,你的安卓手机里那些看似普通的APP,其实可能正在悄悄地帮你打通网络世界的任督二脉呢?没...
win怎么安装安卓系统,Win... 亲爱的读者,你是不是对Win系统上的安卓应用垂涎已久,但又苦于不知道如何安装安卓系统呢?别急,今天我...
升级小米平板安卓系统,畅享全新... 你有没有发现,你的小米平板用久了,是不是感觉有点卡呢?别急,今天就来教你怎么给它来个系统升级,让它焕...
捷豹安卓系统车载,捷豹安卓系统... 哇,你有没有想过,当你的手机和汽车融为一体,会是怎样的体验呢?想象你正驾驶着你的捷豹,车窗外的风景如...
安卓1到10系统,安卓1.0至... 你有没有想过,手机里的安卓系统就像是我们生活中的好朋友,从青涩的少年成长为稳重的青年呢?从安卓1.0...
安卓8.0停用系统应用,提升使... 你知道吗?最近安卓系统又来了一次大动作,那就是安卓8.0系统开始停用一些系统应用了。这可真是让人有点...
安卓系统修改mtu值,轻松提升... 你有没有想过,你的安卓手机其实是个小小的电脑呢?它里面藏着许多可以自定义的秘密功能,就像修改MTU值...
安卓平板改window系统,探... 你有没有想过,你的安卓平板其实可以摇身一变,变成一个Windows系统的电脑呢?没错,就是那种可以运...
时空猎人安卓苹果系统,探索无尽... 你知道吗?最近在手机游戏圈里,有一款叫做《时空猎人》的游戏可是火得一塌糊涂呢!不管是安卓用户还是苹果...
安卓9.0系统的电视,新一代电... 亲爱的读者们,你是否也像我一样,对科技新玩意儿充满好奇?今天,我要和你聊聊一个让人眼前一亮的话题——...
小pc安装安卓系统,轻松安装安... 你有没有想过,你的小PC也能变身成为安卓系统的超级玩家呢?没错,就是那个平时默默无闻的小家伙,现在也...
高通备份安卓系统,全方位数据安... 你知道吗?在这个科技飞速发展的时代,手机备份可是个不得不提的话题。尤其是对于安卓用户来说,选择一个靠...
谷歌安卓系统有多少,从诞生到全... 你有没有想过,那个无处不在的谷歌安卓系统,究竟在全球有多少用户呢?它就像一个神秘的数字,每天都在悄悄...
fc黄金传说安卓系统,畅享复古... 你有没有听说最近安卓系统上的一款超酷的游戏——《FC黄金传说》?这款游戏可是让不少玩家都沉迷其中,今...
变小的我安卓系统,安卓系统演变... 你有没有发现,最近你的手机好像变轻了?没错,说的就是你,那个陪伴你多年的安卓系统。它悄无声息地进行了...
vivo安卓系统小彩蛋,体验科... 你知道吗?在vivo的安卓系统中,竟然隐藏着一些超有趣的小彩蛋!这些小彩蛋就像是在手机里埋下的宝藏,...
安卓系统如何强制重启,安卓系统... 手机突然卡壳了,是不是又该给它来个“大保健”了?没错,今天就来聊聊安卓系统如何强制重启。别小看这个看...