牛客NC404 最接近的K个元素【中等 二分查找+双指针 Java/Go/PHP】

2024-05-12 19:20

本文主要是介绍牛客NC404 最接近的K个元素【中等 二分查找+双指针 Java/Go/PHP】,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题目

在这里插入图片描述
题目链接:
https://www.nowcoder.com/practice/b4d7edc45759453e9bc8ab71f0888e0f

知识点

	二分查找;找到第一个大于等于x的数的位置idx;然后从idx开始往两边扩展

Java代码

import java.util.*;public class Solution {/*** 代码中的类名、方法名、参数名已经指定,请勿修改,直接返回方法规定的值即可*** @param nums int整型ArrayList* @param k int整型* @param x int整型* @return int整型ArrayList*/public ArrayList<Integer> closestElement (ArrayList<Integer> nums, int k,int x) {//二分+双指针int n = nums.size();int right = firstGt(nums, x);int left = right - 1;ArrayList<Integer> ans = new ArrayList<>();LinkedList<Integer> ll = new LinkedList<>();if (right == 0) {for (int i = 0; i < k ; i++) {ll.add(nums.get(i));}} else if (right == n - 1) {for (int i = n - 1 - k; i < n ; i++) {ll.add(nums.get(i));}} else {while (left >= 0 || right < n) {int diffleft = -1;int diffright = -1;if (left >= 0) {diffleft = x - nums.get(left);}if (right < n) {diffright = nums.get(right) - x;}if (diffleft != -1 && diffright != -1) {if (diffleft <= diffright) {ll.addFirst(nums.get(left--));} else {ll.addLast(nums.get(right++));}} else if (diffleft != -1) {ll.addFirst(nums.get(left--));} else if (diffright != -1) {ll.addLast(nums.get(right++));}if (ll.size() == k)break;}}return new ArrayList<>(ll);}//找到大于等于x的下标位置public int firstGt(ArrayList<Integer> nums, int x) {int n = nums.size();int left = 0;int right = n;while (left < right) {int m = left + (right - left) / 2;if (nums.get(m) > x) {right = m - 1;} else if (nums.get(m) < x) {left = m + 1;} else {return m;}}return left;}
}

Go代码

package main//import "fmt"/*** 代码中的类名、方法名、参数名已经指定,请勿修改,直接返回方法规定的值即可*** @param nums int整型一维数组* @param k int整型* @param x int整型* @return int整型一维数组*/
func closestElement(nums []int, k int, x int) []int {//二分+双指针n := len(nums)right := firstGt(nums, x)arrleft := []int{}arrright := []int{}ans := []int{}if right == 0 {for i := 0; i < k; i++ {ans = append(ans, nums[i])}} else if right == n-1 {for i := n - 1 - k; i < n; i++ {ans = append(ans, nums[i])}} else {left := right - 1cnt := 0for left >= 0 || right < n {diffleft := -1diffright := -1if left >= 0 {diffleft = x - nums[left]}if right < n {diffright = nums[right] - x}if diffleft != -1 && diffright != -1 {if diffleft <= diffright {arrleft = append(arrleft, nums[left])left--} else {arrright = append(arrright, nums[right])right++}} else if diffleft != -1 {arrleft = append(arrleft, nums[left])left--} else if diffright != -1 {arrright = append(arrright, nums[right])right++}cnt++if cnt == k {for i := len(arrleft) - 1; i >= 0; i-- {ans = append(ans, arrleft[i])}for i := 0; i < len(arrright); i++ {ans = append(ans, arrright[i])}break}}}return ans
}//找到大等于x的位置
func firstGt(nums []int, x int) int {n := len(nums)left := 0right := n - 1for left < right {m := left + (right-left)/2if nums[m] > x {right = m - 1} else if nums[m] < x {left = m + 1} else {return m}}return left
}

PHP代码

<?php/*** 代码中的类名、方法名、参数名已经指定,请勿修改,直接返回方法规定的值即可** * @param nums int整型一维数组 * @param k int整型 * @param x int整型 * @return int整型一维数组*/
function closestElement( $nums ,  $k ,  $x )
{// 二分+双指针$n = count($nums);$right = firstGt($nums,$x);$ans = [];$arrleft=[];$arrright =[];if($right ==0 ){for($i=0;$i<$k;$i++){$ans[$i] = $nums[$i];}}else if($right ==$n-1){for($i=$n-1-$k;$i>=0;$i++){$ans[count($ans)] = $nums[$i];}}else {$left = $right-1;$cnt =0;while ($left>=0 || $right < $n){$diffleft=-1;$diffright =-1;if($left>=0) {$diffleft = $x-$nums[$left];}if($right<$n){$diffright = $nums[$right]-$x;}if($diffleft!=-1 && $diffright!=-1){if($diffleft<=$diffright){$arrleft[count($arrleft)] = $nums[$left--];}else{$arrright[count($arrright)] = $nums[$right++];}}else if($diffleft!=-1){$arrleft[count($arrleft)] = $nums[$left--];}else if($diffright!=-1){$arrright[count($arrright)] = $nums[$right++];}$cnt++;if($cnt==$k){for($i=count($arrleft)-1;$i>=0;$i--){$ans[count($ans)] = $arrleft[$i];}for($i=0;$i<count($arrright);$i++){$ans[count($ans)] = $arrright[$i];}break;}}}return $ans;
}//找到大于等于x的位置
function firstGt($nums,$x){$n = count($nums);$left =0;$right = $n;while ($left<$right){$m = $left+(($right-$left)>> 1);if($nums[$m] > $x){$right = $m-1;}else if($nums[$m] < $x){$left = $m+1;}else{return $m;}}return $left;
}

这篇关于牛客NC404 最接近的K个元素【中等 二分查找+双指针 Java/Go/PHP】的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/983517

相关文章

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Spring Security--Architecture Overview

1 核心组件 这一节主要介绍一些在Spring Security中常见且核心的Java类,它们之间的依赖,构建起了整个框架。想要理解整个架构,最起码得对这些类眼熟。 1.1 SecurityContextHolder SecurityContextHolder用于存储安全上下文(security context)的信息。当前操作的用户是谁,该用户是否已经被认证,他拥有哪些角色权限…这些都被保

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

Java进阶13讲__第12讲_1/2

多线程、线程池 1.  线程概念 1.1  什么是线程 1.2  线程的好处 2.   创建线程的三种方式 注意事项 2.1  继承Thread类 2.1.1 认识  2.1.2  编码实现  package cn.hdc.oop10.Thread;import org.slf4j.Logger;import org.slf4j.LoggerFactory

hdu2241(二分+合并数组)

题意:判断是否存在a+b+c = x,a,b,c分别属于集合A,B,C 如果用暴力会超时,所以这里用到了数组合并,将b,c数组合并成d,d数组存的是b,c数组元素的和,然后对d数组进行二分就可以了 代码如下(附注释): #include<iostream>#include<algorithm>#include<cstring>#include<stack>#include<que

hdu2289(简单二分)

虽说是简单二分,但是我还是wa死了  题意:已知圆台的体积,求高度 首先要知道圆台体积怎么求:设上下底的半径分别为r1,r2,高为h,V = PI*(r1*r1+r1*r2+r2*r2)*h/3 然后以h进行二分 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#includ