JAVA面试——JAVA算法

简介: JAVA面试——JAVA算法

21.1.1.

二分查找

又叫折半查找,要求待查找的序列有序。每次取中间位置的值与待查关键字比较,如果中间位置

的值比待查关键字大,则在前半部分循环这个查找的过程,如果中间位置的值比待查关键字小,

则在后半部分循环这个查找的过程。直到查找到了为止,否则序列中没有待查的关键字。

public static int biSearch(int []array,int a){

int lo=0;

int hi=array.length-1;

int mid;

while(lo<=hi){

mid=(lo+hi)/2;//中间位置

if(array[mid]==a){

return mid+1;

}else if(array[mid]<a){ //向右查找

lo=mid+1;

}else{ //向左查找

hi=mid-1;

}

}

return -1;

}

21.1.2. 冒泡排序算法

1)比较前后相邻的二个数据,如果前面数据大于后面的数据,就将这二个数据交换。

2)这样对数组的第 0 个数据到 N-1 个数据进行一次遍历后,最大的一个数据就“沉”到数组第

N-1 个位置。

3)N=N-1,如果 N 不为 0 就重复前面二步,否则排序完成。

public static void bubbleSort1(int [] a, int n){

int i, j;13/01/2022

Page 233 of 283

for(i=0; i<n; i++){//表示 n 次排序过程

for(j=1; j<n-i; j++){

if(a[j-1] > a[j]){//前面的数字大于后面的数字就交换

//交换 a[j-1]和 a[j]

int temp;

temp = a[j-1];

a[j-1] = a[j];

a[j]=temp;

}

}

}

}

21.1.3. 插入排序算法

通过构建有序序列,对于未排序数据,在已排序序列中从后向前扫描,找到相应的位置并插入。

插入排序非常类似于整扑克牌。在开始摸牌时,左手是空的,牌面朝下放在桌上。接着,一次从

桌上摸起一张牌,并将它插入到左手一把牌中的正确位置上为了找到这张牌的正确位置,要将

它与手中已有的牌从右到左地进行比较。无论什么时候,左手中的牌都是排好序的

如果输入数组已经是排好序的话,插入排序出现最佳情况,其运行时间是输入规模的一个线性函

数。如果输入数组是逆序排列的,将出现最坏情况。平均情况与最坏情况一样,其时间代价是(n2)

image.png

public void sort(int arr[])13/01/2022

Page 234 of 283

{

for(int i =1; i<arr.length;i++)

{

//插入的数

int insertVal = arr[i];

//被插入的位置(准备和前一个数比较)

int index = i-1;

//如果插入的数比被插入的数小

while(index>=0&&insertVal<arr[index])

{

//将把 arr[index] 向后移动

arr[index+1]=arr[index];

//让 index 向前移动

index--;

}

//把插入的数放入合适位置

arr[index+1]=insertVal;

}

}

21.1.4.

快速排序算法

快速排序的原理:选择一个关键值作为基准值。比基准值小的都在左边序列(一般是无序的),

比基准值大的都在右边(一般是无序的)。一般选择序列的第一个元素。

一次循环:从后往前比较,用基准值和最后一个值比较,如果比基准值小的交换位置,如果没有

继续比较下一个,直到找到第一个比基准值小的值才交换。找到这个值之后,又从前往后开始比

,如果有比基准值大的,交换位置,如果没有继续比较下一个,直到找到第一个比基准值大的

值才交换。直到从前往后的比较索引>从后往前比较的索引,结束第一次循环,此时,对于基准值

来说,左右两边就是有序的了。

public void sort(int[] a,int low,int high){

int start = low;

int end = high;13/01/2022

Page 235 of 283

int key = a[low];

while(end>start){

//从后往前比较

while(end>start&&a[end]>=key)

//如果没有比关键值小的,比较下一个,直到有比关键值小的交换位置,然后又从前往后比较

end--;

if(a[end]<=key){

int temp = a[end];

a[end] = a[start];

a[start] = temp;

}

//从前往后比较

while(end>start&&a[start]<=key)

//如果没有比关键值大的,比较下一个,直到有比关键值大的交换位置

start++;

if(a[start]>=key){

int temp = a[start];

a[start] = a[end];

a[end] = temp;

}

//此时第一次循环比较结束,关键值的位置已经确定了。左边的值都比关键值小,右边的

值都比关键值大,但是两边的顺序还有可能是不一样的,进行下面的递归调用

}

//递归

if(start>low) sort(a,low,start-1);//左边序列。第一个索引位置到关键值索引-1

if(end<high) sort(a,end+1,high);//右边序列。从关键值索引+1 到最后一个

}

}

image.png

21.1.1. 希尔排序算法

基本思想:先将整个待排序的记录序列分割成为若干子序列分别进行直接插入排序,待整个序列

中的记录“基本有序”时,再对全体记录进行依次直接插入排序。

1. 操作方法:

选择一个增量序列 t1,t2,…,tk,其中 ti>tj,tk=1;

2. 按增量序列个数 k,对序列进行 k 趟排序;

3. 每趟排序,根据对应的增量 ti,将待排序列分割成若干长度为 m 的子序列,分别对各子表进

行直接插入排序。仅增量因子为1 时,整个序列作为一个表来处理,表长度即为整个序列的长

度。

image.png

private void shellSort(int[] a) {

int dk = a.length/2;

while( dk >= 1 ){

ShellInsertSort(a, dk);

dk = dk/2;

}

}

private void ShellInsertSort(int[] a, int dk) {

//类似插入排序,只是插入排序增量是 1,这里增量是 dk,把 1 换成 dk 就可以了

for(int i=dk;i<a.length;i++){

if(a[i]<a[i-dk]){

int j;

int x=a[i];//x 为待插入元素

a[i]=a[i-dk];

for(j=i-dk; j>=0 && x<a[j];j=j-dk){

//通过循环,逐个后移一位找到要插入的位置。

a[j+dk]=a[j];

}

a[j+dk]=x;//插入

}

}

}

21.1.2. 归并排序算法

归并(Merge)排序法是将两个(或两个以上)有序表合并成一个新的有序表,即把待排序序列

分为若干个子序列,每个子序列是有序的。然后再把有序子序列合并为整体有序序列。

image.png

public class MergeSortTest {

public static void main(String[] args) {

int[] data = new int[] { 5, 3, 6, 2, 1, 9, 4, 8, 7 };

print(data);

mergeSort(data);

System.out.println("排序后的数组:");

print(data);

}

public static void mergeSort(int[] data) {

sort(data, 0, data.length - 1);

}

public static void sort(int[] data, int left, int right) {

if (left >= right)

return;

// 找出中间索引

int center = (left + right) / 2;

// 对左边数组进行递归

sort(data, left, center);

// 对右边数组进行递归

sort(data, center + 1, right);

// 合并

merge(data, left, center, right);

print(data);

}

/**

* 将两个数组进行归并,归并前面 2 个数组已有序,归并后依然有序

*

* @param data

* 数组对象

* @param left

* 左数组的第一个元素的索引

* @param center

* 左数组的最后一个元素的索引,center+1 是右数组第一个元素的索引

* @param right

* 右数组最后一个元素的索引

*/

public static void merge(int[] data, int left, int center, int right) {

// 临时数组

int[] tmpArr = new int[data.length];

// 右数组第一个元素索引

int mid = center + 1;

// third 记录临时数组的索引

int third = left;

// 缓存左数组第一个元素的索引

int tmp = left;

while (left <= center && mid <= right) {

// 从两个数组中取出最小的放入临时数组

if (data[left] <= data[mid]) {

tmpArr[third++] = data[left++];

} else {

tmpArr[third++] = data[mid++];

}

}

// 剩余部分依次放入临时数组(实际上两个 while 只会执行其中一个)

while (mid <= right) {

tmpArr[third++] = data[mid++]; 13/01/2022

Page 240 of 283

}

while (left <= center) {

tmpArr[third++] = data[left++];

}

// 将临时数组中的内容拷贝回原数组中

// (原 left-right 范围的内容被复制回原数组)

while (tmp <= right) {

data[tmp] = tmpArr[tmp++];

}

}

public static void print(int[] data) {

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

System.out.print(data[i] + "\t");

}

System.out.println();

}

}

21.1.3. 桶排序算法

桶排序的基本思想是: 把数组 arr 划分为 n 个大小相同子区间(桶),每个子区间各自排序,最

后合并 。计数排序是桶排序的一种特殊情况,可以把计数排序当成每个桶里只有一个元素的情况。

1.找出待排序数组中的最大值 max、最小值 min

2.我们使用 动态数组 ArrayList 作为桶,桶里放的元素也用 ArrayList 存储。桶的数量为(max

min)/arr.length+1

3.遍历数组 arr,计算每个元素 arr[i] 放的桶

4.每个桶各自排序

public static void bucketSort(int[] arr){

int max = Integer.MIN_VALUE;

int min = Integer.MAX_VALUE;

for(int i = 0; i < arr.length; i++){13/01/2022

Page 241 of 283

max = Math.max(max, arr[i]);

min = Math.min(min, arr[i]);

}

//创建桶

int bucketNum = (max - min) / arr.length + 1;

ArrayList<ArrayList<Integer>> bucketArr = new ArrayList<>(bucketNum);

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

bucketArr.add(new ArrayList<Integer>());

}

//将每个元素放入桶

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

int num = (arr[i] - min) / (arr.length);

bucketArr.get(num).add(arr[i]);

}

//对每个桶进行排序

for(int i = 0; i < bucketArr.size(); i++){

Collections.sort(bucketArr.get(i));

}

}

21.1.4. 基数排序算法

将所有待比较数值(正整数)统一为同样的数位长度,数位较短的数前面补零。然后,从最低位

开始,依次进行一次排序。这样从最低位排序一直到最高位排序完成以后,数列就变成一个有序序

列。

public class radixSort {

inta[]={49,38,65,97,76,13,27,49,78,34,12,64,5,4,62,99,98,54,101,56,17,18,23,34,15,35,2

5,53,51};

public radixSort(){

sort(a);

for(inti=0;i<a.length;i++){

System.out.println(a[i]);

}

}

public void sort(int[] array){

//首先确定排序的趟数;

int max=array[0];

for(inti=1;i<array.length;i++){

if(array[i]>max){13/01/2022

Page 242 of 283

max=array[i];

}

}

int time=0;

//判断位数;

while(max>0){

max/=10;

time++;

}

//建立 10 个队列;

List<ArrayList> queue=newArrayList<ArrayList>();

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

ArrayList<Integer>queue1=new ArrayList<Integer>();

queue.add(queue1);

}

//进行 time 次分配和收集;

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

//分配数组元素;

for(intj=0;j<array.length;j++){

//得到数字的第 time+1 位数;

int x=array[j]%(int)Math.pow(10,i+1)/(int)Math.pow(10, i);

ArrayList<Integer>queue2=queue.get(x);

queue2.add(array[j]);

queue.set(x, queue2);

}

int count=0;//元素计数器;

//收集队列元素;

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

while(queue.get(k).size()>0){

ArrayList<Integer>queue3=queue.get(k);

array[count]=queue3.get(0);

queue3.remove(0);

count++;

}

}

}

}

}13/01/2022

Page 243 of 283

21.1.5. 剪枝算法

在搜索算法中优化中,剪枝,就是通过某种判断,避免一些不必要的遍历过程,形象的说,就是

剪去了搜索树中的某些“枝条”,故称剪枝。应用剪枝优化的核心问题是设计剪枝判断方法,即

确定哪些枝条应当舍弃,哪些枝条应当保留的方法。

image.png

21.1.6. 回溯算法

回溯算法实际上一个类似枚举的搜索尝试过程,主要是在搜索尝试过程中寻找问题的解,当发现

已不满足求解条件时,就“回溯”返回,尝试别的路径。

21.1.7. 最短路径算法

从某顶点出发,沿图的边到达另一顶点所经过的路径中,各边上权值之和最小的一条路径叫做最

短路径。解决最短路的问题有以下算法,Dijkstra 算法,Bellman-Ford 算法,Floyd 算法和 SPFA

算法等。

21.1.8. 最大子数组算法

21.1.9. 最长公共子序算法

21.1.10. 最小生成树算法

现在假设有一个很实际的问题:我们要在 n 个城市中建立一个通信网络,则连通这 n 个城市需要

布置 n-1 一条通信线路,这个时候我们需要考虑如何在成本最低的情况下建立这个通信网?

于是我们就可以引入连通图来解决我们遇到的问题,n 个城市就是图上的 n 个顶点,然后,边表示

两个城市的通信线路,每条边上的权重就是我们搭建这条线路所需要的成本,所以现在我们有 n 个

顶点的连通网可以建立不同的生成树,每一颗生成树都可以作为一个通信网,当我们构造这个连

通网所花的成本最小时,搭建该连通网的生成树,就称为最小生成树。13/01/2022

Page 244 of 283

构造最小生成树有很多算法,但是他们都是利用了最小生成树的同一种性质:MST 性质(假设

N=(V,{E})是一个连通网,U 是顶点集 V 的一个非空子集,如果(

u,v)是一条具有最小权值的边,

其中 u 属于 U,v 属于 V-U,则必定存在一颗包含边(

u,v)的最小生成树),下面就介绍两种使

用 MST 性质生成最小生成树的算法:普里姆算法和克鲁斯卡尔算法。

image.png


目录
相关文章
|
4天前
|
存储 算法 安全
探究‘公司禁用 U 盘’背后的哈希表算法与 Java 实现
在数字化办公时代,信息安全至关重要。许多公司采取“禁用U盘”策略,利用哈希表算法高效管理外接设备的接入权限。哈希表通过哈希函数将设备标识映射到数组索引,快速判断U盘是否授权。例如,公司预先将允许的U盘标识存入哈希表,新设备接入时迅速验证,未授权则禁止传输并报警。这有效防止恶意软件和数据泄露,保障企业信息安全。 代码示例展示了如何用Java实现简单的哈希表,模拟公司U盘管控场景。哈希表不仅用于设备管理,还在文件索引、用户权限等多方面助力信息安全防线的构建,为企业数字化进程保驾护航。
|
12天前
|
监控 算法 网络协议
Java 实现局域网电脑屏幕监控算法揭秘
在数字化办公环境中,局域网电脑屏幕监控至关重要。本文介绍用Java实现这一功能的算法,涵盖图像采集、数据传输和监控端显示三个关键环节。通过Java的AWT/Swing库和Robot类抓取屏幕图像,使用Socket进行TCP/IP通信传输图像数据,并利用ImageIO类在监控端展示图像。整个过程确保高效、实时和准确,为提升数字化管理提供了技术基础。
48 15
|
2月前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
100 2
|
4天前
|
运维 监控 算法
企业局域网监控软件中 Java 优先队列算法的核心优势
企业局域网监控软件是数字化时代企业网络安全与高效运营的基石,犹如一位洞察秋毫的卫士。通过Java实现的优先队列算法,它能依据事件优先级排序,确保关键网络事件如异常流量、数据泄露等被优先处理,保障系统稳定与安全。代码示例展示了如何定义网络事件类并使用PriorityQueue处理高优先级事件,尤其在面对疑似风险时迅速启动应急措施。这一核心技术助力企业在复杂网络环境中稳健前行,护航业务腾飞。
49 32
|
2天前
|
监控 Dubbo Java
Java Dubbo 面试题
Java Dubbo相关基础面试题
|
3天前
|
存储 监控 算法
探秘局域网桌面监控:深入剖析 Java 语言核心算法
在数字化办公时代,局域网桌面监控如同企业的“智慧鹰眼”,确保工作效率与数据安全。本文以Java为载体,揭示哈希表在监控中的关键应用。通过高效的数据结构和算法,哈希表能快速索引设备连接信息,大幅提升监控的时效性和响应速度。代码示例展示了如何用Java实现设备网络连接监控,结合未来技术如AI、大数据,展望更智能的监控体系,助力企业在数字化浪潮中稳健前行。
|
2天前
|
SQL Java 数据库连接
Java MyBatis 面试题
Java MyBatis相关基础面试题
|
2天前
|
存储 监控 算法
Java JVM 面试题
Java JVM(虚拟机)相关基础面试题
|
2天前
|
安全 架构师 Java
Java大厂面试高频:Collection 和 Collections 到底咋回答?
Java中的`Collection`和`Collections`是两个容易混淆的概念。`Collection`是集合框架的根接口,定义了集合的基本操作方法,如添加、删除等;而`Collections`是一个工具类,提供了操作集合的静态方法,如排序、查找、同步化等。简单来说,`Collection`关注数据结构,`Collections`则提供功能增强。通过小王的面试经历,我们可以更好地理解这两者的区别及其在实际开发中的应用。希望这篇文章能帮助你掌握这个经典面试题。
17 4
|
2天前
|
SQL 监控 druid
Java Druid 面试题
Java Druid 连接池相关基础面试题