LinkedList序列分先进先出FIFO,先进后出FILO

FIFO在Java中又叫Queue 队列

FILO在Java中又叫Stack 栈

LinkedList 与 List接口与ArrayList一样,LinkedList也实现了List接口,诸如add,remove,contains等等方法。 详细使用,请参考 ArrayList 常用方法,在此不作赘述。

接下来要讲的是LinkedList的一些特别的地方

双向链表 - Deque除了实现了List接口外,LinkedList还实现了双向链表结构Deque,可以很方便的在头尾插入删除数据

什么是链表结构: 与数组结构相比较,数组结构,就好像是电影院,每个位置都有标示,每个位置之间的间隔都是一样的。 而链表就相当于佛珠,每个珠子,只连接前一个和后一个,不用关心除此之外的其他佛珠在哪里。

package collection;

import java.util.LinkedList;

import charactor.Hero;

public class TestCollection {

public static void main(String[] args) {

//LinkedList是一个双向链表结构的list

LinkedList ll =new LinkedList();

//所以可以很方便的在头部和尾部插入数据

//在最后插入新的英雄

ll.addLast(new Hero("hero1"));

ll.addLast(new Hero("hero2"));

ll.addLast(new Hero("hero3"));

System.out.println(ll);

//在最前面插入新的英雄

ll.addFirst(new Hero("heroX"));

System.out.println(ll);

//查看最前面的英雄

System.out.println(ll.getFirst());

//查看最后面的英雄

System.out.println(ll.getLast());

//查看不会导致英雄被删除

System.out.println(ll);

//取出最前面的英雄

System.out.println(ll.removeFirst());

//取出最后面的英雄

System.out.println(ll.removeLast());

//取出会导致英雄被删除

System.out.println(ll);

}

}

队列 - QueueLinkedList 除了实现了List和Deque外,还实现了Queue接口(队列)。

Queue是先进先出队列 FIFO,常用方法:

offer在最后添加元素

poll取出第一个元素

peek 查看第一个元素

package collection;

import java.util.LinkedList;

import java.util.List;

import java.util.Queue;

import charactor.Hero;

public class TestCollection {

public static void main(String[] args) {

//和ArrayList一样,LinkedList也实现了List接口

List ll =new LinkedList();

//所不同的是LinkedList还实现了Deque,进而又实现了Queue这个接口

//Queue代表FIFO 先进先出的队列

Queue q= new LinkedList();

//加在队列的最后面

System.out.print("初始化队列:\t");

q.offer(new Hero("Hero1"));

q.offer(new Hero("Hero2"));

q.offer(new Hero("Hero3"));

q.offer(new Hero("Hero4"));

System.out.println(q);

System.out.print("把第一个元素取poll()出来:\t");

//取出第一个Hero,FIFO 先进先出

Hero h = q.poll();

System.out.println(h);

System.out.print("取出第一个元素之后的队列:\t");

System.out.println(q);

//把第一个拿出来看一看,但是不取出来

h=q.peek();

System.out.print("查看peek()第一个元素:\t");

System.out.println(h);

System.out.print("查看并不会导致第一个元素被取出来:\t");

System.out.println(q);

}

}

使用LinkedList实现Stack栈与FIFO(先入先出的)队列类似的一种数据结构是FILO先入后出栈Stack

根据接口Stack :

实现类:MyStack

public class MyStack implements Stack并向这个栈中,压入5个英雄,接着弹出5个英雄

package collection;

import charactor.Hero;

public interface Stack {

//把英雄推入到最后位置

public void push(Hero h);

//把最后一个英雄取出来

public Hero pull();

//查看最后一个英雄

public Hero peek();

}

package collection;

import java.util.LinkedList;

import charactor.Hero;

public class MyStack implements Stack{

LinkedList heros = new LinkedList();

@Override

public void push(Hero h) {

heros.addLast(h);

}

@Override

public Hero pull() {

return heros.removeLast();

}

@Override

public Hero peek() {

return heros.getLast();

}

public static void main(String[] args) {

MyStack heroStack = new MyStack();

for (int i = 0; i < 5; i++) {

Hero h = new Hero("hero name " + i);

System.out.println("压入 hero:" + h);

heroStack.push(h);

}

for (int i = 0; i < 5; i++) {

Hero h =heroStack.pull();

System.out.println("弹出 hero" + h);

}

}

}

二叉树

二叉树概念二叉树由各种节点组成

二叉树特点:每个节点都可以有左子节点,右子节点

每一个节点都有一个值

package collection;

public class Node {

// 左子节点

public Node leftNode;

// 右子节点

public Node rightNode;

// 值

public Object value;

}

二叉树排序-插入数据假设通过二叉树对如下10个随机数进行排序

67,7,30,73,10,0,78,81,10,74

排序的第一个步骤是把数据插入到该二叉树中

插入基本逻辑是,小、相同的放左边,大的放右边

1. 67 放在根节点

2. 7 比 67小,放在67的左节点

3. 30 比67 小,找到67的左节点7,30比7大,就放在7的右节点

4. 73 比67大, 放在67的右节点

5. 10 比 67小,找到67的左节点7,10比7大,找到7的右节点30,10比30小,放在30的左节点。

...

...9. 10比67小,找到67的左节点7,10比7大,找到7的右节点30,10比30小,找到30的左节点10,10和10一样大,放在左边

package collection;

public class Node {

// 左子节点

public Node leftNode;

// 右子节点

public Node rightNode;

// 值

public Object value;

// 插入 数据

public void add(Object v) {

// 如果当前节点没有值,就把数据放在当前节点上

if (null == value)

value = v;

// 如果当前节点有值,就进行判断,新增的值与当前值的大小关系

else {

// 新增的值,比当前值小或者相同

if ((Integer) v -((Integer)value) <= 0) {

if (null == leftNode)

leftNode = new Node();

leftNode.add(v);

}

// 新增的值,比当前值大

else {

if (null == rightNode)

rightNode = new Node();

rightNode.add(v);

}

}

}

public static void main(String[] args) {

int randoms[] = new int[] { 67, 7, 30, 73, 10, 0, 78, 81, 10, 74 };

Node roots = new Node();

for (int number : randoms) {

roots.add(number);

}

}

}

二叉树排序-遍历通过上一个步骤的插入行为,实际上,数据就已经排好序了。 接下来要做的是看,把这些已经排好序的数据,遍历成我们常用的List或者数组的形式

二叉树的遍历分左序,中序,右序左序即: 中间的数遍历后放在左边

中序即: 中间的数遍历后放在中间

右序即: 中间的数遍历后放在右边如图所见,我们希望遍历后的结果是从小到大的,所以应该采用中序遍历

package collection;

import java.util.ArrayList;

import java.util.List;

public class Node {

// 左子节点

public Node leftNode;

// 右子节点

public Node rightNode;

// 值

public Object value;

// 插入 数据

public void add(Object v) {

// 如果当前节点没有值,就把数据放在当前节点上

if (null == value)

value = v;

// 如果当前节点有值,就进行判断,新增的值与当前值的大小关系

else {

// 新增的值,比当前值小或者相同

if ((Integer) v -((Integer)value) <= 0) {

if (null == leftNode)

leftNode = new Node();

leftNode.add(v);

}

// 新增的值,比当前值大

else {

if (null == rightNode)

rightNode = new Node();

rightNode.add(v);

}

}

}

// 中序遍历所有的节点

public List values() {

List values = new ArrayList<>();

// 左节点的遍历结果

if (null != leftNode)

values.addAll(leftNode.values());

// 当前节点

values.add(value);

// 右节点的遍历结果

if (null != rightNode)

values.addAll(rightNode.values());

return values;

}

public static void main(String[] args) {

int randoms[] = new int[] { 67, 7, 30, 73, 10, 0, 78, 81, 10, 74 };

Node roots = new Node();

for (int number : randoms) {

roots.add(number);

}

System.out.println(roots.values());

}

}案例:英雄二叉树

根据上面的学习和理解,设计一个Hero二叉树,HeroNode.

可以向这个英雄二叉树插入不同的Hero对象,并且按照Hero的血量倒排序。

随机生成10个Hero对象,每个Hero对象都有不同的血量值,插入这个HeroNode后,把排序结果打印出来。Hero

package charactor;

public class Hero {

public String name;

public float hp;

public int damage;

public Hero() {

}

public Hero(String name) {

this.name = name;

}

public String toString() {

return String.format("[name:%s hp:%.0f]%n", name,hp);

}

}HeroNode

package collection;

import java.util.ArrayList;

import java.util.List;

import charactor.Hero;

public class HeroNode {

public HeroNode leftHero;

public HeroNode rightHero;

// 声明为Hero类型

public Hero value;

public void add(Hero v) {

if (null == value)

value = v;

else {

// 如果新英雄血量,比本节点大,就放在左边

if (v.hp > value.hp) {

if (null == leftHero)

leftHero = new HeroNode();

leftHero.add(v);

}

else {

if (null == rightHero)

rightHero = new HeroNode();

rightHero.add(v);

}

}

}

public List values() {

List values = new ArrayList<>();

if (null != leftHero)

values.addAll(leftHero.values());

values.add(value);

if (null != rightHero)

values.addAll(rightHero.values());

return values;

}

public static void main(String[] args) {

List hs = new ArrayList<>();

for (int i = 0; i < 10; i++) {

Hero h = new Hero();

h.name = "hero " + i;

h.hp = (float) (Math.random() * 900 + 100); // 100-1000的随机血量

hs.add(h);

}

System.out.println("初始化10个Hero");

System.out.println(hs);

HeroNode heroTree = new HeroNode();

for (Hero hero : hs) {

heroTree.add(hero);

}

System.out.println("根据血量倒排序后的Hero");

List treeSortedHeros = heroTree.values();

System.out.println(treeSortedHeros);

}

}创建4万个随机数,然后用分别用冒泡法,选择法,二叉树3种排序算法进行排序,比较哪种更快

package collection;

import java.util.Arrays;

import java.util.List;

public class SortCompare {

public static void main(String[] args) {

//初始化随机数

int total = 40000;

System.out.println("初始化一个长度是"+total+"的随机数字的数组");

int[] originalNumbers = new int[total];

for (int i = 0; i < originalNumbers.length; i++) {

originalNumbers[i] = (int)(Math.random()*total);

}

System.out.println("初始化完毕");

System.out.println("接下来分别用3种算法进行排序");

//从初始化了的随机数组复制过来,以保证,每一种排序算法的目标数组,都是一样的

int[] use4sort;

use4sort= Arrays.copyOf(originalNumbers, originalNumbers.length);

int[] sortedNumbersBySelection= performance(new SelectionSort(use4sort),"选择法");

use4sort= Arrays.copyOf(originalNumbers, originalNumbers.length);

int[] sortedNumbersByBubbling=performance(new BubblingSort(use4sort),"冒泡法");

use4sort= Arrays.copyOf(originalNumbers, originalNumbers.length);

int[] sortedNumbersByTree=performance(new TreeSort(use4sort),"二叉树");

System.out.println("查看排序结果,是否是不同的数组对象");

System.out.println(sortedNumbersBySelection);

System.out.println(sortedNumbersByBubbling);

System.out.println(sortedNumbersByTree);

System.out.println("查看排序结果,内容是否相同");

System.out.println("比较 选择法 和 冒泡法 排序结果:");

System.out.println(Arrays.equals(sortedNumbersBySelection, sortedNumbersByBubbling));

System.out.println("比较 选择法 和 二叉树 排序结果:");

System.out.println(Arrays.equals(sortedNumbersBySelection, sortedNumbersByTree));

}

interface Sort{

void sort();

int[] values();

}

static class SelectionSort implements Sort{

int numbers[];

SelectionSort(int [] numbers){

this.numbers = numbers;

}

@Override

public void sort() {

for (int j = 0; j < numbers.length-1; j++) {

for (int i = j+1; i < numbers.length; i++) {

if(numbers[i]

int temp = numbers[j];

numbers[j] = numbers[i];

numbers[i] = temp;

}

}

}

}

@Override

public int[] values() {

// TODO Auto-generated method stub

return numbers;

}

}

static class BubblingSort implements Sort{

int numbers[];

BubblingSort(int [] numbers){

this.numbers = numbers;

}

@Override

public void sort() {

for (int j = 0; j < numbers.length; j++) {

for (int i = 0; i < numbers.length-j-1; i++) {

if(numbers[i]>numbers[i+1]){

int temp = numbers[i];

numbers[i] = numbers[i+1];

numbers[i+1] = temp;

}

}

}

}

@Override

public int[] values() {

// TODO Auto-generated method stub

return numbers;

}

}

static class TreeSort implements Sort{

int numbers[];

Node n;

TreeSort(int [] numbers){

n =new Node();

this.numbers = numbers;

}

@Override

public void sort() {

for (int i : numbers) {

n.add(i);

}

}

@Override

public int[] values() {

List list = n.values();

int sortedNumbers[] = new int[list.size()];

for (int i = 0; i < sortedNumbers.length; i++) {

sortedNumbers[i] = Integer.parseInt(list.get(i).toString());

}

return sortedNumbers;

}

}

private static int[] performance(Sort algorithm, String type) {

long start = System.currentTimeMillis();

algorithm.sort();

int sortedNumbers[] = algorithm.values();

long end = System.currentTimeMillis();

System.out.printf("%s排序,一共耗时 %d 毫秒%n",type,end-start);

return sortedNumbers;

}

}

HashMap

HashMap的键值对HashMap储存数据的方式是—— 键值对

package collection;

import java.util.HashMap;

public class TestCollection {

public static void main(String[] args) {

HashMap dictionary = new HashMap<>();

dictionary.put("adc", "物理英雄");

dictionary.put("apc", "魔法英雄");

dictionary.put("t", "坦克");

System.out.println(dictionary.get("t"));

}

}

键不能重复,值可以重复对于HashMap而言,key是唯一的,不可以重复的。

所以,以相同的key 把不同的value插入到 Map中会导致旧元素被覆盖,只留下最后插入的元素。

不过,同一个对象可以作为值插入到map中,只要对应的key不一样

package collection;

import java.util.HashMap;

import charactor.Hero;

public class TestCollection {

public static void main(String[] args) {

HashMap heroMap = new HashMap();

heroMap.put("gareen", new Hero("gareen1"));

System.out.println(heroMap);

//key为gareen已经有value了,再以gareen作为key放入数据,会导致原英雄,被覆盖

//不会增加新的元素到Map中

heroMap.put("gareen", new Hero("gareen2"));

System.out.println(heroMap);

//清空map

heroMap.clear();

Hero gareen = new Hero("gareen");

//同一个对象可以作为值插入到map中,只要对应的key不一样

heroMap.put("hero1", gareen);

heroMap.put("hero2", gareen);

System.out.println(heroMap);

}

}查找内容性能比较

准备一个ArrayList其中存放3000000(三百万个)Hero对象,其名称是随机的,格式是hero-[4位随机数]

hero-3229

hero-6232

hero-9365

...

因为总数很大,所以几乎每种都有重复,把名字叫做 hero-5555的所有对象找出来

要求使用两种办法来寻找

1. 不使用HashMap,直接使用for循环找出来,并统计花费的时间

2. 借助HashMap,找出结果,并统计花费的时间

package collection;

import java.util.ArrayList;

import java.util.HashMap;

import java.util.List;

import charactor.Hero;

public class TestCollection {

public static void main(String[] args) {

List hs =new ArrayList<>();

System.out.println("初始化开始");

for (int i = 0; i < 3000000; i++) {

Hero h = new Hero( "hero-" + random());

hs.add(h);

}

//名字作为key

//名字相同的hero,放在一个List中,作为value

HashMap> heroMap =new HashMap();

for (Hero h : hs) {

List list= heroMap.get( h.name);

if(list==null){

list = new ArrayList<>();

heroMap.put(h.name, list);

}

list.add(h);

}

System.out.println("初始化结束");

System.out.println("开始查找");

findByIteration(hs);

findByMap(heroMap);

}

private static List findByMap(HashMap> m) {

long start =System.currentTimeMillis();

List result= m.get("hero-5555");

long end =System.currentTimeMillis();

System.out.printf("通过map查找,一共找到%d个英雄,耗时%d 毫秒%n",result.size(),end-start);

return result;

}

private static List findByIteration (List hs) {

long start =System.currentTimeMillis();

List result =new ArrayList<>();

for (Hero h : hs) {

if(h.name.equals("hero-5555")){

result.add(h);

}

}

long end =System.currentTimeMillis();

System.out.printf("通过for查找,一共找到%d个英雄,耗时%d 毫秒%n", result.size(),end-start);

return result;

}

public static int random(){

return ((int)(Math.random()*9000)+1000);

}

}

感谢你看到这里,我是程序员麦冬,一个java开发从业者,深耕行业六年了,每天都会分享java相关技术文章或行业资讯

欢迎大家关注和转发文章,后期还有福利赠送!

java map用二叉树_【课堂笔记分享】linkedlist、二叉树、hashmap相关推荐

  1. java数据结构 农夫过河_数据结构笔记分享:18 农夫过河(图的算法运用)

    问题描述 一个农夫带着一只狼,一棵白菜和一只山羊要从一条河的南岸到北岸,农夫每次只能带一样东西过过河,但是任意时刻如果农夫不在场时,狼要吃羊.羊要吃菜,请为农夫设计过河方案. 分析: 要求解农夫过河问 ...

  2. 二叉树为空意味着二叉树_我是怎么调试出来二叉树的遍历(超精彩配图),从此遍历不再愁了...

    推荐学习 牛掰!"基础-中级-高级"Java程序员面试集结,看完献出我的膝盖 数据结构与算法,程序员必过的坎?不掌握一定挤不进BATJ的神技? 我是怎么调试出来二叉树的遍历(超精彩 ...

  3. java map 值排序_使用Java8 Stream API对Map类型按照键或值进行排序

    在这篇文章中,您将学习如何使用Java对Map按照键或值进行排序.前几日有位朋友面试遇到了这个问题,看似很简单的问题,但是如果不仔细研究一下也是很容易让人懵圈的面试题.所以我决定写这样一篇文章.在Ja ...

  4. java map同步访问_同步 - Java synchronized块与Collections.synchronizedMap

    您的代码中可能存在细微的错误. [更新:因为他正在使用map.remove(),所以这种描述并不完全有效. 我第一次错过了这个事实. :(感谢问题的作者指出这一点.我将其余部分保留原样,但改变了主要声 ...

  5. Python3入门与进阶_课堂笔记

    目录 1 介绍 1.2 用途 1.3 特点 2 基础 2.1 基本数据类型 2.1.1 Number 2.1.2 String 2.1.3 List 2.1.4 Tuple 2.1.5 Set 2.1 ...

  6. java map遍历顺序_深入理解HashMap遍历元素的顺序

    HashMap遍历元素的顺序. 一,HashMap元素的底层存储顺序 我们都知道HashMap是"无序"的,也就是说不能保证插入顺序.但是,HashMap其实也是有序的,一组相同的 ...

  7. java map类说_在Java中,关于HashMap类的描述,以下说法错误的是(   )。

    [单选题]所谓覆盖,也称为重写,是指子类中定义了一个与父类某一方法具有相同型构的方法,这里所说的相同型构所指的不是(). [填空题(主观)]半导体二极管实质上是由 [填空(1)] 构成,其主要特性是 ...

  8. java map 结构体_业务代码的救星——Java 对象转换框架 MapStruct 妙用

    简介 在业务项目的开发中,我们经常需要将 Java 对象进行转换,比如从将外部微服务得到的对象转换为本域的业务对象 domain object,将 domain object 转为数据持久层的 dat ...

  9. java map数组定义_如何定义一个空的Map[]数组?

    Map[] apsMap = (Map[])request.getAttribute("APSMAP"):有时候我们需要从后台获取一个Map数组,并且使用他. 但是这个Map数组有 ...

最新文章

  1. GNU make manual 翻译( 一百二十一)
  2. 随机过程:高斯函数导数、梯度
  3. 【Android开发学习笔记之一】5大布局方式详解
  4. Linux bashrc和profile的用途和区别
  5. Java 中 10 大坑爹功能!
  6. php 日期 间隔,PHP实现计算日期间隔天数的方法
  7. Spark常用算子讲解二
  8. Oracle exp/imp导入不同的表空间
  9. vscode+unity+Emmylua调试使用教程
  10. Linux查看系统自启动服务
  11. redis设置用户名和密码
  12. 今日头条信息流 - 橙子建站
  13. 可复用的显示隐藏按钮组件
  14. 【CTF题解NO.00003】moeCTF 2020 - official write up by arttnba3
  15. python送程序员收入_程序员学Python后惊叹,这么丰厚的收入是Java给不了的!
  16. 2021-05-24 Nyquist图背后的信息
  17. 可视门铃竟然可以泄露WiFi密码
  18. 白杨SEO:微信视频号怎么玩?推广、涨粉、上热门及赚钱技巧大全
  19. python程序中如何合理处理[WinError 5]拒绝访问的错误
  20. 角速度与rpm两者如何转换

热门文章

  1. Java 回调 (Callback) 接口学习使用
  2. Unity在运行时(代码中)设置材质的渲染模式(RenderingMode)
  3. windows + cmake + vs2019 编程
  4. 双11特刊|一站式在线数据管理平台DMS技术再升级,高效护航双11
  5. 10个Bug环环相扣,你能解开几个?
  6. 揭秘!阿里实时数仓分布式事务Scale Out设计
  7. 对话鲁直:蚂蚁金服中间件的开源头羊 | 穿山甲专访
  8. 表格存储TableStore全新升级,打造统一的在线数据存储平台!
  9. 日志服务与SIEM(如Splunk)集成方案实战
  10. 曦智科技发布最新光子计算处理器PACE