数据结构第08小节:双端队列

双端队列(deque,double-ended queue)是一种具有队列和栈特性的数据结构,允许在其两端进行插入和删除操作。在Java中,java.util.Deque接口就是双端队列的实现,而ArrayDequeLinkedList是其中的具体实现类。

下面是一个使用ArrayDeque实现的双端队列的例子,我们将使用它来管理一个在线购物车系统中的商品列表。用户可以从任何一端(前面或后面)添加或移除商品。

import java.util.ArrayDeque;
import java.util.Deque;

// 商品类
class Product {
    String name;
    double price;

    public Product(String name, double price) {
        this.name = name;
        this.price = price;
    }

    @Override
    public String toString() {
        return "Product{" +
                "name='" + name + '\'' +
                ", price=" + price +
                '}';
    }
}

public class ShoppingCart {
    private Deque<Product> cart;

    public ShoppingCart() {
        cart = new ArrayDeque<>();
    }

    // 向购物车尾部添加商品
    public void addProduct(Product product) {
        cart.addLast(product);
    }

    // 向购物车头部添加商品
    public void addFirstProduct(Product product) {
        cart.addFirst(product);
    }

    // 从购物车尾部移除商品
    public Product removeProduct() {
        return cart.pollLast();
    }

    // 从购物车头部移除商品
    public Product removeFirstProduct() {
        return cart.pollFirst();
    }

    // 显示购物车中的所有商品
    public void displayCart() {
        while (!cart.isEmpty()) {
            System.out.println(cart.pollFirst());
        }
    }

    public static void main(String[] args) {
        ShoppingCart cart = new ShoppingCart();

        // 添加商品
        cart.addProduct(new Product("Apple iPhone 13", 999.99));
        cart.addFirstProduct(new Product("Google Pixel 6", 599.99));
        cart.addProduct(new Product("Samsung Galaxy S21", 799.99));

        // 显示购物车内容
        System.out.println("Shopping Cart Contents:");
        cart.displayCart();

        // 移除商品
        System.out.println("\nAfter removing the last item:");
        cart.removeProduct();
        cart.displayCart();

        // 再次移除商品
        System.out.println("\nAfter removing the first item:");
        cart.removeFirstProduct();
        cart.displayCart();
    }
}

在这个例子中:

  • Product 类表示商品,包含名称和价格属性。
  • ShoppingCart 类使用ArrayDeque作为内部数据结构来存储商品。它提供了添加商品到队列尾部或头部的方法,以及从队列尾部或头部移除商品的方法。
  • main 方法创建了一个ShoppingCart对象,添加了一些商品,显示了购物车的内容,然后移除了商品并再次显示购物车的内容。

这个例子展示了如何使用Java中的双端队列来管理一个购物车系统中的商品列表,允许用户从任意一端添加或移除商品,这在许多实际应用场景中都非常有用。

让我们通过一个详细的案例和表格来进一步说明如何使用双端队列(Deque)在Java中管理一个购物车系统。我们将使用ArrayDeque作为我们的双端队列实现,并跟踪购物车中的商品添加和移除操作。

示例代码

import java.util.ArrayDeque;
import java.util.Deque;

// 商品类
class Product {
    String name;
    double price;

    public Product(String name, double price) {
        this.name = name;
        this.price = price;
    }

    @Override
    public String toString() {
        return "Product{" +
                "name='" + name + '\'' +
                ", price=" + price +
                '}';
    }
}

public class ShoppingCart {
    private Deque<Product> cart;

    public ShoppingCart() {
        cart = new ArrayDeque<>();
    }

    // 向购物车尾部添加商品
    public void addProduct(Product product) {
        cart.addLast(product);
    }

    // 向购物车头部添加商品
    public void addFirstProduct(Product product) {
        cart.addFirst(product);
    }

    // 从购物车尾部移除商品
    public Product removeProduct() {
        return cart.pollLast();
    }

    // 从购物车头部移除商品
    public Product removeFirstProduct() {
        return cart.pollFirst();
    }

    // 显示购物车中的所有商品
    public void displayCart() {
        System.out.println("Current Cart Contents:");
        for (Product product : cart) {
            System.out.println(product);
        }
    }

    public static void main(String[] args) {
        ShoppingCart cart = new ShoppingCart();

        // 添加商品到购物车
        cart.addProduct(new Product("Apple iPhone 13", 999.99));
        cart.addFirstProduct(new Product("Google Pixel 6", 599.99));
        cart.addProduct(new Product("Samsung Galaxy S21", 799.99));

        // 显示购物车内容
        cart.displayCart();

        // 移除商品
        System.out.println("\nAfter removing the last item:");
        cart.removeProduct();
        cart.displayCart();

        // 再次移除商品
        System.out.println("\nAfter removing the first item:");
        cart.removeFirstProduct();
        cart.displayCart();
    }
}

表格说明

购物车操作记录
操作描述购物车状态(从左至右为队首至队尾)
初始化创建一个空的购物车[]
添加到队尾(addLast)添加“Apple iPhone 13”[“Apple iPhone 13”]
添加到队首(addFirst)添加“Google Pixel 6”[“Google Pixel 6”, “Apple iPhone 13”]
添加到队尾(addLast)添加“Samsung Galaxy S21”[“Google Pixel 6”, “Apple iPhone 13”, “Samsung Galaxy S21”]
移除队尾(removeLast)移除“Samsung Galaxy S21”[“Google Pixel 6”, “Apple iPhone 13”]
移除队首(removeFirst)移除“Google Pixel 6”[“Apple iPhone 13”]

通过这个案例和表格,我们可以清楚地看到双端队列在购物车管理中的应用:商品可以被添加到队列的任一端,也可以从任一端移除。这种灵活性使得双端队列成为处理双向数据流或需要快速访问两端数据的场景的理想选择。

让我们继续深入探讨双端队列(Deque)的应用,这一次我们将使用它来解决一个常见的编程问题:回文检测。回文是指正读反读都一样的字符串,例如“racecar”。我们将使用双端队列的特性来高效地检查一个给定的字符串是否是回文。

示例代码

import java.util.ArrayDeque;
import java.util.Deque;

public class PalindromeChecker {
    public static boolean isPalindrome(String str) {
        Deque<Character> deque = new ArrayDeque<>();

        // 将字符串中的字符添加到双端队列中
        for (char c : str.toLowerCase().toCharArray()) {
            if (Character.isLetter(c)) {
                deque.addLast(c);
            }
        }

        char first, last;
        while (deque.size() > 1) {
            first = deque.removeFirst();
            last = deque.removeLast();

            if (first != last) {
                return false;
            }
        }
        return true;
    }

    public static void main(String[] args) {
        String testString = "A man a plan a canal Panama";
        if (isPalindrome(testString)) {
            System.out.println(testString + " is a palindrome.");
        } else {
            System.out.println(testString + " is not a palindrome.");
        }
    }
}

代码解释

在这段代码中,我们定义了一个名为PalindromeChecker的类,其中包含一个静态方法isPalindrome,该方法接受一个字符串参数并返回一个布尔值,指示该字符串是否为回文。

  1. 初始化双端队列:我们使用ArrayDeque作为双端队列的实现,并将其实例化。
  2. 添加字符:遍历输入字符串的每个字符,只将字母字符转换为小写并添加到队列的尾部。
  3. 比较字符:从双端队列的头部和尾部同时移除字符,并比较它们是否相同。如果任何时候两个字符不匹配,那么字符串就不是回文,方法立即返回false
  4. 结束条件:如果队列中只剩下一个字符或没有字符,则说明字符串是回文,方法返回true

运行结果

当你运行main方法时,它会检查字符串“A man a plan a canal Panama”是否为回文。由于忽略空格和大小写,这个字符串实际上是一个回文,因此控制台将输出:“A man a plan a canal Panama is a palindrome.”

这个例子展示了如何利用双端队列的特性来简化和优化一些算法,如回文检测,这在实际编程中是非常有用的技巧。

接下来,我们将使用双端队列(Deque)来解决一个有趣的问题:浏览器的历史记录管理。当我们浏览网页时,通常会使用浏览器的前进和后退按钮来在历史记录之间导航。双端队列非常适合模拟这一行为,因为它允许我们在队列的两端高效地添加和移除元素。

示例代码

import java.util.ArrayDeque;
import java.util.Deque;

public class BrowserHistory {
    private Deque<String> history;
    private Deque<String> forwardHistory;

    public BrowserHistory() {
        history = new ArrayDeque<>();
        forwardHistory = new ArrayDeque<>();
    }

    // 访问新的网址
    public void visit(String url) {
        history.addLast(url);
        forwardHistory.clear(); // 清空前进历史
    }

    // 返回上一个页面
    public String back() {
        if (history.size() > 1) {
            String currentUrl = history.removeLast();
            forwardHistory.addFirst(currentUrl);
            return history.getLast();
        } else {
            return "No more pages to go back.";
        }
    }

    // 前往下一个页面
    public String forward() {
        if (!forwardHistory.isEmpty()) {
            String nextUrl = forwardHistory.removeFirst();
            history.addLast(nextUrl);
            return nextUrl;
        } else {
            return "No more pages to go forward.";
        }
    }

    public static void main(String[] args) {
        BrowserHistory browser = new BrowserHistory();

        // 访问一些网址
        browser.visit("google.com");
        browser.visit("youtube.com");
        browser.visit("facebook.com");

        // 后退到上一个页面
        System.out.println("Back: " + browser.back());

        // 再次后退
        System.out.println("Back again: " + browser.back());

        // 前进到下一个页面
        System.out.println("Forward: " + browser.forward());
    }
}

表格说明

浏览器历史记录操作记录
操作描述当前页面历史记录(从左至右为最新至最旧)前进历史(从左至右为最近至最远)
初始化创建一个新的浏览器历史管理器N/A[][]
访问(visit)访问“google.com”google.com[“google.com”][]
访问(visit)访问“youtube.com”youtube.com[“google.com”, “youtube.com”][]
访问(visit)访问“facebook.com”facebook.com[“google.com”, “youtube.com”, “facebook.com”][]
后退(back)后退到上一个页面youtube.com[“google.com”, “youtube.com”][“facebook.com”]
后退(back)再次后退google.com[“google.com”][“youtube.com”, “facebook.com”]
前进(forward)前进到下一个页面youtube.com[“google.com”, “youtube.com”][“facebook.com”]

通过这个案例和表格,我们可以看到双端队列如何有效地模拟浏览器的历史记录管理。每当访问一个新的网址时,它会被添加到历史记录的尾部,并清空前进历史。后退操作会从历史记录的尾部移除当前页面并将其添加到前进历史的头部,而前进操作则相反。这种方法保证了高效且直观的网页导航体验。

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

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

相关文章

计算机组成原理学习笔记(一)

计算机组成原理 [类型:: [[计算机基础课程]] ] [来源:: [[B站]] ] [主讲人:: [[咸鱼学长]] ] [评价:: ] [知识点:: [[系统软件]] & [[应用软件]] ] [简单解释:: 管理计算机系统的软件&#xff1b; 按照任务需要编写的程序 ] [问题:: ] [知识点:: [[机器字长]] ] [简单…

盘点2024年6月Sui生态发展,了解Sui近期成长历程

随着区块链技术的迅猛发展&#xff0c;Sui生态在2024年6月取得了令人欣喜的进步。作为创新的L1协议&#xff0c;Sui不仅在技术革新方面表现突出&#xff0c;还在DeFi、游戏应用和开发者工具等领域展现出强大的潜力。本篇文章将全面盘点Sui在过去一个月内的生态发展&#xff0c;…

【融合ChatGPT等AI模型】Python-GEE遥感云大数据分析、管理与可视化及多领域案例实践应用

随着航空、航天、近地空间遥感平台的持续发展&#xff0c;遥感技术近年来取得显著进步。遥感数据的空间、时间、光谱分辨率及数据量均大幅提升&#xff0c;呈现出大数据特征。这为相关研究带来了新机遇&#xff0c;但同时也带来巨大挑战。传统的工作站和服务器已无法满足大区域…

JavaSE第10篇:常用类

文章目录 一、Object1、Object使用2、toString3、equals和4、hashCode5、clone6、finalize7、getClass8、wait、notify和notifyAll 二、使用步骤 一、Object 1、Object使用 Object类是所有Java的根父类 如果在类的声明中未使用extends关键字指明其父类&#xff0c;则默认父类…

如何监控和优化 PostgreSQL 中的连接池使用?

文章目录 一、连接池的基本概念二、监控 PostgreSQL 连接池使用的重要性&#xff08;一&#xff09;性能优化&#xff08;二&#xff09;资源管理&#xff08;三&#xff09;故障排查 三、PostgreSQL 连接池监控指标&#xff08;一&#xff09;活跃连接数&#xff08;二&#x…

数据结构练习

1. 快速排序的非递归是通过栈来实现的&#xff0c;则前序与层次可以通过控制入栈的顺序来实现&#xff0c;因为递归是会一直开辟栈区空间&#xff0c;所以非递归的实现只需要一个栈的大小&#xff0c;而这个大小是小于递归所要的&#xff0c; 非递归与递归的时间复杂度是一样的…

springboot解压文件流zip压缩包

springboot解压文件流zip压缩包 原始文件存储的地方&#xff1a; 需要在当前目录下解压该文件&#xff0c;如下图&#xff1a; 代码示例&#xff1a; private Result<String> getLocationGuideLayerName(YbYstbtqTaskResolveParam params, String fishnetLayerName)…

我们严重低估了MiniMax;扎克伯格站在了奥特曼的对面;欧洲最强大模型的天才创始人;Notion AI在LLM来临时快速转身奔跑 | ShowMeAI

&#x1f440;日报&周刊合集 | &#x1f3a1;生产力工具与行业应用大全 | &#x1f9e1; 点赞关注评论拜托啦&#xff01; 1. MiniMax 创始人闫俊杰&#xff1a;我选的技术路线是上限最高的&#xff0c;几乎没有退路&#xff0c;选的算力方式也激进 MiniMax 官网 → https:…

30多款简洁个人博客网站网页模板演示学习

30多款个人博客个人网站divcss,html在线预览,静态页面模板免费下载.这些简洁和优雅的博客网页模板,为那些想成为创建博客的个人或媒体提供灵感设计。网页模板可以记录旅游、生活方式、食品或摄影博客等网站。 http://www.bokequ.com/blog/1/ http://www.bokequ.com/blog/2/ htt…

近千含分步骤做法图片菜谱ACCESS\EXCEL数据库

菜谱类的数据已经有一些了&#xff0c;比如《近5万份有图菜谱大全》、《3万多条含图片的菜谱资料数据库》、《无图片的2万多条菜谱》、《5000个菜谱食谱大全》、《4千多带图片的美食菜谱数据内容采集》&#xff0c;但是我还是偏向更喜欢有步骤图片的菜谱&#xff0c;比如《2千8…

2025 百度提前批校招内推

百度2025校园招聘内推开始啦&#xff0c;被推荐人可以免笔试直接面试&#xff0c;提前批结果不影响校招&#xff0c;机会1&#xff0c;还可直推心仪部门&#xff0c;可扫描下面二维码或点击链接进行投递&#xff0c;快来投递你心仪的职位吧&#xff08; 网申链接地址 &#xff…

机器学习的遗忘——基于文章“Forgetting“ in Machine Learning and Beyond: A Survey

文章概要 这篇调查文章仅关注选择性遗忘&#xff0c;承认遗忘某些信息可以通过允许模型优先考虑和保留更重要或相关的信息&#xff0c;以及保护用户隐私&#xff0c;从而带来好处。选择性遗忘&#xff08;Selective forgetting&#xff09;涉及有选择地忽略无关或噪声数据。这…

C语言 | Leetcode C语言题解之第220题存在重复元素III

题目&#xff1a; 题解&#xff1a; struct HashTable {int key;int val;UT_hash_handle hh; };int getID(int x, long long w) {return x < 0 ? (x 1ll) / w - 1 : x / w; }struct HashTable* query(struct HashTable* hashTable, int x) {struct HashTable* tmp;HASH_F…

亚马逊如何用自养号测评打造权重提升排名带来更多的自然流量

亚马逊通过自养号测评来提升流量是一种被广泛采用的运营手段&#xff0c;它可以帮助卖家快速提高商品的曝光度和吸引潜在买家。以下是自养号测评的详细分析&#xff1a; 一、自养号测评的定义与原理 自养号测评是指卖家通过注册并管理海外买家账号&#xff0c;对自家商品进行…

PyQT: 开发一款ROI绘制小程序

在一些基于图像或者视频流的应用中&#xff0c;比如电子围栏/客流统计等&#xff0c;我们需要手动绘制一些感兴趣&#xff08;Region of Interest&#xff0c;简称ROI&#xff09;区域。 在这里&#xff0c;我们基于Python和PyQt5框架开发了一款桌面应用程序&#xff0c;允许用…

java中Request和Response的详细介绍

1.Request和Response的概述 # 重点 1. service方法的两个参数request和response是由tomcat创建的void service(ServletRequest var1, ServletResponse var2) 2. request 表示请求数据, tomcat将浏览器发送过来的请求数据解析并封装到request对象中servlet开发者可以通过reques…

AI免费英语学习在线工具:Pi;gpt;其他大模型AI 英语学习智能体工具

1、pi(强烈推荐&#xff1a;可以安卓下载使用) https://pi.ai/talk &#xff08;网络国内使用方便&#xff09; 支持实时聊天与语音对话 2、chatgpt&#xff08;强烈推荐&#xff1a;可以安卓下载使用) https://chat.openai.com/ &#xff08;网络国内使用不方便&#xf…

element-ui el-select选择器组件下拉框增加自定义按钮

element-ui el-select选择器组件下拉框增加自定义按钮 先看效果 原理&#xff1a;在el-select下添加禁用的el-option&#xff0c;将其value绑定为undefined&#xff0c;然后覆盖el-option禁用状态下的默认样式即可 示例代码如下&#xff1a; <template><div class…

27_电子电路设计基础

电路设计 电路板的设计 电路板的设计主要分三个步骤&#xff1a;设计电路原理图、生成网络表、设计印制电路板。 (1)设计电路原理图&#xff1a;将元器件按逻辑关系用导线连接起来。设计原理图的元件来源是“原理图库”,除了元件库外还可以由用户自己增加建立新的元件&#…

@Builder注解详解:巧妙避开常见的陷阱

欢迎来到我的博客&#xff0c;代码的世界里&#xff0c;每一行都是一个故事 &#x1f38f;&#xff1a;你只管努力&#xff0c;剩下的交给时间 &#x1f3e0; &#xff1a;小破站 Builder注解详解&#xff1a;巧妙避开常见的陷阱 前言1. Builder的基本使用使用示例示例类创建对…