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

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

目录

一、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

“现在的AI就像1880年的笨重工厂!”微软CSO斯坦福泼冷水:别急着造神

“现在的AI就像1880年的笨重工厂!”微软CSO斯坦福泼冷水:别急着造神

大模型仍未对上商业的齿轮? 编译 | 王启隆 来源 | youtu.be/aWqfH0aSGKI 出品丨AI 科技大本营(ID:rgznai100) 现在的硅谷,空气里都飘着一股“再不上车就晚了”的焦躁感。 最近 OpenClaw 风头正旺,强势登顶 GitHub,终结了 React 神话,许多人更是觉得“AI 自己干活赚钱”的日子就在明天了。 特别是在斯坦福商学院(GSB)这种地方,台下坐着的都是成天琢磨怎么用下一个技术风口搞个独角兽出来的狠人。 微软的首席科学官(CSO)Eric Horvitz 被请到了这个几乎全美最想用 AI 变现的礼堂里。作为从上世纪 80 年代就开始搞 AI 的绝对老炮、也是微软技术底座的“扫地僧”,这位老哥并没有顺着台下的胃口,去吹捧下个月大模型又要颠覆什么行业,而是兜头给大家浇了一盆带点学术味的冷水。 他讲了一个挺有画面感的比喻:大家都在聊

By Ne0inhk
Godot被AI代码“围攻”!维护者崩溃发声:“不知道还能坚持多久”

Godot被AI代码“围攻”!维护者崩溃发声:“不知道还能坚持多久”

整理 | 郑丽媛 出品 | ZEEKLOG(ID:ZEEKLOGnews) 当大模型能在几秒钟内生成一段“看起来像那么回事”的补丁时,开源社区却开始付出另一种代价。 最近,开源游戏引擎 Godot 的核心维护团队公开吐槽:他们正被大量“AI 生成的低质量代码”淹没。那些代码往往结构完整、注释齐全、描述洋洋洒洒,但真正的问题是——提交者可能并不理解自己交上来的内容。 这件事,并不是简单的“有人偷懒用 AI 写代码”。它正在触及开源协作最核心的东西:信任。 一场悄无声息的“AI 洪水” 事情的导火索来自一条 Bluesky 讨论帖。 Godot 主要维护者之一、同时也是 Godot 商业支持公司 W4 Games 联合创始人的 Rémi Verschelde 表示,所谓的“AI slop”

By Ne0inhk
诺奖得主辛顿最新访谈:1 万个 AI 可以瞬间共享同一份“灵魂”,这就是为什么人类注定被超越

诺奖得主辛顿最新访谈:1 万个 AI 可以瞬间共享同一份“灵魂”,这就是为什么人类注定被超越

当宇宙级的“嘴炮”遇到降维打击。 编译 | 王启隆 来源 | youtu.be/l6ZcFa8pybE 出品丨AI 科技大本营(ID:rgznai100) 打开最新一期知名播客 StarTalk 的 YouTube 评论区,最高赞的一条留言是这样写的: “我长这么大,第一次看到尼尔·德葛司·泰森(Neil deGrasse Tyson)在一档节目里几乎全程闭嘴,像个手足无措的小学生一样乖乖听讲。” 作为全美最知名的天体物理学家,泰森平时的画风是充满激情、喋喋不休、用宇宙的宏大来震撼嘉宾。但这一次,坐在他对面的那位满头银发、带着温和英音的英国老人,仅仅用最平淡的语气,就让整个演播室陷入了数次令人窒息的沉默。 这位老人是 Geoffrey Hinton。深度学习三巨头之一,2024 年诺贝尔物理学奖得主,被公认为“AI 教父”。 对经常阅读 Hinton 演讲的我来说,这也是比较新奇的一幕—

By Ne0inhk
48小时“烧光”56万!三人创业团队濒临破产,仅因Gemini API密钥被盗:“AI账单远超我们的银行余额”

48小时“烧光”56万!三人创业团队濒临破产,仅因Gemini API密钥被盗:“AI账单远超我们的银行余额”

整理 | 苏宓 出品 | ZEEKLOG(ID:ZEEKLOGnews) 「仅过了 48 小时,一笔 8.2 万美元的天价费用凭空出现,较这家小型初创公司的正常月费暴涨近 46000%。」 这不是假设的虚幻故事,而是一家墨西哥初创公司正在经历的真实危机。 近日,一位名为 RatonVaquero 的开发者在 Reddit 发帖求助称,由于他的 Gemini API 密钥被盗用,原本每月仅约 180 美元(约 1242 元)的费用,在短短 48 小时内暴涨到 82,314.44 美元(约 56.8 万元)。对于这家只有三名开发者的小型创业团队来说,这笔突如其来的账单,几乎等同于灭顶之灾。 “我现在整个人都处在震惊和恐慌之中。”RatonVaquero

By Ne0inhk