面试题——算法

Java算法面试题

二分查找

又叫折半查找,要求待查找的序列有序。每次取中间位置的值与待查关键字比较,如果中间位置的值比待查关键字大,则在前半部分循环这个查找的过程,如果中间位置的值比待查关键字小,则在后半部分循环这个查找的过程。直到查找到了为止,否则序列中没有待查的关键字。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
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;
}

冒泡排序

(1)比较前后相邻的二个数据,如果前面数据大于后面的数据,就将这二个数据交换。 (2)这样对数组的第0 个数据到N-1 个数据进行一次遍历后,最大的一个数据就“沉”到数组第N-1 个位置。 (3)N=N-1,如果N 不为0 就重复前面二步,否则排序完成。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
public static void bubbleSort1(int [] a, int n){
	int i, j;
	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;
			}
		}
	}
}

插入排序

通过构建有序序列,对于未排序数据,在已排序序列中从后向前扫描,找到相应的位置并插入。插入排序非常类似于整扑克牌。在开始摸牌时,左手是空的,牌面朝下放在桌上。接着,一次从桌上摸起一张牌,并将它插入到左手一把牌中的正确位置上。为了找到这张牌的正确位置,要将它与手中已有的牌从右到左地进行比较。无论什么时候,左手中的牌都是排好序的。如果输入数组已经是排好序的话,插入排序出现最佳情况,其运行时间是输入规模的一个线性函数。如果输入数组是逆序排列的,将出现最坏情况。平均情况与最坏情况一样,其时间代价是(n2)。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
public void sort(int arr[])
{
	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. 快速排序 快速排序的原理:选择一个关键值作为基准值。比基准值小的都在左边序列(一般是无序的),比基准值大的都在右边(一般是无序的)。一般选择序列的第一个元素。 一次循环:从后往前比较,用基准值和最后一个值比较,如果比基准值小的交换位置,如果没有继续比较下一个,直到找到第一个比基准值小的值才交换。找到这个值之后,又从前往后开始比较,如果有比基准值大的,交换位置,如果没有继续比较下一个,直到找到第一个比基准值大的值才交换。直到从前往后的比较索引>从后往前比较的索引,结束第一次循环,此时,对于基准值来说,左右两边就是有序的了。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
public void sort(int[] a,int low,int high){
	int start = low;
	int end = high;
	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 到最后一个
	}
}

希尔排序

基本思想:先将整个待排序的记录序列分割成为若干子序列分别进行直接插入排序,待整个序列中的记录“基本有序”时,再对全体记录进行依次直接插入排序。

  1. 操作方法: 选择一个增量序列t1,t2,…,tk,其中ti>tj,tk=1;
  2. 按增量序列个数k,对序列进行k 趟排序;
  3. 每趟排序,根据对应的增量ti,将待排序列分割成若干长度为m 的子序列,分别对各子表进行直接插入排序。仅增量因子为1 时,整个序列作为一个表来处理,表长度即为整个序列的长度。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
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;//插入
		}
	}
}

归并排序

归并(Merge)排序法是将两个(或两个以上)有序表合并成一个新的有序表,即把待排序序列分为若干个子序列,每个子序列是有序的。然后再把有序子序列合并为整体有序序列。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
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++];
		}
		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();
	}
}

桶排序

桶排序的基本思想是: 把数组 arr 划分为n 个大小相同子区间(桶),每个子区间各自排序,最后合并 。计数排序是桶排序的一种特殊情况,可以把计数排序当成每个桶里只有一个元素的情况。

  1. 找出待排序数组中的最大值max、最小值min
  2. 我们使用 动态数组ArrayList 作为桶,桶里放的元素也用 ArrayList 存储。桶的数量为(maxmin)/arr.length+1
  3. 遍历数组 arr,计算每个元素 arr[i] 放的桶
  4. 每个桶各自排序
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
public static void bucketSort(int[] arr){
	int max = Integer.MIN_VALUE;
	int min = Integer.MAX_VALUE;
	for(int i = 0; i < arr.length; i++){
		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));
	}
}

基数排序

将所有待比较数值(正整数)统一为同样的数位长度,数位较短的数前面补零。然后,从最低位开始,依次进行一次排序。这样从最低位排序一直到最高位排序完成以后,数列就变成一个有序序列。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
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,25,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){
				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++;
				}
			}
		}
	}
}

剪枝算法

在搜索算法中优化中,剪枝,就是通过某种判断,避免一些不必要的遍历过程,形象的说,就是剪去了搜索树中的某些“枝条”,故称剪枝。应用剪枝优化的核心问题是设计剪枝判断方法,即确定哪些枝条应当舍弃,哪些枝条应当保留的方法。 剪枝算法

回溯算法

回溯算法实际上一个类似枚举的搜索尝试过程,主要是在搜索尝试过程中寻找问题的解,当发现已不满足求解条件时,就“回溯”返回,尝试别的路径。

最短路径算法

从某顶点出发,沿图的边到达另一顶点所经过的路径中,各边上权值之和最小的一条路径叫做最短路径。解决最短路的问题有以下算法,Dijkstra 算法,Bellman-Ford 算法,Floyd 算法和SPFA算法等。

最小生成树

现在假设有一个很实际的问题:我们要在n 个城市中建立一个通信网络,则连通这n 个城市需要布置n-1 一条通信线路,这个时候我们需要考虑如何在成本最低的情况下建立这个通信网? 于是我们就可以引入连通图来解决我们遇到的问题,n 个城市就是图上的n 个顶点,然后,边表示两个城市的通信线路,每条边上的权重就是我们搭建这条线路所需要的成本,所以现在我们有n 个顶点的连通网可以建立不同的生成树,每一颗生成树都可以作为一个通信网,当我们构造这个连通网所花的成本最小时,搭建该连通网的生成树,就称为最小生成树。

构造最小生成树有很多算法,但是他们都是利用了最小生成树的同一种性质:MST 性质(假设N=(V,{E})是一个连通网,U 是顶点集V 的一个非空子集,如果(u,v)是一条具有最小权值的边,其中u 属于U,v 属于V-U,则必定存在一颗包含边(u,v)的最小生成树),下面就介绍两种使用MST 性质生成最小生成树的算法:普里姆算法和克鲁斯卡尔算法。

Java Geek Tech wechat
欢迎订阅 Java 技术指北,这里分享关于 Java 的一切。