java-List

news2025/1/12 9:37:29

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<E> {
            private E[] array;
            private int size;
            ...
        }

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

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

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

1.5 泛型类的使用

// 定义了一个元素是 Book 引用的 MyArrayList
MyArrayList<Book> books = new MyArrayList<Book>();
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_0
       1: invokespecial #1 // Method java/lang/Object."<init>":()V
       4: return
       
     public static void main(java.lang.String[]);
       Code:
       0: bipush 10
       2: istore_1
       3: iload_1
       4: invokestatic #2 // Method java/lang/Integer.valueOf:
(I)Ljava/lang/Integer;
       7: astore_2
       8: iload_1
       9: invokestatic #2 // Method java/lang/Integer.valueOf:
(I)Ljava/lang/Integer;
       12: astore_3
       13: aload_2
       14: invokevirtual #3 // Method java/lang/Integer.intValue:()I
       17: istore 4
       19: aload_2
       20: invokevirtual #3 // Method java/lang/Integer.intValue:()I
       23: istore 5
       25: return
}

3. List 的使用

List的官方文档

ArrayList的官方文档

LinkedList的官方文档

3.1 常见方法

List(线性表):

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

String方法博客

subList方法演示:

ArrayList<String> 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<String> 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<String> subCourses = courses.subList(1, 3);
        System.out.println(subCourses);
// 重新构造
        List<String> courses2 = new ArrayList<>(courses);
        System.out.println(courses2);
        List<String> courses3 = new LinkedList<>(courses);
        System.out.println(courses3);
// 引用的转换
        ArrayList<String> courses4 = (ArrayList<String>)courses2;
        System.out.println(courses4);
// LinkedList<String> c = (LinkedList<String>)course2; 错误的类型
        LinkedList<String> courses5 = (LinkedList<String>)courses3;
        System.out.println(courses5);
// ArrayList<String> c = (ArrayList<String>)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; // 花色
        @Override
        public String toString() {
            return String.format("[%s %d]", suit, rank);
        }
    }
    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));
        }
    }

运行结果:

3.4 面试题练习

杨辉三角

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/179241.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Windows压缩工具 “ Bandizip与7-zip ”(带你快速了解)

&#x1f4dc; “作者 久绊A” 专注记录自己所整理的Java、web、sql等&#xff0c;IT技术干货、学习经验、面试资料、刷题记录&#xff0c;以及遇到的问题和解决方案&#xff0c;记录自己成长的点滴。 &#x1f341; 操作系统【带你快速了解】对于电脑来说&#xff0c;如果说…

机器视觉在烟草领域中的应用

一个不知名大学生&#xff0c;江湖人称菜狗 original author: jacky Li Email : 3435673055qq.com Time of completion&#xff1a;2023.1.26 Last edited: 2023.1.26 目录 机器视觉在烟草领域中的应用 烟叶外观检测 烟末原料异物检测 叶梗烟丝缺陷检测 香烟过滤嘴外型检测…

(21)go-micro微服务logstash使用

文章目录一 Logstash介绍二 Logstash作用三 Logstash工作原理四 Logstash安装1.拉取镜像2.运行命令3.查看是否运行五 Logstash使用六 最后一 Logstash介绍 Logstash是具有实时流水线能力的开源的数据收集引擎。Logstash可以动态统一不同来源的数据&#xff0c;并将数据标准化到…

活动星投票自控好声音网络评选微信的投票方式线上免费投票

“自控好声音”网络评选投票_投票平台的陆续发展_小程序投票的好处与坏处_如何进行有效的图文投票近些年来&#xff0c;第三方的微信投票制作平台如雨后春笋般络绎不绝。随着手机的互联网的发展及微信开放平台各项基于手机能力的开放&#xff0c;更多人选择微信投票小程序平台&…

恶意代码分析实战 17 C++代码分析

17.1 Lab20-01 问题 在0x401040处的函数采用了什么参数&#xff1f; 首先&#xff0c;以①处的一个对new操作符的调用开始&#xff0c;这表明它正在创建一个对象。一个对象的引用会在EAX寄存器中返回最终存储在②处的var_8变量和③处的var_4变量中。var_4变量在④处被移到了E…

pytorch深度学习基础(十)——常用线性CNN模型的结构与训练

线性CNN模型的结构与训练引入包LeNet模型结构模型构建AlexNet模型结构模型构建VGG模型结构模型构建加载数据集累加器精度训练引入包 import torch from torch import nn from torchvision import datasets from torchvision import transforms from torch.utils.data import D…

基于蜣螂算法的极限学习机(ELM)分类算法-附代码

基于蜣螂算法的极限学习机(ELM)分类算法 文章目录基于蜣螂算法的极限学习机(ELM)分类算法1.极限学习机原理概述2.ELM学习算法3.分类问题4.基于蜣螂算法优化的ELM5.测试结果6.参考文献7.Matlab代码摘要&#xff1a;本文利用蜣螂算法对极限学习机进行优化&#xff0c;并用于分类问…

【华为上机真题】连续字母长度

&#x1f388; 作者&#xff1a;Linux猿 &#x1f388; 简介&#xff1a;CSDN博客专家&#x1f3c6;&#xff0c;华为云享专家&#x1f3c6;&#xff0c;Linux、C/C、云计算、物联网、面试、刷题、算法尽管咨询我&#xff0c;关注我&#xff0c;有问题私聊&#xff01; &…

C语言--指针初阶

目录什么是指针&#xff1f;指针变量指针类型指针类型的意义在数组中举例野指针概念野指针成因如何规避野指针指针运算指针-整数指针关系运算指针-指针应用(求字符串长度)结语什么是指针&#xff1f; 在计算机科学中&#xff0c;指针&#xff08;Pointer&#xff09;是编程语言…

直接在Notepad++中运行GO

1.Windows上安装Go语言开发包参考链接&#xff1a;http://c.biancheng.net/view/3992.html1.1.下载Go语言开发包可以在Go语言官网 &#xff08;https://golang.google.cn/dl/&#xff09; 下载Windows 系统下的Go语言开发包&#xff0c;如下图所示。这里我们下载的是64 位的开发…

深度学习:轻量级神经网络MobileNet 从v1 到v2

深度学习&#xff1a;轻量级神经网络MoblieNet 从v1 到 v2MobileNet V1前言深度可分离卷积传统卷积Depth Wise ConvPoint Wise Conv性能对比MobileNet V2前言主要改进Inverted Residuals BlockResidual BlockExpansion LayerReLU6Linear Activation Function小结实验MobileNet …

大年初二、初三—— 牛客网刷题经验分享~

2023年大年初二、初三 —— 牛客网刷题经验分享~&#x1f60e;大年初初二、初三 —— 牛客网刷题经验分享~&#x1f60e;)前言&#x1f64c;牛客网——基础语法【循环输出图形篇】&#x1f64c;BC98 线段图案 &#x1f64c;BC99 正方形图案 &#x1f64c;BC100 直角三角形图案 …

计算机毕业设计选题推荐之Springboot校园篮球足球竞赛预约平台-Vue

&#xff0c;本系统分为用户和管理员两个角色&#xff0c;其中用户可以在线注册登陆&#xff0c;查看平台公告&#xff0c;查看篮球比赛介绍&#xff0c;在线预约参加篮球比赛。管理员可以对用户信息&#xff0c;比赛项目&#xff0c;比赛分类&#xff0c;平台公告信息等进行管…

Linux中如何给普通用户提权

引言&#xff1a; 北京时间2023/1/26/11:00 &#xff0c;看到这个日期&#xff0c;我第一时间想到的是还有十几天就要开学啦&#xff01;开学我是向往的&#xff0c;但是我并不怎么向往开学的考试&#xff0c;比如什么毛概和什么信息技术&#xff0c;可能是我深知自己在这些课…

实现自己的数据库一

一 前言从上篇原创文章到现在又是新的一年&#xff0c;今天是2023年的大年初三&#xff0c;先在这里祝各位亲爱的老铁们新年快乐&#xff0c;身体健康&#xff0c;在新的一年里更帅气、更漂亮&#xff0c;都能完成自己的小目标。一直以来&#xff0c;我对数据存储还是比较感兴趣…

卓有成效的用例设计方法

持续坚持原创输出&#xff0c;点击蓝字关注我吧用例设计作为测试工程师的立身之本&#xff0c;是衡量测试工程师综合素质的重要参考&#xff0c;时间是测试工作中重要的测试资源&#xff0c;通过设计高质量的测试用例可以有效地提升测试效率。本文旨在介绍测试工作中常用的五种…

恶意代码分析实战 18 64位

18.1 Lab21-01 当你不带任何参数运行程序时会发生什么&#xff1f; 当你运行这个程序却没带任何参数&#xff0c;它会立即退出。 根据你使用的IDAPro的版本&#xff0c;main函数可能没有被自动识别&#xff0c;你如何识别对main函数的调用&#xff1f; main函数有三个参数入…

NodeJS 中 Express 之中间件

NodeJS 中 Express 之中间件参考描述中间件next()一个简单的中间件函数使用全局中间件局部中间件共享注意事项位置next()分类错误级中间件内置中间件express.urlencoded()express.json()第三方中间件参考 项目描述哔哩哔哩黑马程序员搜索引擎Bing 描述 项目描述Edge109.0.151…

【web前端】盒子模型

border 边框 content 内容 padding内边距 margin外边距 1.边框 border 边框粗细 用px作为单位 border-style &#xff1a; solid 实线的 dashed虚线的 dotted 点的 边框的符合写法&#xff1a; 那三个没有先后顺序 边框可以分开写 表格的细线边框 border-collapse …

【编程入门】开源记事本(微信小程序版)

背景 前面已输出多个系列&#xff1a; 《十余种编程语言做个计算器》 《十余种编程语言写2048小游戏》 《17种编程语言10种排序算法》 《十余种编程语言写博客系统》 《十余种编程语言写云笔记》 本系列对比云笔记&#xff0c;将更为简化&#xff0c;去掉了网络调用&#xff0…