Java学习路线(13)——Collection集合类:List集合与Set集合

一、集合类体系结构
在这里插入图片描述


二、部分Collection类型对象
在这里插入图片描述
Collection集合特点

  • List系列集合是有序、可重复、有索引。

    • ArrayList:有序、可重复、有索引
    • LinkedList:有序、可重复、有索引
  • Set系列集合是无序、不重复、无索引。

    • HashSet:无序、不重复、无索引
    • LinkedHashSet:有序、不重复、无索引
    • TreeSet:按照大小默认升序、不重复、无索引

约束存储数据

  • 泛型约束: 集合是支持任意类型进行存储的。
  • 引用类型约束: 集合不支持基本类型,但支持引用数据类型。

三、Collection常用API

方法说明
boolean add(E e)添加指定数据并返回添加结果
void clear()清空数据
boolean isEmpty()判断数据为空
int size()获取集合大小
boolean contains(Object o)判断集合是否包含某个元素
boolean remove(E e)删除集合中某个元素默认删除第一个
Object[] toArray()集合转对象数组

拓展API

方法说明
Collection addAll(Collection c)合并

四、Collection遍历方式

1、迭代器(Iterator)

获取迭代器

方法说明
Iterator iterator()返回迭代器对象,默认指向0索引

Iterator中常用方法

方法说明
boolean hasNext()询问当前位置是否有元素
E next()获取当前元素,并向下移动,注意越界(NoSuchElementException)

示例

/*1、创建集合对象*/
Collection<String> collection = new ArrayList<>();
collection.add("我");
collection.add("爱");
collection.add("中");
collection.add("国");
collection.add("!");

/*2、获取迭代器*/
Iterator<String> integer = collection.iterator();

/*循环遍历迭代器*/
while(integer.hasNext()){	//判断有没有值
	System.out.print(integer.next());	//获取值并下移
}

/*打印结果*/
我爱中国!

2、foreach
foreach实际上是重写了迭代器进行遍历。

格式:

for(数据类型 element : 数组/集合){
	循环体
}

示例

/*1、创建集合对象*/
Collection<String> collection = new ArrayList<>();
collection.add("我");
collection.add("爱");
collection.add("中");
collection.add("国");
collection.add("!");

/*2、foreach遍历*/
for (String str : collection) {
	System.out.print(str);
}

/*打印结果*/
我爱中国!

3、lambda
Collection类提供了一个叫做 forEach() 的方法进行遍历,该方法采用的是匿名类进行重写。

 /*1、创建集合对象*/
Collection<String> collection = new ArrayList<>();
collection.add("我");
collection.add("爱");
collection.add("中");
collection.add("国");
collection.add("!");

/*2、forEach()遍历*/
//System.out::print 是将当前元素进行引用打印
collection.forEach(System.out::print);

/*打印结果*/
我爱中国!

五、常见数据结构

概念: 数据结构是计算机底层存储、组织数据的方式,合适的数据结构能提高运行效率和存储效率。

常见结构
1、栈
特点: 后进先出,先进后出。

2、队列
特点: 先进先出,后进后出
三种队列: 单进单出队列,双进单出队列,双进双出队列

3、数组
存储特点: 一块内存连续的空间。
操作特点:

  • 查询速度快: 通过地址值和索引定位。
  • 删除速率低: 需要移动大量元素。
  • 添加速率低: 需要移动大量元素。

4、链表
存储特点: 内存中不连续,每个节点包含数据和下一个元素地址。
操作特点:

  • 查询速度慢: 因为内存不连续,所以需要遍历每一个元素。
  • 删除速率高: 通过指针指向更变即可完成删除。
  • 添加相对速率高: 因为查询慢,所以速率较低,高效添加是通过指针指向更变即可完成删除。
    两种链表: 单向链表(数据+下一地址),双向链表(上一地址+数据+下一地址)

5、二叉树
存储特点: 内存不连续,每个节点包含父节点地址,左右节点地址,数据值。
结构特点:

  • 只能有一个根节点,每个节点最多支持2个直接子节点。
  • 节点的度: 节点拥有的子树个数,二叉树的度不大于2。
  • 叶子结点: 度为0的节点,也称为终端节点
  • 高度: 叶子结点高度为1,叶子结点的父节点高度为2,…,根节点高度最高。
  • 层: 根节点在第1层,以此类推。
  • 兄弟节点: 拥有共同父节点的节点互称兄弟节点。

特殊的二叉树: 二叉排序树。

6、平衡二叉树
原则: 任意节点的左右两个子树高度差不超过1,任意节点的左右两个子树都是平衡二叉树。

7、红黑树
概念: 一种自平衡的二叉排序树,1972年被称为平衡二叉B树,1978年更名为**“红黑树”**。
特点: 每一个节点可以是红或者黑,通过 “红黑规则” 进行平衡。
红黑规则

  • 每个节点是红色或黑色,根节点必须是黑色。
  • 如果节点没有子节点或父节点,则节点对应指针属性为Nil,这些Nil视为叶节点,叶节点是黑色。
  • 如果一个节点是红色,则子节点必须是黑色。
  • 每个节点,从该节点到其所有后代叶节点的简单路径上,均包含相同数目的黑色节点。

六、List集合
特有方法:

方法说明
void add(int index,E element指定位置插入元素
E remove(int index指定位置的元素,并返回删除元素
E set(int index,E element修改指定位置的元素,并返回修改元素
E get(int index获取指定位置的元素

List实现类的底层原理

1、ArrayList底层原理:

  • 底层基于数组实现,根据索引定位元素,增删需要做大量元素的移位操作。
  • 第一次创建集合并添加元素时,默认创建长度为10的数组。
  • 当数组满元素时,自动扩容二分之一。

2、LinkedList底层原理

  • 基于双链表实现,查询速度慢,增删首尾元素快(最适合实现栈与队列结构)。
  • 独有方法
方法说明
void addFirst(E e)首位插入指定元素
void addLast(E e)末位插入指定元素
E getFirst()获取首位元素
E getLast()获取末位元素
E removeFirst()删除首位元素
E removeLast()删除末位元素
E push()压栈/入栈
E pop()弹栈/出栈
E offerLast()入队
E offerFrist()出队

补充知识
1、集合的并发修改异常
问题的出现: 当我们从集合中查找某个元素并删除时,可能会出现并发修改异常。
问题出现的原因: 当我们在循环遍历时,因为索引改变而产生跳跃查询的情况。
解决方案:

  • 使用迭代器遍历,删除方法使用迭代器删除方法 iterator.remove(); 删除当前元素并不下移。(推荐使用)
  • 使用倒着for循环。

2、深入泛型

(1)泛型定义的位置

  • 类的后面。【class ArrayList】
  • 方法声明。【public void get(T t)】
  • 接口声明。【interface Collection】

(2)自定义泛型类

  • 格式: 【修饰符 class 类名<泛型变量> { … }】
public class MyArrayList<T>{ }
  • 泛型变量标识

常见标识:T、E、T、K、V

  • 作用: 编译阶段指定数据类型。

  • 原理: 出现泛型变量的地方全部替换成传输的真实数据类型。

  • 示例

class Main{
    public static void main(String[] args) {
        MyArrayList<String> mal = new MyArrayList<>();
        mal.add("String");
    }
}

public class MyArrayList<E> {
    public void add(E element){

    }
    public void remove(E element){

    }
}

(3)自定义泛型方法

  • 概念: 定义方法时同时定义泛型的方法。
  • 格式:
修饰符 <泛型变量> 返回值 方法名称(参数列表){ ... }
  • 作用: 方法中可以使用泛型接收一切实际类型的参数,方法更具备通用性。
  • 示例:
class Main{
    public static void main(String[] args) {
        String[] names = {"x1","x2","x3"};
        printArray(names);
        System.out.println("——————————");
        Integer[] ages = {10,11,12};
        printArray(ages);
    }

    public static <T> void printArray(T[] arr){
        StringBuilder builder = new StringBuilder("[");
        Iterator<T> iterator = Arrays.stream(arr).iterator();
        while(iterator.hasNext()){
            builder.append(iterator.next());
            builder.append(iterator.hasNext()?", ":"");
        }
        builder.append("]");
        System.out.println(builder.toString());
    }
}


/*输出结果*/
[x1, x2, x3]
——————————
[10, 11, 12]

(4)自定义泛型接口

  • 概念: 使用泛型定义的接口。
  • 格式:
修饰符 interface 接口名称<泛型变量>{ }
  • 作用: 泛型接口让实现类选择当前功能需要操作的数据类型。
  • 示例:
interface Info<E>{
    void add(E element);
    void removeByID(int id);
    E queryByID(int id);
    void update(int id,E element);
}

class Student implements Info<Student>{

    @Override
    public void add(Student element) {

    }

    @Override
    public void removeByID(int id) {

    }

    @Override
    public Student queryByID(int id) {
        return null;
    }

    @Override
    public void update(int id, Student element) {

    }

}

(5)通配符:?

  • 概念: ?可以在“使用泛型”时代表一切类型,而E,T,K,V在定义泛型时使用。
  • 泛型的上下限:
    【? extends Car】?必须是Car或者其子类,表示泛型上限
    【? super Car】?必须是Car或者其父类,表示泛型下限
  • 示例
  /**
	*使用场景:当我们需要传入多类型的引用集合时使用
	*/
  /*? extends Car使用*/
  /*1、创建Car类以及子类*/
  class Car{}
  class BMW extends Car{}
  class BENZ extends Car{}
  
  /*2、运行方法,这里的<? extends Car>表示引用类型是Car或是其子类*/
  public static void go(ArrayList<? extends Car> cars){}

  /*3、使用方法*/
  public static void main(String[] args){
	ArrayList<BMW> bmw = new ArrayList<>();
	bmw.add(new BMW());
	bmw.add(new BMW());
	bmw.add(new BMW());
	
	ArrayList<BENZ> benzs= new ArrayList<>();
	benzs.add(new BENZ());
	benzs.add(new BENZ());
	benzs.add(new BENZ());

	go(bmw);
	go(benzs);
  }

七、Set集合体系

体系特点

  • 无序: 存取顺序不一致
  • 不重复: 可以去重
  • 无索引: 没有带索引的方法

实现类特点

  • HashSet: 无序、不重复、无索引
  • LinkedHashSet: 有序、不重复、无索引
  • TreeSet: 排序、不重复、无索引

实现类详情
1、HashSet

(1)元素无序的底层原理:哈希表

hash表的概念: 是一种对于增删改查数据性能较好的结构。

hash表的组成: JDK8之前使用数组+链表组成;JDK8之后使用数组+链表+红黑树

JDK7版本的HashSet的底层原理解析:数组+链表+Hash算法

  • 1、优先创建一个默认长度为16,默认加载扩容因子为0.75的数组的数组,数组名table
  • 2、根据hash值跟数组长度求余计算存入位置(Hash算法)。
  • 3、当发生冲突时调用equals()比较。若已知则不存;若不一致则存储数组。
    • JDK7新元素占据老元素的位置,再将老元素挂载到新元素节点上。
    • JDK8新元素直接挂载到老元素节点上。
  • 4、当数组存满到16*0.75=12时,就自动扩容,每次扩容原先的两倍。

JDK8版本的HashSet原理解析:数组+链表+红黑树
两者区别在于由于当挂载链表长度超过8时,自动转换成红黑树。

(2)要理解hash表,必须先理解hash值。

hash值的概念: 是JDK根据对象的地址,按照某种自定义或预设规则计算的int类型数值。

获取hash值的方法: Object类API【int hashCode();】-> 返回对象的hash值

对象hash值特点

  • 同一个对象多次调用hashCode()返回相同的hash值。
  • 默认情况下,不同对象的hash值不同。

示例

/*1、创建对象获取hash值*/
String str = "石头人";
System.out.println(str.hashCode());
String integerV = "李雷";
System.out.println(integerV.hashCode());

/*打印结果*/
30237497
858473

(3)HashSet去重底层原理: Hash算法一致节点挂载

注意: 当我们需要去重对象集合时,需要在对象中重写hashCode()以及equals()

重写的hashCode()和equals


@Override
public boolean equals(Object o){
	if(this == o) return true; //判断是否为当前对象
	if(o == null || getClass() != o.getClass()) return true; //判断空值以及对象类型是否相同。
	类型 变量名 = (类型)o;
	return 属性比较 && Objects.equals(xx,o.xx)}

@Override
public int hashCode(){
	return Objects.hash(你比较的参数列表);
}

2、LinkedHashSet
(1)特点:有序不重复无索引
(2)有序特性:保证存储和取出的元素顺序一致。
(3)存储结构:Hash表
(4)有序原理:数据依次进入,每一个数据都会与前一个数据建立双向链表机制记录顺序。

3、TreeSet集合
(1)特点:可排序不重复无索引
(2)可排序特性:按照元素大小默认升序排列。
(3)存储结构:红黑树
(4)默认排序规则:

  • 数值类型: Integer,Double,官方默认按照大小升序排序。
  • 字符串类型: 默认按照首字符的ASCII码升序排序。
  • 自定义类型: 采用自定义排序规则。

(5)示例

/*数值类型排序*/
Set<Integer> integerSet = new TreeSet<>();
integerSet.add(3);
integerSet.add(1);
integerSet.add(6);
integerSet.add(8);
integerSet.add(0);
integerSet.add(2);
System.out.println(integerSet);
        
/*字符串类型排序*/
Set<String> stringSet = new TreeSet<>();
stringSet.add("Java");
stringSet.add("MySQL");
stringSet.add("PHP");
stringSet.add("C++");
stringSet.add("Golang");
stringSet.add("JavaScript");
System.out.println(stringSet);
        
/*打印结果*/
[0, 1, 2, 3, 6, 8]
[C++, Golang, Java, JavaScript, MySQL, PHP]

/*自定义类型排序*/
//方法一:Comparable接口重写compareTo()
public class AppleTemp {
    public static void main(String[] args) {
        Set<Apple> set = new TreeSet<>();
        set.add(new Apple("绿苹果",82.2,50));
        set.add(new Apple("黄苹果",54.1,20));
        set.add(new Apple("金苹果",10.2,11));
        set.add(new Apple("青苹果",100.0,1));
        System.out.println(set);
    }

}

class Apple implements Comparable<Apple>{

    private String name;
    private double weight;
    private int count;

    public Apple() {
    }

    public Apple(String name, double weight, int count) {
        this.name = name;
        this.weight = weight;
        this.count = count;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public double getWeight() {
        return weight;
    }

    public void setWeight(double weight) {
        this.weight = weight;
    }

    public int getCount() {
        return count;
    }

    public void setCount(int count) {
        this.count = count;
    }

	@Override
    public int compareTo(Apple o) {
        return this.getName().compareTo(o.getName());
    }

    @Override
    public String toString() {
        return "Apple{" +
                "name='" + name + '\'' +
                ", weight=" + weight +
                ", count=" + count +
                '}';
    }
}

//方法二:TreeSet集合有参构造器,设置Comparator接口对应的比较器对象。(优先使用)
public class AppleTemp {
    public static void main(String[] args) {
        Set<Apple> set = new TreeSet<>(Comparator.comparing(Apple::getName));
        set.add(new Apple("绿苹果",82.2,50));
        set.add(new Apple("黄苹果",54.1,20));
        set.add(new Apple("金苹果",10.2,11));
        set.add(new Apple("青苹果",100.0,1));
        System.out.println(set);
    }
}

class Apple{

    private String name;
    private double weight;
    private int count;

    public Apple() {
    }

    public Apple(String name, double weight, int count) {
        this.name = name;
        this.weight = weight;
        this.count = count;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public double getWeight() {
        return weight;
    }

    public void setWeight(double weight) {
        this.weight = weight;
    }

    public int getCount() {
        return count;
    }

    public void setCount(int count) {
        this.count = count;
    }

    @Override
    public String toString() {
        return "Apple{" +
                "name='" + name + '\'' +
                ", weight=" + weight +
                ", count=" + count +
                '}';
    }
}

/*打印结果*/
[Apple{name='绿苹果', weight=82.2, count=50}, Apple{name='金苹果', weight=10.2, count=11}, Apple{name='青苹果', weight=100.0, count=1}, Apple{name='黄苹果', weight=54.1, count=20}]

补充知识
1、可变参数

概念: 可变参数用在形参中接收多个数据。

格式:

/*数据类型... 参数名称*/
public static void sum(int... num);

作用:

  • 1、传输参数灵活方便。可以不传参,也可以传输1个或者多个,也可以传输一个数组。
  • 2、可变参数在方法内部按照数组进行使用。

注意事项

  • 1、 一个形参列表中可变参数只能有一个。
  • 2、可变参数必须放在形参列表的最后。

2、Collections操作类

概念: java.utils.Collections是集合工具类

常用API

方法说明
static boolean addAll(Collection<? super T> c,T… elements)批量添加元素
static void shuffle(List<?> list)打乱集合顺序
static void sort(List<?> list)集合排序
static void sort(List list,Comparator<? super T> c)按指定规则排序

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

下载YouTube视频的一种方法

文章目录 工具名称下载方法使用方法1.只下载音频2.下载音频转换成mp3&#xff08;加上-x –audio-format参数&#xff09;3.下载视频&#xff08;带音频&#xff09;ID&#xff1a;22 | EXT&#xff1a;mp4 | 1280*720 下载的数据集&#xff1a;YouCook2 工具名称 yt-dlp 下载…

ajax使用

说明&#xff1a;ajax是一门异步处理请求的技术&#xff1b;可以实现不重新加载整个页面的情况下&#xff0c;访问后台后服务&#xff1b;比如百度搜索引擎&#xff0c;输入关键字后&#xff0c;下面会实时出现待选项&#xff0c;这就是一种异步处理请求的技术。 原生Ajax 原…

chatgpt赋能python:Python中未定义变量的默认值

Python中未定义变量的默认值 在Python编程中&#xff0c;有时候我们会使用未经定义的变量。如果这些变量没有被定义&#xff0c;那么它们将没有任何值。在这篇文章中&#xff0c;我们将讨论Python中未定义变量默认值的问题&#xff0c;并深入研究为什么这些默认值如此重要。 …

如何保证三个线程按顺序执行?不会我教你

&#x1f468;‍&#x1f393;作者&#xff1a;bug菌 ✏️博客&#xff1a;CSDN、掘金、infoQ、51CTO等 &#x1f389;简介&#xff1a;CSDN|阿里云|华为云|51CTO等社区博客专家&#xff0c;历届博客之星Top30&#xff0c;掘金年度人气作者Top40&#xff0c;51CTO年度博主Top12…

linux内核内存管理slab

一、概述 linux内存管理核心是伙伴系统&#xff0c;slab&#xff0c;slub&#xff0c;slob是基于伙伴系统之上提供api&#xff0c;用于内核内存分配释放管理&#xff0c;适用于小内存&#xff08;小于&#xff11;页&#xff09;分配与释放&#xff0c;当然大于&#xff11;页…

基于html+css的图展示99

准备项目 项目开发工具 Visual Studio Code 1.44.2 版本: 1.44.2 提交: ff915844119ce9485abfe8aa9076ec76b5300ddd 日期: 2020-04-16T16:36:23.138Z Electron: 7.1.11 Chrome: 78.0.3904.130 Node.js: 12.8.1 V8: 7.8.279.23-electron.0 OS: Windows_NT x64 10.0.19044 项目…

运动员最佳配对问题——算法设计与分析(C实现)

目录 一、问题简述 二、分析 三、代码展示 四、结果验证 一、问题简述 问题描述&#xff1a;羽毛球队有男女运动员各n人。给定2个n*n矩阵P和Q。P[i][j]是男运动员i和女运动员j配对组成混合双打的男运动员竞争优势&#xff1b;Q[i][j]是男运动员i和女运动员j配合的女运动员竞…

SSM框架学习-拦截器

1. 简介 在Spring框架中&#xff0c;拦截器是一种很重要的组件&#xff0c;它们允许在请求到达控制器之前或之后执行一些代码。拦截器在请求处理的特定点进行拦截&#xff0c;然后通过逻辑来决定是否将控制器的处理传递给下一个处理程序。 在Spring中&#xff0c;拦截器是由实现…

KVM虚拟化技术学习-基础入门

1.虚拟化技术概述 虚拟化[Virtualization]技术最早出现在 20 世纪 60 年代的 IBM ⼤型机系统&#xff0c;在70年代的 System 370 系列中逐渐流⾏起来&#xff0c;这些机器通过⼀种叫虚拟机监控器[Virtual Machine Monitor&#xff0c;VMM]的程序 在物理硬件之上⽣成许多可以运⾏…

Codeforces Round 764 (Div. 3)

比赛链接 Codeforces Round 764 A. Plus One on the SubsetB. Make APC. Division by Two and PermutationD. Palindromes ColoringE. Masha-forgetful A. Plus One on the Subset Example input 3 6 3 4 2 4 1 2 3 1000 1002 998 2 12 11output 3 4 1题意&#xff1a; 你可…

计算机网络考试多选题汇总Ⅱ

https://cadyin.blog.csdn.nethttps://blog.csdn.net/qq_38639612?spm1010.2135.3001.5421 计算机网络考试多选题汇总 1、在Windows中&#xff0c;任务管理器的作用是() A&#xff0e;终止未响应的应用程序 B&#xff0e;终止进程的运行 C&#xff0e;查看系统当前的信息 …

车载网络测试 - CANCANFD - 基础篇_01

目录 问题思考&#xff1a; 一、为什么需要总线? 二、什么是CAN总线? 三、为什么是CAN总线? 四、曾经的车用总线 1、SAEJ1850(Class2) 2、SAEJ1708 3、K-Line 4、BEAN 5、 byteflight, K-Bus 6、D2B 五、当前的车用总线 1、CAN 2、LIN 3、FlexRay 4、MOST 六…

python-sqlite3使用指南

python下sqlite3使用指南 文章目录 python下sqlite3使用指南开发环境sqlite3常用APICRUD实例参考 开发环境 vscode ​ 开发语言&#xff1a; python vscode SQLite插件使用方法&#xff1a; 之后在这里就可以发现可视化数据&#xff1a; sqlite3常用API Python 2.5.x 以上…

E往无前 | 腾讯云大数据 ElasticSearch 高级功能:Cross Cluster Replication实战

前言 Elasticsearch在platinum版本中&#xff0c;推出了Cross Cluster Replication特性&#xff08;以下简称CCR&#xff09;&#xff0c;也即跨集群远程复制。 该特性可以解决两类问题&#xff1a; 1&#xff0c;数据迁移&#xff1b; 2&#xff0c;异地备份。 本文以实战为主…

微服务和领域驱动

一、微服务 1.1 什么是微服务 微服务就是一些协同工作的小而自治的服务。 关键词&#xff1a; 小而自治 -- 小 “小”这个概念&#xff0c;一方面体现在微服务的内聚性上。 内聚性也可以称之为单一职责原则&#xff1a;“把因相同原因而变化的东西聚合到一起&#xff0c;…

企业电子招投标采购系统源码之登录页面-java spring cloud

​ 信息数智化招采系统 服务框架&#xff1a;Spring Cloud、Spring Boot2、Mybatis、OAuth2、Security 前端架构&#xff1a;VUE、Uniapp、Layui、Bootstrap、H5、CSS3 涉及技术&#xff1a;Eureka、Config、Zuul、OAuth2、Security、OSS、Turbine、Zipkin、Feign、Monitor、…

202312读书笔记|《赶时间的人》——灰暗的从前会成为照亮未来的光,艰难的生活里,诗歌是那陡峭的另一面

202312读书笔记|《赶时间的人》——灰暗的从前会成为照亮未来的光&#xff0c;艰难的生活里&#xff0c;诗歌是那陡峭的另一面 《赶时间的人》 作者王计兵&#xff0c;一个外卖员的诗&#xff0c;饱含对生活的热情&#xff0c;向上的力量&#xff0c;仿若身在炼狱&#xff0c;心…

【计算机网络】3、IO 多路复用:select、poll、epoll、reactor | 阻塞非阻塞、同步异步

文章目录 一、select()1.1 用法1.1 实战 二、poll()2.1 用法2.2 实战 三、阻塞、非阻塞3.1 非阻塞 IO3.1.1 read()3.1.2 write()3.1.3 accept()3.1.4 connect()3.1.5 非阻塞IO select() 多路复用实战 四、epoll()4.1 epoll_create()4.2 epoll_ctl()4.3 epoll_wait()4.4 实战4.…

Dubbo源码篇07---SPI神秘的面纱---原理篇---下

Dubbo源码篇07---SPI神秘的面纱---原理篇---下 引言根据name获取扩展实例对象获取默认扩展实例对象按条件批量获取扩展实例对象实例演示 小结 引言 上篇文章&#xff1a; Dubbo源码篇06—SPI神秘的面纱—原理篇—上 我们追踪了getAdaptiveExtension获取自适应扩展点的整个流程…

(常见)数据模型

文章目录 数据模型概述一、数据模型概要1.模型、建模与抽象2.数据模型3.两类数据模型 二、数据库模型的组成要素1.数据结构2.数据操作3.数据的完整性约束 三、概念模型1.概要2.基本概念3.概念模型的表示方法 常用数据模型一、层次模型1.简介2.数据结构3.数据操纵与完整性约束4.…