14.集合、常见的数据结构

集合

概念

Java中的集合就是一个容器,用来存放Java对象。

集合在存放对象的时候,不同的容器,存放的方法实现是不一样的,

Java中将这些不同实现的容器,往上抽取就形成了Java的集合体系。

Java集合中的根接口:Collection 和 Map

Collection是一个单值集合,每次添加只能存放一个值进去

Map是一个双值集合,每次添加可以添加一对数据到Map

Collection介绍

1.由来

Java是一个面向对象语言,将来肯定会处理对象。

为了方便操作多个对象,就需要把这些对象存储起来,存储多个对象,就需要一个容器

Java之前提供了数组、StringBuffer这两个容器,但是他们用来存放对象不是很方便

所以,Java就提供了Collection集合。

2.数组和集合的区别

长度区别:

数组的长度是固定的

集合的长度可变的

内容不同:

数组存储的是同一种数据类型

集合可以存储不同类型的元素

元素的数据类型不同:

数组可以存放基本类型、引用类型

集合只能存放引用类型,如果存放int类型,默认会被提升为Integer

3.Collection的发展

集合可以存放多个元素,但是在存放的时候也会有不同的需求,

比如:多个元素不能重复、多个元素按照规则排序...

根据不同的需求,Java中提供了很多的集合类

每个集合根据需求,使用不同的数据结构,不管什么数据结构,他们都有一些共同的功能,

比如可以添加、删除、判断...

所以Collection其实就是把不同集合类的共同内容不断往上抽取,

最终,形成了集合的继承体系。

Collection中子接口、子类比较多,只需要掌握常用的一些集合类

Collection

--Set

--HashSet

--TreeSet

--LinkedHashSet

--List

--ArrayList

--LinkedList

4.Collection的功能

(1)添加

boolean add(E e)

确保此集合包含指定的元素(可选操作)。

boolean addAll(Collection<? extends E> c)

将指定集合中的所有元素添加到此集合(可选操作)。

(2)获取

Iterator<E> iterator()

返回此集合中的元素的迭代器。

(3)判断

boolean contains(Object o)

如果此集合包含指定的元素,则返回 true 。

boolean containsAll(Collection<?> c)

如果此集合包含指定 集合中的所有元素,则返回true。

boolean isEmpty()

如果此集合不包含元素,则返回 true 。

(4)大小

int size()

返回此集合中的元素数。

(5)移除

void clear()

从此集合中删除所有元素(可选操作)。

boolean remove(Object o)

从该集合中删除指定元素的单个实例(如果存在)(可选操作)。

boolean removeAll(Collection<?> c)

删除指定集合中包含的所有此集合的元素(可选操作)。

default boolean removeIf(Predicate<? super E> filter)

删除满足给定谓词的此集合的所有元素。

package com.day14.collet;import java.util.ArrayList;
import java.util.Collection;
import java.util.Iterator;public class CollectionDemo01 {public static void main(String[] args) {//多态的方式创建Collection对象Collection c = new ArrayList();//调用方法//添加c.add(123);c.add("hello");c.add(new Person("jack"));c.add(2.58);c.add(null);c.add("hello");System.out.println(c);//判断包含boolean b1 = c.contains("hello");System.out.println(b1);//判断是否为空boolean b2 = c.isEmpty();System.out.println(b2);//大小System.out.println(c.size());//        //移除//        c.remove("hello");//移除指定内容//        System.out.println(c);////        c.clear();//        System.out.println(c);//清除全部System.out.println("---------------------");//遍历 获取ArrayList arrayList =(ArrayList) c;for (int i = 0; i < arrayList.size(); i++) {System.out.println(arrayList.get(i));}System.out.println("---------------------");//增强forfor (Object o : c){System.out.println(o);}System.out.println("---------------------");//iteratorIterator iterator = c.iterator();while (iterator.hasNext()){System.out.println(iterator.next());}//创建一个指定存储类型的集合//指定之后,这个集合中只能存放StringCollection<String> c1 = new ArrayList();c1.add("hello");//c1.add(123);c1.add("world");c1.add("world");c1.add("java");System.out.println("---------------------");//遍历for (String s : c1) {System.out.println(s);}System.out.println("---------------------");//迭代器遍历Iterator<String> iterator1 = c1.iterator();while (iterator1.hasNext()){System.out.println(iterator1.next());}}
}

Collection中迭代器原理 iterator()

1,iterator() 方法,继承自Iterable接口,这个接口主要用来规定对于迭代获取的规则接口

2, Iterator<T> iterator(); 这个方法的返回值是一个 Iterator对象

Iterator也是一个接口,提供了两个抽象方法,分别是hasNext() 和 Next()方法

3,当我们使用多态创建Collection对象,

Collection<String> c1 = new ArrayList();

然后使用c1调用iterator()方法做迭代的时候,实际上调用的是ArrayList()中重写的iterator()方法

ArrayList的iterator()方法,重写之后,返回了 new Itr(); 也就Itr是一个类,并且这个类一定是Iterator接口的实现类

往下看,发现,在ArrayList内部声明了一个内部类,Itr 并且实现了Iterator接口,重写了 hasNext() 和 Next()方法

hasNext()方法是在判断,cursor是否移动最后了

next()方法,是在以数组下标的方式取出元素,并返回,同时给cursor做增加(往后移动)

常见的数据结构

将来,不同的集合

栈结构:先进后出,入口和出口在同一侧

队列结构:先进先出,入口和出口不在同一侧

数组结构:存放元素通过下标来存放,获取元素通过下标获取

查询快:数组中的元素存放是连续的,通过数组的下标快速找到指定的元素

增删慢:数组的长度是固定的,增删的时候,其实是在做数组复制

数组做增删操作:

1.先创建一个新数组,然后将老数组的内容复制到新数组中

2.再往新数组中添加内容

3.删除则是往新数组中复制指定的内容

链表结构:链表中的每个元素,称为一个节点,就是一个类 Node类

一个节点中,包含一个数据区,还有两个指针区,就是类的属性

特点:

查询慢:查询元素需要一个个往下遍历获取,需要遍历的次数比较多

增删快:增删元素的时候,只需要修改节点中的引用地址就行了

单向链表:节点中,只有一个地址,指向下个地址

双向链表:节点中,有两个地址,一个指向上一个节点,一个指向下一个节点,

LinkedList使用了双向链表

package com.day14.jiegou;public class Node {Node pri;Object data;Node next;public Node(Object data, Node addr) {this.data = data;this.next = addr;}public Node(Node pri, Object data, Node next) {this.pri = pri;this.data = data;this.next = next;}public void add(Object o){//第一次添加Node node = new Node(null,"hello",null );//第二次添加Node node1 = new Node(node,"java", null);node.next = node1;//第二次添加Node node2 = new Node(node1,"oracle", null);node1.next = node2;}}

树结构:有多个节点组成,具有层次关系的一种结构。

特点:

因为组成的结构,看起来像一棵倒挂的树,也就是根朝上,叶子朝下,称为树结构。

每个节点都有零个或多个子节点,没有父节点的节点成为根节点, 每个非根节点,有且仅有一个父节点,除了根节点之外,每个子结点都可以分为多个不相交的子树。将来应用中,主要使用的是二叉树。

二叉树特点:添加、删除元素都很快,查找也有很多算法优化,所以二叉树既有集合的好处,也有数组的好处,是两者的优化方案,处理大批量的动态数据应用很多。

二叉树扩展:平衡二叉树、红黑树、B+树..

B+树是MySQL底层索引结构

红黑树是HashMap的底层结构

红黑树特点:查询很快,查询叶子节点的最大次数和最小次数不超过2倍。节点是红色或者黑色,根节点是黑色,叶子节点是黑色,每个红色节点的子节点都是黑色,任何节点到每个叶子节点路径上的黑色节点数相同。

List接口

List集合的特点:有序(存取有序),可以重复的,可以有null,用户可以通过索引(类似于数组下标)精确地访问或者操作List中的元素

常用子类

ArrayList:底层数据结构是数组,线程不安全。

LinkedList:底层数据结构是双向链表,线程不安全

Vector:底层数据结构是数组,线程安全

ArrayList

构造方法

ArrayList()

构造一个初始容量为十的空列表。

ArrayList(Collection<? extends E> c)

构造一个包含指定集合的元素的列表,按照它们由集合的迭代器返回的顺序。

ArrayList(int initialCapacity)

构造具有指定初始容量的空列表。

普通方法

boolean add(E e)

将指定的元素追加到此列表的末尾。

void add(int index, E element)

在此列表中的指定位置插入指定的元素。

E get(int index)

返回此列表中指定位置的元素。

int indexOf(Object o)

返回此列表中指定元素的第一次出现的索引,如果此列表不包含元素,则返回-1。

Iterator<E> iterator()

以正确的顺序返回该列表中的元素的迭代器。

ListIterator<E> listIterator()

返回列表中的列表迭代器(按适当的顺序)。

E remove(int index)

删除该列表中指定位置的元素。

boolean remove(Object o)

从列表中删除指定元素的第一个出现(如果存在)。

int size()

返回此列表中的元素数。

E set(int index, E element)

用指定的元素替换此列表中指定位置的元素。


package com.day14.list;import com.day14.collet.Person;import java.util.ArrayList;
import java.util.Iterator;
import java.util.ListIterator;public class ArrayListDemo {public static void main(String[] args) {//创建ArrayList对象ArrayList arrayList = new ArrayList();//调用方法//int size()//返回集合中的实际有几个元素。System.out.println(arrayList.size());//add()arrayList.add(123);arrayList.add("java");arrayList.add("hello");//arrayList.add(5,"oracle");//不能跳着添加arrayList.add(3,"oracle");arrayList.add(null);arrayList.add("java");System.out.println(arrayList);//get()System.out.println(arrayList.get(3));//set()arrayList.set(3,"spring");System.out.println(arrayList);//remove()arrayList.remove(2);System.out.println(arrayList);//通过iterator()方法遍历Iterator iterator = arrayList.iterator();while (iterator.hasNext()){System.out.println(iterator.next());}System.out.println("-------------------");//从前向后遍历ListIterator listIterator = arrayList.listIterator();while (listIterator.hasNext()){System.out.println(listIterator.next());}System.out.println("-------------------");//从后向前遍历while (listIterator.hasPrevious()){System.out.println(listIterator.previous());}//从中ArrayList存入对象ArrayList<Person> list = new ArrayList<>();Person p1 = new Person("jack");Person p2 = new Person("Tom");Person p3 = new Person("Lucy");list.add(p1);list.add(p2);list.add(p3);System.out.println(list);}
}

ArrayList源码分析

属性

默认初始容量

private static final int DEFAULT_CAPACITY = 10;

空实例对象共享的一个空数组

private static final Object[] EMPTY_ELEMENTDATA = {};

默认大小的空示例共享的一个共数组

用来区分传入具体的空间参数值的,这样可以知道将来存放第一个元素

之后,需要创建多大的空间

private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {};

表示当前的arrayList对象

transient Object[] elementData;

当前arrayList中的真实空间大小

private int size;

1,new ArrayList 新创建的ArrayList对象就是空数组,容量为0

2,添加元素

public boolean add(E e) {

ensureCapacityInternal(size + 1); // Increments modCount!!

elementData[size++] = e;

return true;

}

add方法中,

第一步先调用 ensureCapacityInternal(size + 1);

第二步将传递进来的元素赋值到数组中的索引上

第三步返回结果true

private void ensureCapacityInternal(int minCapacity) {

ensureExplicitCapacity(calculateCapacity(elementData, minCapacity));

}

ensureCapacityInternal方法将传递过来的 size+1 在传入

calculateCapacity(elementData, minCapacity) 计算空间,计算完传入

ensureExplicitCapacity();

private static int calculateCapacity(Object[] elementData, int minCapacity) {

if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {

return Math.max(DEFAULT_CAPACITY, minCapacity);

}

return minCapacity;

}

calculateCapacity方法的判断是,将当前size+1之后,和默认的空间做比较

返回一个大的值,带入到ensureExplicitCapacity()方法中做计算

private void ensureExplicitCapacity(int minCapacity) {

modCount++;

// overflow-conscious code

if (minCapacity - elementData.length > 0)

grow(minCapacity);

}

ensureExplicitCapacity这个方法会将传递过来的空间值和当前数组的长度做比较

如果传递过来的空间值,比当前数组的长度大,就开始扩容

ArrayList的扩容方式:

如果默认创建无参构造ArrayList,开始是空数组,没有内容,空间为0,

当调用add方法加入第一个元素,通过计算空间方法,开始扩容,扩容到10,

当10个元素放满了,放入第11个元素,继续下次扩容,扩容为原来的1.5倍,

扩容的时候,其实是在使用数组复制功能。

ArrayList总结

ArrayList底层是一个可以动态扩容的数组,可以存放任意类型的元素,可以存放null元素,存取有序,元素可以重复,线程是不安全的。

创建无参构造ArrayList,默认初始空间为0,放入第一个元素,扩容为10,后续元素放满后,放入下一个元素会继续扩容,扩容1.5倍。

如果知道集合中存放的内容是多少,可以在初始化的时候,指定它的初始空间,可以提升效率。

效率上,非首尾元素增删慢,查询快,可以根据索引快速找到元素

ArrayList和Vector区别

Vector是一个比较老的集合类,jdk1.2

Vector底层也是数组,和ArrayList的最大区别就是,方法前面有Synchronized,

也就是方法是同步的,线程安全

相对来说,效率比ArrayList低

扩容上来说,Vector每次扩容2倍

LinkedList

说明:

1.底层是双向链表,可以存放任意类型元素

2.可以存放null元素,存取有序,元素可以重复,线程不安全

3.效率上,查询慢,增删快

4.结构上,它实现了Deque接口,因此LinkedList中还具有类似于操作队列和栈一样的方法

既然是链表,LinkedList中必然含有Node节点对象

private static class Node<E> {

E item; //节点中数据对象

Node<E> next; //当前节点的下一个节点

Node<E> prev; //当前节点的上一个节点

Node(Node<E> prev, E element, Node<E> next) {

this.item = element;

this.next = next;

this.prev = prev;

}

}

构造方法

LinkedList()

构造一个空列表。

普通方法

boolean add(E e)

将指定的元素追加到此列表的末尾。

void add(int index, E element)

在此列表中的指定位置插入指定的元素。

void addFirst(E e)

在该列表开头插入指定的元素。

void addLast(E e)

将指定的元素追加到此列表的末尾。

E get(int index)

返回此列表中指定位置的元素。

E getFirst()

返回此列表中的第一个元素。

E getLast()

返回此列表中的最后一个元素。

boolean offer(E e)

将指定的元素添加为此列表的尾部(最后一个元素)。

boolean offerFirst(E e)

在此列表的前面插入指定的元素。

boolean offerLast(E e)

在该列表的末尾插入指定的元素。

E pop()

从此列表表示的堆栈中弹出一个元素。

void push(E e)

将元素推送到由此列表表示的堆栈上。

package com.day14.list;import java.util.LinkedList;
import java.util.ListIterator;public class LinkedListDemo {public static void main(String[] args) {//创建一个LinkedList对象,存放String类型的元素LinkedList<String> linkedList = new LinkedList<>();//调用方法//addlinkedList.add("hello");linkedList.add("world");linkedList.add("java");System.out.println(linkedList);linkedList.addFirst("spring");linkedList.addLast("mysql");System.out.println(linkedList);//push方法相当于addFirstlinkedList.push("abc");linkedList.push("oracle");//往里加System.out.println(linkedList);//pop方法,弹栈,会将第一个元素取出linkedList.pop();//取出System.out.println(linkedList);//普通的get方法,并不会取出元素,只是获取了元素内容System.out.println(linkedList.get(1));System.out.println(linkedList);//getFirst获取第一个元素,不会将元素弹出System.out.println(linkedList.getFirst());System.out.println(linkedList);//遍历for (int i = 0; i < linkedList.size(); i++) {System.out.println(linkedList.get(1));}System.out.println("---------------------");for (String s : linkedList){System.out.println(s);}System.out.println("-----------------------");ListIterator<String> listIterator = linkedList.listIterator();while (listIterator.hasNext()){System.out.println(listIterator.next());}}
}

List总结

List接口具有有序、可重复、可以存放null、存放任意类型元素的特点,他的子实现类也具有相同的特点

ArrayList:底层是数组,初始空间默认为0,放入第一个元素,扩容为10,

后续元素放满后,放入下一个元素会继续扩容,扩容1.5倍。

增删慢,查询快

LinkedList:底层是双向链表,查询慢,增删快

Vector:底层也是数组,每次扩容2倍,同步的,效率比ArrayList低,被ArrayList替代

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

MySQL数据库---增删查改汇总

前言 欢迎来到我的博客 个人主页:北岭敲键盘的荒漠猫-CSDN博客 本文着重整理MySQL数据库增删查改功能 主要是整理语法 争取做到要用什么语法 可以快速找到复制粘贴 增添语法 INSERT into tab(列名,列名,列名) values(内容,内容,内容); 插入一行数据 INSERT into tab(列名,…

邦注科技 模具保护器 CCD电子眼 专业工业视觉检测设备

模具保护器是一种用于保护模具的设备&#xff0c;可以在塑料压铸和冲床等加工过程中起到保护模具的作用。以下是关于模具保护器在保护塑料压铸和冲床模具方面的应用&#xff1a; 塑料压铸模具保护器&#xff1a; 防止碰撞&#xff1a;在塑料压铸过程中&#xff0c;模具可能会…

Android Studio报错:Constant expression required

【出现的问题】&#xff1a; 使用JDK17以上版本&#xff0c;switch语句报错&#xff1a;Constant expression required 【解决方法】&#xff1a; 在gradle.properties配置文件下添加代码&#xff1a; android.nonFinalResIdsfalse 如图&#xff1a; 接着再点击右上角的Sync…

数仓开发:DIM层数据处理

一、了解DIM层 这个就是数仓开发的分层架构 我们现在是在DIM层&#xff0c;从ods表中数据进行加工处理&#xff0c;导入到dwd层&#xff0c;但是记住我们依然是在DIM层&#xff0c;而非是上面的ODS和DWD层。 二、处理维度表数据 ①先确认hive的配置 -- 开启动态分区方案 -- …

Python深度学习基于Tensorflow(6)神经网络基础

文章目录 使用Tensorflow解决XOR问题激活函数正向传播和反向传播解决过拟合权重正则化Dropout正则化批量正则化 BatchNormal权重初始化残差连接 选择优化算法传统梯度更新算法动量算法NAG算法AdaGrad算法RMSProp算法Adam算法如何选择优化算法 使用tf.keras构建神经网络使用Sequ…

活动图与状态图:UML中流程图的精细化表达——专业解析系统动态性与状态变迁

流程图是一种通用的图形表示法&#xff0c;用以展示步骤、决策和循环等流程控制结构。它通常用于描述算法、程序执行流程或业务过程&#xff0c;关注于任务的顺序执行。流程图强调顺序、分支和循环&#xff0c;适用于详细说明具体的处理步骤&#xff0c;图形符号相对基础和通用…

Logfire-Python可观测平台快速上手

我最近在优化之前的FastAPI接入可观测性平台&#xff0c;正好分享一下Pydantic团队推出的logfire&#xff0c;希望对大家的Python工程化有帮助。 Github: https://github.com/pydantic/logfire 官网链接: Pydantic Logfire Documentation Logfire是Pydantic团队推出的可观测…

数据结构复习指导之二叉树的遍历

文章目录 二叉树 考纲内容 复习提示 1.二叉树的遍历 1.1先序遍历&#xff08;PreOrder&#xff09; 1.2中序遍历&#xff08;InOrder&#xff09; 1.3后序遍历&#xff08;PostOrder&#xff09; 1.4递归算法和非递归算法的转换 1.5层次遍历 1.6由遍历序列构造二叉树…

3.yolov5训练前的图片处理详解(python)

其实&#xff0c;yolov5模型可以分为深度网络、数据处理&#xff08;图片处理&#xff09;、损失函数、优化器选择、训练和预测及部分构成&#xff0c;相信大家对训练和预测的代码比较熟悉。前面两章我们根据代码和结构图了解了yolov5的深度网络&#xff0c;接下来看数据处理的…

力扣刷题--数组--第三天

今天再做两道二分查找的题目&#xff0c;关于二分查找的知识可看我前两篇博客。话不多说&#xff0c;直接开干&#xff01; 题目1&#xff1a;69.x 的平方根 题目详情&#xff1a;   给你一个非负整数 x &#xff0c;计算并返回 x 的 算术平方根 。由于返回类型是整数&#…

首席数据官CCRC-CDO如何构筑企业数据合规的坚固防线

在当今信息化快速发展的时代&#xff0c;数据已经成为企业最宝贵的资产之一。然而&#xff0c;随着数据规模的迅速增长&#xff0c;数据合规问题也日益凸显。首席数据官&#xff08;CDO&#xff09;作为企业中负责数据战略和管理的核心人物&#xff0c;构筑企业数据合规的坚固防…

吴恩达2022机器学习专项课程C2(高级学习算法)W1(神经网络):2.5 更复杂的神经网络

目录 示例填写第三层的层数1.问题2.答案 公式&#xff1a;计算任意层的激活值激活函数 示例 层数有4层&#xff0c;不包括输入层。 填写第三层的层数 1.问题 你能把第二个神经元的上标和下标填写出来吗&#xff1f; 2.答案 根据公式g(wxb)&#xff0c;这里的x对应的是上…

Unity EventSystem入门

概述 相信在学习Unity中&#xff0c;一定有被UI事件困扰的时候把&#xff0c;当添加UICanvas的时候&#xff0c;Unity会为我们自动添加EventSystem&#xff0c;这个是为什么呢&#xff0c;Unity的UI事件是如何处理的呢&#xff0c;在使用各个UI组件的时候&#xff0c;一定有不…

Springboot+Vue项目-基于Java+MySQL的个人云盘管理系统(附源码+演示视频+LW)

大家好&#xff01;我是程序猿老A&#xff0c;感谢您阅读本文&#xff0c;欢迎一键三连哦。 &#x1f49e;当前专栏&#xff1a;Java毕业设计 精彩专栏推荐&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb; &#x1f380; Python毕业设计 &…

电脑文件x3daudio1 7.dll怎么修复?快速修复x3daudio1 7.dll的方法

你试过电脑文件x3daudio1 7.dll丢失么&#xff1f;如果你有遇到这种情况&#xff0c;那么可能你的某些程序就会启动不了&#xff0c;毕竟这个文件是用来处理音频功能的&#xff0c;那么我们要怎么去修复&#xff1f;下面我们一起来详细的了解电脑文件x3daudio1 7.dll这个文件吧…

顺序栈的操作

归纳编程学习的感悟&#xff0c; 记录奋斗路上的点滴&#xff0c; 希望能帮到一样刻苦的你&#xff01; 如有不足欢迎指正&#xff01; 共同学习交流&#xff01; &#x1f30e;欢迎各位→点赞 &#x1f44d; 收藏⭐ 留言​&#x1f4dd;既然选择了远方&#xff0c;当不负青春…

DNS、ICMP、NAT以及代理服务器

目录 1. DNS 1.1. DNS 背景 1.2. 域名简介 1.3. 域名解析过程 2. ICMP 2.1. ICMP 的功能 2.2. ICMP 的报文格式 2.3. ping 命令 2.4. traceroute 命令 3. NAT和代理服务器 3.1. NAT 技术 3.2. NAT IP转换过程 3.3. NAT 技术的缺陷 3.4. 代理服务器 3.4.1. 正向…

1.使用uniapp搭建微信小程序项目并引入前端组件资源

文章目录 1. 项目配置1.1. 新建vue3项目1.2. 关联云空间1.3. 运行到微信开发者工具 2. 前端组件2.1. uniCloud的内置组件和扩展组件2.2. uView3.02.3. 在uniapp项目引入uview3 1. 项目配置 1.1. 新建vue3项目 由于我们要使用vue3而不是vue2&#xff0c;所以要选好版本&#x…

SpringBoot实现统一返回值+全局异常处理

在这里首先感谢的就是程序员老罗&#xff0c;从他的项目里面学到了这些东西。 首先就是去创建一个SpringBoot项目&#xff0c;这里我就不多做赘述了 封装一个统一返回对象 package com.example.demo.vo;public class ResponseVO<T> {private String status;private In…

本地搭建AI环境

本地搭建AI 这几天刚刚看到好兄弟分享的一段关于本地搭建AI的短视频&#xff0c;于是我按照视频里的讲解&#xff0c;进行了实践。感觉非常棒&#xff01;&#xff01;&#xff0c;马上整理成文字与大家分享一下。 在本地启动并运行大型语言模型&#xff0c;运行llama3、phi3…