【算法】【优选算法】优先级队列

【算法】【优选算法】优先级队列

目录

一、1046.最后一块石头的重量

题目链接:1046.最后一块石头的重量
题目描述:

题目解析:

  • 题意就是让我们拿出提供的数组的最大两个值,大减小作差,将差值再放入数组,直到数组空了或者只有一个元素为止。

解题思路:

  • 题目要求我们在一个乱序的数组中找最大两个值,我们首先想到数组排序,但是由于我们还需要将差值放入数组,我们放一次就需要排序一次。
  • 使用优先级队列,大根堆,开销会小一些,我们只需要每次拿堆顶元素即可。

解题代码:

//时间复杂度 O(n)//空间复杂度 O(n)classSolution{publicintlastStoneWeight(int[] stones){//创建大根堆PriorityQueue<Integer> queue =newPriorityQueue<>((a,b)-> b - a );//入堆for(int i =0; i < stones.length; i++) queue.offer(stones[i]);//执行逻辑while(!queue.isEmpty()&& queue.size()!=1){int y = queue.poll();int x = queue.poll(); queue.offer(y-x);}//返回值return queue.isEmpty()?0: queue.poll();}}

二、703. 数据流中的第 K 大元素

题目链接:703. 数据流中的第 K 大元素

题目描述:

题目解析:

  • 给我们一个数组和一个数K,让我们在使用类的add方法后,返回数组中的第K大的数。
    解题思路:
  • 我们使用一个大小为K的小根堆,那么我们剩下在堆中的数,就是数组中第K大到最大的值。
  • 返回数组中的第K大的数,就是当前的堆顶。

解题代码:

//时间复杂度 O(nLogK)//空间复杂度 O(K)classKthLargest{PriorityQueue<Integer> heap;int param_1;publicKthLargest(int k,int[] nums){ heap =newPriorityQueue<Integer>(); param_1 = k;for(int i =0; i < nums.length; i++){ heap.offer(nums[i]);if(heap.size()> param_1){ heap.poll();}}}publicintadd(int val){ heap.offer(val);if(heap.size()> param_1){ heap.poll();}return heap.peek();}}

三、692. 前 K 个⾼频单词

题目链接:692. 前 K 个⾼频单词

题目描述:

题目解析:

  • 给我们一个words的字符串数组,让我们返回数组中出现的频率次数最多到第K多的字符串。
  • 当出现频次相同的时候,就直接按照字典顺序,字母前到后比较大小,大在前。

解题思路:

  • 我们使用一个hash表,记录下字符串与其出现的次数。
  • 在使用一个大小为K的堆,当我们的频次就是hash表中的value相同的时候,我们使用compare比较大小,创建的是大根堆,其余比较频次是小根堆。总体上看还是一个小根堆。
  • 最后一次取出堆中的字符串即可,但是由于返回值又是从小到大,最后将结果数组逆序即可。
    解题代码:
//时间复杂度:O(NLogK)//空间复杂U度:O(N)classSolution{publicList<String>topKFrequent(String[] words,int k){Map<String,Integer> hash =newHashMap<>();PriorityQueue<Pair<String,Integer>> heap =newPriorityQueue<>((a,b)->{//频次相同大根堆if(a.getValue().equals(b.getValue())){return b.getKey().compareTo(a.getKey());}//小根堆return a.getValue()- b.getValue();});//hash初始化for(String s : words){ hash.put(s, hash.getOrDefault(s ,0)+1);}//入堆for(Map.Entry<String,Integer> e : hash.entrySet()){ heap.offer(newPair<>(e.getKey(),e.getValue()));if(heap.size()> k){ heap.poll();}}//结果处理List<String> ret =newArrayList<String>();while(!heap.isEmpty()){ ret.add(heap.poll().getKey());}//逆置Collections.reverse(ret);return ret;}}

四、295. 数据流的中位数

题目链接:295. 数据流的中位数

题目描述:

  • 就是让我们实现一个类,有初始化,添加元素(每次添加一个),查看元素中位数

题目解析:

  • 我们只需要每次拿取类中的元素的时候,能够直接拿到中位数即可。
  • 我们可以使用两个堆,小根堆记录数的中位数之后的部分,大根堆记录中位数的前半部分。
  • 这样当元素个数是偶数个的时候,我们直接拿到两个堆的堆顶元素即可。为奇数个元素的时候,直接取出堆元素多的那个的堆顶元素即可。

解题思路:

  • 我们使用两个堆,一个大根堆,一个小根堆,在记录下当前的元素个数。
  • 当插入元素后,元素个数为偶数:
    • 当插入元素比大根堆堆顶元素大:
      • 大根堆中元素个数比小根堆多:直接将待插入元素插入小根堆即可。
      • 大根堆中元素个数比小根堆少:将小根堆堆顶元素和待插入元素较小值,插入大根堆。另一个给小根堆。
    • 当插入元素比大根堆堆顶元素小:
      • 大根堆中元素个数比小根堆多:将大根堆堆顶元素插入小根堆。待插入元素给大根堆。
      • 大根堆中元素个数比小根堆少:直接将待插入元素插入大根堆即可。
  • 当插入元素后,元素个数为奇数:
    • 当插入元素比大根堆堆顶元素大:插入小根堆。
    • 当插入元素比大根堆堆顶元素小:插入大根堆。

解题代码:

//时间复杂度:O(LogN)//空间复杂度:O(N)classMedianFinder{//列表中元素个数int n =0;//大根堆记录前半部分值PriorityQueue<Integer> big;//小根堆记录后半部分值PriorityQueue<Integer> little;publicMedianFinder(){ big =newPriorityQueue<>((a,b)->{return b-a;}); little =newPriorityQueue<>();}publicvoidaddNum(int num){ n +=1;if(n ==1){ big.offer(num);return;}//元素个数为偶数,比前面的大if(n %2==0&& big.peek()<= num){//保持前后数据平衡if(big.size()< little.size()){//比后面小if(!little.isEmpty()&& little.peek()>= num){ big.offer(num);}else{int tmp = little.poll(); big.offer(tmp); little.offer(num);}}else{ little.offer(num);}return;}//元素个数为偶数,比前面的小if(n %2==0&& big.peek()> num){//保持前后数据平衡if(big.size()< little.size()){ big.offer(num);}else{int tmp = big.poll(); big.offer(num); little.offer(tmp);}return;}//元素个数为奇数,比前面小if(n %2!=0&& big.peek()>= num){ big.offer(num);}else{ little.offer(num);}}publicdoublefindMedian(){if(n %2==0){return(double)((big.peek()+ little.peek())/2.0);}if(big.size()> little.size()){return big.peek();}else{return little.peek();}}}

Read more

Flutter 组件 mock_client 的适配 鸿蒙Harmony 实战 - 驾驭 HTTP 协议级测试模拟、实现鸿蒙端离线环境下的接口断言与质量门禁方案

Flutter 组件 mock_client 的适配 鸿蒙Harmony 实战 - 驾驭 HTTP 协议级测试模拟、实现鸿蒙端离线环境下的接口断言与质量门禁方案

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net Flutter 组件 mock_client 的适配 鸿蒙Harmony 实战 - 驾驭 HTTP 协议级测试模拟、实现鸿蒙端离线环境下的接口断言与质量门禁方案 前言 在鸿蒙(OpenHarmony)生态的大型分布式政务办公系统、极繁金融交易内核以及对交互逻辑边界有极致审计要求的各种工业级应用开发中,“测试的确定性”是架构设计的定海神针。面对包含多级鉴权、动态速率限制(Rate Limiting)且环境依赖极重的后端 API。如果仅仅依靠真实的沙箱网络环境进行逻辑验收。那么不仅会导致测试套件由于网络波动引发由于非预期超时导致的频繁“假失败(Flaky Tests)”,更会因为无法模拟极端错误分位(如:服务器 500 时特定的 Payload 反馈)产生严重的代码覆盖率黑洞。 我们需要一种“契约自洽、逻辑伪造”的协议审计艺术。 mock_client(通常作为

By Ne0inhk

在 Mac 上完美配置 VSCode 的 C/C++ 开发环境(GCC/G++ 详细教程 )

本文手把手教你如何在 macOS 系统上配置 VSCode 的 C/C++ 开发环境,解决各种常见问题,让你轻松开启 C/C++ 编程之旅! 前言 作为程序员,一个顺手的开发环境至关重要。VSCode 作为轻量级但功能强大的代码编辑器,配合 GCC/G++ 编译器,能够在 Mac 上提供优秀的 C/C++ 开发体验。本文将详细介绍从零开始的完整配置过程。 一、环境准备:安装编译工具 1.1 安装 Xcode Command Line Tools(推荐首选) 打开终端,执行以下命令: xcode-select --install 执行后会弹出安装对话框,点击"安装"即可。

By Ne0inhk
Flutter for OpenHarmony:Flutter 三方库 confetti 为你的应用庆典瞬间注入极其灵动的漫天彩带惊喜互动反馈引擎(适配鸿蒙 HarmonyOS Next ohos)

Flutter for OpenHarmony:Flutter 三方库 confetti 为你的应用庆典瞬间注入极其灵动的漫天彩带惊喜互动反馈引擎(适配鸿蒙 HarmonyOS Next ohos)

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net。 Flutter for OpenHarmony:Flutter 三方库 confetti 为你的应用庆典瞬间注入极其灵动的漫天彩带惊喜(互动反馈引擎)(适配鸿蒙 HarmonyOS Next ohos) 前言 在电商大促夺魁、游戏升级或是任务达成时,一套精美的“彩带喷发”动画往往能让用户的多巴胺瞬间翻倍。confetti 库为 Flutter 开发者提供了一套高性能、可高度定制的物理粒子系统。在本文中,我们将探讨如何在 OpenHarmony 环境下部署这一“氛围大师”,让你的鸿蒙应用关键节点充满惊喜。 1. 为什么你的应用需要 confetti? 心理学研究表明,适时的正向视觉反馈能显著提高用户的留存率。confetti 的优势在于: * 纯物理模拟:每一片彩带的旋转、飘落和路径均符合物理重力公式,表现极其自然。 * 高性能粒子引擎:即使在屏幕上同时渲染数百个碎片,依然能保持满帧运行。 * 全平台一致性:不依赖原生动画库,

By Ne0inhk
Flutter 三方库 json_extractor 的鸿蒙化适配指南 - 支持声明式 JSON 数据提取、复杂嵌套结构解析与强类型转换

Flutter 三方库 json_extractor 的鸿蒙化适配指南 - 支持声明式 JSON 数据提取、复杂嵌套结构解析与强类型转换

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net Flutter 三方库 json_extractor 的鸿蒙化适配指南 - 支持声明式 JSON 数据提取、复杂嵌套结构解析与强类型转换 前言 在 Flutter for OpenHarmony 的日常开发中,处理后端返回的“排山倒海”般的 JSON 数据是每个开发者的必经之路。虽然 json_serializable 很强大,但如果你只需要从一个极其庞大且嵌套复杂的 JSON 中提取特定的几个字段,定义完整的 Model 类就显得过于繁琐。json_extractor 提供了一种基于声明式路径的轻量级提取方案。本文将指导大家如何在鸿蒙端利用该库高效“榨取”JSON 数据。 一、原理解析 / 概念介绍 1.1 基础原理 json_

By Ne0inhk