目录
一.数组(Array):
1.1 特点:
1.2 基本操作:
1.3 使用数组的好处包括:
1.4 数组也有一些限制:
二.集合框架(Collections Framework):
2.1 列表(List):
2.1.1 数组列表(ArrayList):
扩展知识:
方法:
ArrayList的遍历方式
ArrayList使用泛型
Arrays工具类
2.1.2 链表(LinkedList):
常用方法:
2.1.3 使用选择:
2.1.4 示例代码:
2.2 集(Set):
2.2.1 哈希集(HashSet):
基本类型对应的包装类表如下:
HashSet的常用方法包括:
2.2.2 树集(TreeSet):
TreeSet的常用方法包括:
2.2.3 使用选择:
2.2.4 示例代码:
2.3 映射(Map):
2.3.1 哈希映射(HashMap):
2.3.2 树映射(TreeMap):
2.3.3 使用选择
2.3.4 示例代码
三.栈(Stack)和队列(Queue):
3.1方法
3.2示例代码
四.树(Tree):
4.1 二叉树(Binary Tree):
4.2二叉搜索树(Binary Search Tree):
4.3 操作方法
4.4 树的遍历
4.4.1树的遍历包括两种主要方式:
4.4.2 树的两种基本遍历方式:
4.5 示例代码
五.图(Graph):
六.堆(Heap):
七.链表(LinkedList):
八.MAP:
Java 提供了多种数据结构来有效地组织和处理数据。以下是一些常见的 Java 数据结构:
数组(Array)是一种线性数据结构,用于存储固定大小的相同类型元素的连续序列,可以通过索引快速访问和修改元素。。在 Java 中,数组可以包含基本类型(如 int、double、char 等)或引用类型(如对象、字符串等)的元素。
初始化:创建一个数组,并为其赋初值。
遍历:使用循环遍历数组中的每个元素。
打印:将数组中的元素依次输出。
最大值:遍历数组,通过比较找到数组中的最大值。
最大值下标:遍历数组,记录当前最大值的下标。
System.arraycopy()
方法进行数组复制。clone()
方法复制数组。以下的Java代码示例,包含数组的初始化、遍历、打印、最大值、最大值下标、使用增强for循环遍历、复制数组以及线性查找、二分查找和选择排序算法的实现。
import java.util.Arrays;
public class myClass {
public static void main(String[] args) {
// 初始化数组
int[] arr = {5, 3, 9, 1, 7};
// 遍历数组
for (int i = 0; i < arr.length; i++) {
System.out.print(arr[i] + " ");
}
System.out.println();
// 打印数组
System.out.println(Arrays.toString(arr));
// 最大值
int max = arr[0];
for (int i = 1; i < arr.length; i++) {
if (arr[i] > max) {
max = arr[i];
}
}
System.out.println("最大值:" + max);
// 最大值下标
int maxIndex = 0;
for (int i = 1; i < arr.length; i++) {
if (arr[i] > arr[maxIndex]) {
maxIndex = i;
}
}
System.out.println("最大值下标:" + maxIndex);
// 使用增强for循环遍历
for (int num : arr) {
System.out.print(num + " ");
}
System.out.println();
// 复制数组方法一(遍历原数组,逐个复制到新数组)
int[] copyArr1 = new int[arr.length];
for (int i = 0; i < arr.length; i++) {
copyArr1[i] = arr[i];
}
// 复制数组方法二(使用System.arraycopy()方法)
int[] copyArr2 = new int[arr.length];
System.arraycopy(arr, 0, copyArr2, 0, arr.length);
// 复制数组方法三(使用数组的clone()方法)
int[] copyArr3 = arr.clone();
System.out.println("复制后的数组:" + Arrays.toString(copyArr3));
// 线性查找
int target = 9;
int linearSearchIndex = -1;
for (int i = 0; i < arr.length; i++) {
if (arr[i] == target) {
linearSearchIndex = i;
break;
}
}
System.out.println("线性查找:" + (linearSearchIndex != -1 ? linearSearchIndex : "未找到"));
// 二分查找(前提是数组必须有序)
Arrays.sort(arr); // 先排序
int binarySearchIndex = Arrays.binarySearch(arr, target);
System.out.println("二分查找:" + (binarySearchIndex >= 0 ? binarySearchIndex : "未找到"));
// 选择排序
selectionSort(arr);
System.out.println("选择排序后的数组:" + Arrays.toString(arr));
}
// 选择排序算法实现
public static void selectionSort(int[] arr) {
int n = arr.length;
for (int i = 0; i < n - 1; i++) {
int minIndex = i;
for (int j = i + 1; j < n; j++) {
if (arr[j] < arr[minIndex]) {
minIndex = j;
}
}
if (minIndex != i) {
int temp = arr[i];
arr[i] = arr[minIndex];
arr[minIndex] = temp;
}
}
}
}
运行结果:
5 3 9 1 7
[5, 3, 9, 1, 7]
最大值:9
最大值下标:2
5 3 9 1 7
复制后的数组:[5, 3, 9, 1, 7]
线性查找:2
二分查找:4
选择排序后的数组:[1, 3, 5, 7, 9]
提供了一系列接口和类,用于存储和操作对象的集合。常用的集合类包括:
有序、可重复的集合,例如 ArrayList、LinkedList。
Vector是Java中的一个旧的类,它与ArrayList具有相似的用法,但在性能和使用上存在一些差异。虽然Vector是线程安全的,但在大多数情况下,推荐使用更现代的替代方案。
一个常见的替代方案是使用Collections工具类提供的synchronizedList方法,该方法可以将ArrayList转换为线程安全的List。以下是示例代码:
List lst = new ArrayList<>();
// 往lst中添加元素
List synList = Collections.synchronizedList(lst);
通过将原始的ArrayList通过synchronizedList方法包装,我们可以获得一个在并发环境下安全使用的List。
需要注意的是,在大多数情况下,如果不需要线程安全的特性,推荐使用ArrayList而不是Vector,因为ArrayList在性能上通常更优。只有在确实需要线程安全时,才会考虑使用Vector或通过Collections工具类获得线程安全的List。
Collection相关方法:
这些方法属于Collection类的一部分,可以被List和Set等子类继承并使用。以下是一些常见的Collection类方法:
List接口:
List是Collection接口的子接口,拥有Collection所有方法外,还有一些对索引操作的方法。
更多 API 方法可以查看:ArrayList
for (int i = 0; i < arrayList.size(); i++) {
E element = arrayList.get(i);
// 处理元素
}
for (E element : arrayList) {
// 处理元素
}
Iterator iterator = arrayList.iterator();
while (iterator.hasNext()) {
E element = iterator.next();
// 处理元素
}
泛型数据类型,用于设置 objectName(对象名) 的数据类型,只能为引用数据类型。
对于ArrayList的泛型使用,可以通过在定义ArrayList时指定泛型类型来确保集合中只能存储指定类型的元素。例如:
ArrayList stringList = new ArrayList<>();
stringList.add("Hello");
stringList.add("World");
// 只能添加String类型的元素
String element = stringList.get(0);
System.out.println(element); // 输出: Hello
通过使用泛型,可以提供类型安全,并且在编译期间就能够捕获类型不匹配的错误。
Arrays工具类是Java提供的用于操作数组的工具类,以下是一些常用的方法:
更多 API 方法可以查看:LinkedList
根据具体需求,选择ArrayList还是LinkedList有以下一些考虑因素:
总的来说,ArrayList适用于读取和随机访问操作更多的场景,而LinkedList适用于插入和删除操作更频繁的场景。
下面一个简单的示例代码,展示了如何使用ArrayList和LinkedList进行基本操作:
import java.util.ArrayList;
import java.util.LinkedList;
import java.util.List;
public class ListExample {
public static void main(String[] args) {
// 使用ArrayList
List arrayList = new ArrayList<>();
// 添加元素
arrayList.add("Apple");
arrayList.add("Banana");
arrayList.add("Orange");
// 获取元素
String firstElement = arrayList.get(0);
System.out.println("ArrayList中第一个元素:" + firstElement);
// 遍历元素
System.out.println("ArrayList的元素:");
for (String element : arrayList) {
System.out.println(element);
}
// 删除元素
arrayList.remove("Banana");
// 使用LinkedList
List linkedList = new LinkedList<>();
// 添加元素
linkedList.add(10);
linkedList.add(20);
linkedList.add(30);
// 获取元素
int lastElement = linkedList.get(linkedList.size() - 1);
System.out.println("LinkedList中最后一个元素:" + lastElement);
// 遍历元素
System.out.println("LinkedList的元素:");
for (int element : linkedList) {
System.out.println(element);
}
// 删除元素
linkedList.remove(0);
}
}
运行结果:
ArrayList中第一个元素:Apple
ArrayList的元素:
Apple
Banana
Orange
LinkedList中最后一个元素:30
LinkedList的元素:
10
20
30
集是一种无序的集合,不允许重复元素,例如 HashSet、TreeSet。
HashSet 中的元素实际上是对象,一些常见的基本类型可以使用它的包装类。
基本类型 | 引用类型 |
---|---|
boolean | Boolean |
byte | Byte |
short | Short |
int | Integer |
long | Long |
float | Float |
double | Double |
char | Character |
更多 API 方法可以查看:HashSet
下面一个简单的示例代码,展示了如何使用HashSet和TreeSet进行基本操作:
import java.util.HashSet;// 引入 HashSet 类
import java.util.TreeSet;// 引入 TreeSet 类
public class myClass {
public static void main(String[] args) {
// 使用HashSet
HashSet hashSet = new HashSet<>();
// 添加元素
hashSet.add("Apple");
hashSet.add("Banana");
hashSet.add("Orange");
hashSet.add("Grape");
// 输出集合中的元素
System.out.println("HashSet:");
for (String fruit : hashSet) {
System.out.println(fruit);
}
// 检查元素是否存在
System.out.println("HashSet contains 'Apple': " + hashSet.contains("Apple")); // true
// 删除元素
hashSet.remove("Orange");
// 输出修改后的集合
System.out.println("HashSet after removal:");
for (String fruit : hashSet) {
System.out.println(fruit);
}
// 使用TreeSet
TreeSet treeSet = new TreeSet<>();
// 添加元素
treeSet.add(5);
treeSet.add(10);
treeSet.add(3);
treeSet.add(8);
// 输出集合中的元素
System.out.println("TreeSet:");
for (int num : treeSet) {
System.out.println(num);
}
// 获取最小元素和最大元素
System.out.println("Min element: " + treeSet.first()); // 3
System.out.println("Max element: " + treeSet.last()); // 10
}
}
运行结果:
HashSet:
Apple
Grape
Orange
Banana
HashSet contains 'Apple': true
HashSet after removal:
Apple
Grape
Banana
TreeSet:
3
5
8
10
Min element: 3
Max element: 10
由键值对组成的集合,每个键唯一,例如 HashMap、TreeMap。
常用方法:
更多 API 方法可以查看:HashMap
常用方法:
除了上述方法,哈希映射和树映射还提供了其他一些类似的方法,如获取所有键的集合、清空映射等。具体的方法使用可以参考Java的官方文档或相关教程。
这两种映射都是非常常用的数据结构,在不同的场景下使用。
下面一个简单的示例代码,展示了如何使用HashMap和TreeMap进行基本操作:
import java.util.HashMap;
import java.util.Map;
import java.util.TreeMap;
public class myClass {
public static void main(String[] args) {
// 使用HashMap
Map hashMap = new HashMap<>();
// 添加键值对到HashMap
hashMap.put("Apple", 1);
hashMap.put("Banana", 2);
hashMap.put("Orange", 3);
// 获取指定键的值
int appleValue = hashMap.get("Apple");
System.out.println("Apple的值为:" + appleValue);
// 检查HashMap是否包含指定键
boolean containsKey = hashMap.containsKey("Banana");
System.out.println("HashMap是否包含Banana:" + containsKey);
// 移除指定键的键值对
hashMap.remove("Orange");
// 遍历HashMap的键值对
for (String key : hashMap.keySet()) {
int value = hashMap.get(key);
System.out.println("键:" + key + ",值:" + value);
}
System.out.println("----------------------");
// 使用TreeMap
Map treeMap = new TreeMap<>();
// 添加键值对到TreeMap
treeMap.put("Apple", 1);
treeMap.put("Banana", 2);
treeMap.put("Orange", 3);
// 获取最小的键
String firstKey = ((TreeMap) treeMap).firstKey();
System.out.println("最小的键:" + firstKey);
// 获取最大的键
String lastKey = ((TreeMap) treeMap).lastKey();
System.out.println("最大的键:" + lastKey);
// 遍历TreeMap的键值对
for (String key : treeMap.keySet()) {
int value = treeMap.get(key);
System.out.println("键:" + key + ",值:" + value);
}
}
}
运行结果:
Apple的值为:1
HashMap是否包含Banana:true
键:Apple,值:1
键:Banana,值:2
----------------------
最小的键:Apple
最大的键:Orange
键:Apple,值:1
键:Banana,值:2
键:Orange,值:3
栈是一种后进先出(LIFO)的数据结构,常用操作有入栈和出栈;队列是一种先进先出(FIFO)的数据结构,常用操作有入队和出队。Java 提供了 Stack 类和 Queue 接口,并有相关实现类如 LinkedList。
在Java中,可以使用Stack类表示栈,它是Vector的子类。而Queue接口是Java集合框架的一部分,它定义了队列的基本操作,如入队、出队、获取队头元素等。Java提供了多个Queue接口的实现类,其中常用的是LinkedList,它既实现了List接口,又实现了Deque接口,可以作为队列或双端队列使用。
Stack的常用方法包括:
Queue的常用方法包括:
下面一个简单的示例代码,展示了如何使用Stack和Queue进行基本操作:
import java.util.Stack;
import java.util.LinkedList;
import java.util.Queue;
public class myClass {
public static void main(String[] args) {
// 使用Stack
Stack stack = new Stack<>();
// 入栈操作
stack.push(1);
stack.push(2);
stack.push(3);
// 出栈操作
int poppedElement = stack.pop();
System.out.println("出栈元素:" + poppedElement);
// 获取栈顶元素
int topElement = stack.peek();
System.out.println("栈顶元素:" + topElement);
// 判断栈是否为空
boolean empty = stack.isEmpty();
System.out.println("栈是否为空:" + empty);
System.out.println("----------------------");
// 使用Queue
Queue queue = new LinkedList<>();
// 入队操作
queue.offer("Apple");
queue.offer("Banana");
queue.offer("Orange");
// 出队操作
String polledElement = queue.poll();
System.out.println("出队元素:" + polledElement);
// 获取队头元素
String peekedElement = queue.peek();
System.out.println("队头元素:" + peekedElement);
// 判断队列是否为空
boolean emptyQueue = queue.isEmpty();
System.out.println("队列是否为空:" + emptyQueue);
}
}
运行结果:
出栈元素:3
栈顶元素:2
栈是否为空:false
----------------------
出队元素:Apple
队头元素:Banana
队列是否为空:false
树是一种非线性的数据结构,由节点和边组成,用于表示具有层级关系的数据结构。常见的树结构包括二叉树、平衡树、红黑树等。Java 提供了 TreeSet 和 TreeMap 来实现树结构。
是一种特殊的树结构,每个节点最多有两个子节点:左子节点和右子节点。一个节点可以没有子节点,也可以只有一个子节点。
也称为二叉查找树或排序二叉树,是一种特殊的二叉树。它具有以下性质:
由于二叉搜索树的性质,我们可以利用它来高效地进行查找、插入和删除操作。对于任意节点,其左子树的节点值都小于该节点,因此在查找、插入或删除时,可以通过比较节点的值,有选择性地在左子树或右子树中进行操作,从而提高效率。
Java中可以使用TreeSet和TreeMap来实现二叉搜索树。它们基于红黑树(Red-Black Tree)实现,红黑树是一种自平衡的二叉搜索树,保持了良好的平衡性能。
通过使用TreeSet和TreeMap,我们可以方便地操作二叉搜索树的相关操作,如插入、删除、查找等,并保持元素的有序性。
深度优先遍历(DFS):
广度优先遍历(BFS): 从根节点开始,按层级顺序逐层遍历树的节点,从左到右依次访问每个节点。
无论是深度优先遍历还是广度优先遍历,都有各自的应用场景。可以根据具体需求选择适合的遍历方式来处理树中的节点。
先序遍历的顺序是先访问根节点,然后按照从左到右的顺序依次访问左子树和右子树。
中序遍历的顺序是先按照从左到右的顺序递归访问左子树,然后访问根节点,最后按照从左到右的顺序递归访问右子树。
这两种遍历方法都是通过递归实现的,可以用来遍历二叉树或其他类型的树结构。它们在不同的情况下有各自的应用,具体取决于问题的需求。
下面一个简单的示例代码,展示了如何使用Binary Tree和Binary Search Tree进行基本操作:
// 定义二叉树节点类
class TreeNode {
int val;
TreeNode left;
TreeNode right;
public TreeNode(int val) {
this.val = val;
this.left = null;
this.right = null;
}
}
// 二叉树类
class BinaryTree {
TreeNode root;
public BinaryTree() {
root = null;
}
// 插入节点
public void insert(int val) {
root = insertNode(root, val);
}
private TreeNode insertNode(TreeNode node, int val) {
if (node == null) {
return new TreeNode(val);
}
// 如果值小于当前节点,则插入左子树
if (val < node.val) {
node.left = insertNode(node.left, val);
}
// 如果值大于等于当前节点,则插入右子树
else {
node.right = insertNode(node.right, val);
}
return node;
}
// 先序遍历
public void preOrderTraversal() {
preOrder(root);
}
private void preOrder(TreeNode node) {
if (node == null) {
return;
}
System.out.print(node.val + " ");
preOrder(node.left);
preOrder(node.right);
}
}
// 二叉搜索树类
class BinarySearchTree {
TreeNode root;
public BinarySearchTree() {
root = null;
}
// 插入节点
public void insert(int val) {
root = insertNode(root, val);
}
private TreeNode insertNode(TreeNode node, int val) {
if (node == null) {
return new TreeNode(val);
}
// 如果值小于当前节点,则插入左子树
if (val < node.val) {
node.left = insertNode(node.left, val);
}
// 如果值大于等于当前节点,则插入右子树
else {
node.right = insertNode(node.right, val);
}
return node;
}
// 中序遍历
public void inOrderTraversal() {
inOrder(root);
}
private void inOrder(TreeNode node) {
if (node == null) {
return;
}
inOrder(node.left);
System.out.print(node.val + " ");
inOrder(node.right);
}
}
// 测试代码
public class myClass {
public static void main(String[] args) {
// 创建二叉树,并插入节点
BinaryTree binaryTree = new BinaryTree();
binaryTree.insert(5);
binaryTree.insert(3);
binaryTree.insert(7);
binaryTree.insert(2);
binaryTree.insert(4);
binaryTree.insert(6);
binaryTree.insert(8);
// 先序遍历二叉树
System.out.print("二叉树 -先序遍历: ");
binaryTree.preOrderTraversal();
System.out.println();
// 创建二叉搜索树,并插入节点
BinarySearchTree binarySearchTree = new BinarySearchTree();
binarySearchTree.insert(5);
binarySearchTree.insert(3);
binarySearchTree.insert(7);
binarySearchTree.insert(2);
binarySearchTree.insert(4);
binarySearchTree.insert(6);
binarySearchTree.insert(8);
// 中序遍历二叉搜索树
System.out.print("二叉搜索树 - 中序遍历: ");
binarySearchTree.inOrderTraversal();
System.out.println();
}
}
运行结果:
二叉树 -先序遍历: 5 3 2 4 7 6 8
二叉搜索树 - 中序遍历: 2 3 4 5 6 7 8
图(Graph)是由节点(Vertex)和边(Edge)组成的一种非线性数据结构,用于表示对象之间的关联关系。可以将节点视为图中的元素,并且节点可以与其他节点通过边相连接。
在实际应用中,图可用于解决各种问题,如社交网络分析、路径搜索、最短路径算法等。在 Java 中,可以自行实现图的数据结构和相关算法,或使用第三方库(如JGraphT、Java Universal Network/Graph Framework等)来操作和处理图结构。
堆是一种特殊的树形数据结构,满足堆属性。用于有效地找到最大值或最小值的数据结构。Java 提供了优先级队列(PriorityQueue)来实现堆。
链表是一种线性数据结构,也叫动态数据结构,但是并不会按线性的顺序存储数据,由节点和指针组成,每个节点包含数据和指向下一个节点的指针。
链表可分为单向链表和双向链表。
一个单向链表包含两个值: 当前节点的值和一个指向下一个节点的链接。
一个双向链表有三个整数值: 数值、向后的节点链接、向前的节点链接。
Map是一种用于存储键值对的接口。它提供了一种将键映射到值的方式,可以通过键快速查找对应的值。在Map中,键是唯一的,每个键最多只能映射到一个值。常见的Map实现类包括HashMap、TreeMap、LinkedHashMap等。这些实现类提供了不同的性能特点和迭代顺序,可以根据具体需求选择适合的实现类。使用Map可以方便地进行数据检索、插入、删除和更新操作,非常适合处理需要快速查找的场景。