算法导论 java版_《算法导论》第二章demo代码实现(Java版)

《算法导论》第二章demo代码实现(Java版)

前言

表示晚上心里有些不宁静,所以就写一篇博客,来缓缓。囧

拜读《算法导论》这样的神作,当然要做一些练习啦。除了练习题与思考题那样的理论思考,也离不开编码的实践。

所以,后面每个章节,我都会尽力整理出章节中涉及的算法的Java代码实现。

二分查找

算法实现

package tech.jarry.learning.test.algorithms.binarysearch;

public class BinarySearch {

public static int binarySearch(int[] array, int target) {

return binarySearch(array, target, 0, array.length - 1);

}

// 二分查找,要求输入的线性表必须是顺序的

public static int binarySearch(int[] array, int target, int startIndex, int endIndex) {

if (endIndex > startIndex) {

int middleIndex = (startIndex + endIndex) / 2;

if (target < array[middleIndex]){

return binarySearch(array, target, startIndex, middleIndex);

} else if (target > array[middleIndex]) {

return binarySearch(array, target, middleIndex + 1, endIndex);

} else if (target == array[middleIndex]) {

return middleIndex;

}

}

return -1;

}

}

算法测试

package tech.jarry.learning.test.algorithms.binarysearch;

public class BinarySearchTest {

public static void main(String[] args) {

int[] testArray = new int[] {3, 5, 7, 2, 4, 1, 5, 0, 9, 18 ,12};

System.out.println(BinarySearch.binarySearch(testArray, 100));

}

}

结果输出

-1

这表示没有找到目标数据,可以将测试类中的target修改为其他数字。

冒泡排序

算法实现

package tech.jarry.learning.test.algorithms.bubblesort;

public class BubbleSort {

public static int[] bubbleSort(int[] array) {

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

for (int j = i; j < array.length; j++) {

if (array[j] < array[i]) {

int temp = array[i];

array[i] = array[j];

array[j] = temp;

}

}

}

return array;

}

}

算法测试

package tech.jarry.learning.test.algorithms.bubblesort;

import java.util.Arrays;

public class BubbleSortTest {

public static void main(String[] args) {

int[] testArray = new int[] {3, 5, 7, 2, 4, 1, 5, 0, 9, 18 ,12};

System.out.println(Arrays.toString(BubbleSort.bubbleSort(testArray)));

}

}

结果输出

[0, 1, 2, 3, 4, 5, 5, 7, 9, 12, 18]

插入排序

算法实现

package tech.jarry.learning.test.algorithms.insertsort;

import java.util.Arrays;

public class InsertionSort {

public static int[]insertSort(int[] originArray) {

// 从数组的第二个元素开始进行比较(总不能第一个元素和第一个元素自己比较大小吧)

for (int j = 1; j < originArray.length; j++) {

// 获取当前元素的值

int key = originArray[j];

// 获取前一个元素的下标

int i = j - 1;

// 将key值前移(即将遇到的每个大于key的元素后移)

while (i >= 0 && originArray[i] > key) {

originArray[i + 1] = originArray[i];

i = i - 1;

}

// 直到遇到originArray[i] <= key,才对i+1进行赋值(而i+1元素之前已经后移复制了,即i+2位置保存了i+1位置的值)

originArray[i + 1] =key;

}

return originArray;

}

public static int[] insertSortProWithBinarySearch(int[] array) {

// 如果数据组织形式是数组,那么即使采用二分查找优化,底层的数组元素移动,依旧会导致最终的时间复杂度变为n^2,而不是期待的n*lgn

return null;

}

}

算法测试

package tech.jarry.learning.test.algorithms.insertsort;

import java.util.Arrays;

public class InsertionSortTest {

// test

public static void main(String[] args) {

int[] originArray = new int[]{9, 6, 4, 5, 8};

int[] resultArray = InsertionSort.insertSort(originArray);

System.out.println(Arrays.toString(resultArray));

}

}

结果输出

[4, 5, 6, 8, 9]

归并排序

算法实现

这个代码的实现,可能内容比较多。一方面是由于方法提取(提取哨兵创建的操作),另一方面是由于增加了练习题中提到的无哨兵归并排序的实现(在mergeSort方法中,可以自由选择是否使用哨兵)。

package tech.jarry.learning.test.algorithms.mergesort;

import java.util.Arrays;

/**

* 归并排序

*/

public class MergeSort {

public static int[] mergeSort(int[] array) {

return mergeSort(array, 0, array.length - 1);

}

public static int[] mergeSort(int[] array, int startIndex, int endIndex) {

if (startIndex < endIndex) {

int middleIndex = startIndex + (endIndex - startIndex) / 2;

array = mergeSort(array, startIndex, middleIndex);

array = mergeSort(array, middleIndex + 1, endIndex);

// 使用哨兵,进行合并

// return merge(array, startIndex, middleIndex, endIndex);

// 不适用哨兵,进行合并

return noSentinelMerge(array, startIndex, middleIndex, endIndex);

}

// 如果startIndex = endIndex,表示array只有一个元素

return array;

}

private static int[] merge(int[] array, int startIndex, int middleIndex, int endIndex) {

int[] sentinelLeftArray = createSentinelArray(array, startIndex, middleIndex);

int[] sentinelRightArray = createSentinelArray(array, middleIndex + 1, endIndex);

for (int i = 0, m = 0, n = 0; i < endIndex - startIndex + 1; i++) {

if (sentinelLeftArray[m] < sentinelRightArray[n]) {

// 这里千万别忘了startIndex,因为不同分支的起点不同

array[startIndex + i] = sentinelLeftArray[m++];

} else {

array[startIndex + i] = sentinelRightArray[n++];

}

// 不用考虑两个Integer.MAX_VALUE,因为最后两个数组分别剩下的元素必然是这两个哨兵元素

}

return array;

}

private static int[] createSentinelArray(int[] array, int startIndex, int endIndex) {

int length = endIndex - startIndex + 1;

int[] sentinelArray = new int[length + 1];

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

sentinelArray[i] = array[startIndex + i];

}

sentinelArray[endIndex - startIndex + 1] = Integer.MAX_VALUE;

return sentinelArray;

}

// p.22_practise2.3-2 在不使用哨兵的前提下,进行归并排序的合并操作

private static int[] noSentinelMerge(int[] array, int startIndex, int middleIndex, int endIndex) {

int[] leftArray = createNonSentinelBranchArray(array, startIndex, middleIndex);

int[] rightArray = createNonSentinelBranchArray(array, middleIndex + 1, endIndex);

for (int i = 0, m = 0, n = 0; i < endIndex - startIndex + 1; i++) {

if (leftArray[m] < rightArray[n]) {

array[startIndex + i] = leftArray[m++];

if (m == leftArray.length) {

// 将rightArray剩下的元素全部复制到array对应位置中

array = branchArray2Array(array, startIndex + i + 1, rightArray, n);

break;

}

} else {

array[startIndex + i] = rightArray[n++];

if (n == rightArray.length) {

// 将leftArray剩下的元素全部复制到array对应位置中

array = branchArray2Array(array, startIndex + i + 1, leftArray, m);

break;

}

}

// 不用考虑两个Integer.MAX_VALUE,因为最后两个数组分别剩下的元素必然是这两个哨兵元素

}

return array;

}

private static int[] createNonSentinelBranchArray(int[] array, int startIndex, int endIndex) {

int length = endIndex - startIndex + 1;

int[] branchArray = new int[length];

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

branchArray[i] = array[startIndex + i];

}

return branchArray;

}

private static int[] branchArray2Array(int[] array, int targetIndex, int[] branchArray, int startIndex) {

while (startIndex < branchArray.length) {

array[targetIndex++] = branchArray[startIndex++];

}

return array;

}

// 由于一些情况(如内存空间不足),数据可以直接保存到硬盘中。而不是保存在内存的数组中

private static void merge2Disk(int[] array, int startIndex, int middleIndex, int endIndex){

int[] sentinelLeftArray = createSentinelArray(array, startIndex, middleIndex);

int[] sentinelRightArray = createSentinelArray(array, middleIndex + 1, endIndex);

for (int i = 0, m = 0, n = 0; i < endIndex - startIndex + 1; i++) {

if (sentinelLeftArray[m] < sentinelRightArray[n]) {

Disk.store(sentinelLeftArray[m++]);

} else {

Disk.store(sentinelRightArray[n++]);

}

// 不用考虑两个Integer.MAX_VALUE,因为最后两个数组分别剩下的元素必然是这两个哨兵元素

}

}

// test_creatreSentinelArray

public static void main(String[] args) {

int[] testArray = new int[] {3, 5, 7, 2, 4, 1, 5, 0};

System.out.println(Arrays.toString(createSentinelArray(testArray, 0 , 2)));

}

}

补充:上述代码涉及的Disk类

之所以在归并排序中增加这个硬盘操作,是因为我做这道题想起来很久之前遇到的一道面试题。就是问如何用1G的空间,去排序8G的数据。答案就是采用归并排序(当时原理说出来了,但是白板没写好)。

package tech.jarry.learning.test.algorithms.mergesort;

import java.util.ArrayList;

import java.util.Iterator;

import java.util.List;

/**

* 模仿真实硬盘,进行数据的存储与打印数据

*/

public class Disk {

private static List diskIntegerInstance = new ArrayList<>();

public static void store(int element) {

diskIntegerInstance.add(element);

}

public static void printAll() {

Iterator integerIterator = diskIntegerInstance.iterator();

while (integerIterator.hasNext()) {

System.out.println(integerIterator.next());

}

}

}

算法测试

package tech.jarry.learning.test.algorithms.mergesort;

import java.util.Arrays;

public class MergeSortTest {

public static void main(String[] args) {

int[] testArray = new int[] {3, 5, 7, 2, 4, 1, 5, 0, 9, 18 ,12};

// 3, 5, 7, 2, 4, 1, 5, 0, 9, 18 ,12

// correct result: [0, 1, 2, 3, 4, 5, 5, 7, 9, 12, 18]

int[] resultArray = MergeSort.mergeSort(testArray);

System.out.println(Arrays.toString(resultArray));

}

}

结果输出

[0, 1, 2, 3, 4, 5, 5, 7, 9, 12, 18]

确定两数之和为固定值

这道题在leetcode中是存在的,之前的博客也有对应的解析。甚至leetcode还有求三数之和为确定值的题目。

算法实现

package tech.jarry.learning.ch2.algorithms.twosum;

import tech.jarry.learning.ch2.algorithms.mergesort.MergeSort;

public class TwoSum {

// 题目中只要求实现确定是否存在,而无需返回对应index。否则,需要注意剔除相同index的问题,并修改binarySearch的返回值

public static boolean twoSum(int[] array, int target) {

array = MergeSort.mergeSort(array);

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

int branchTarget = target - array[i];

// 二分查找的时间复杂度为lgn

if (binarySearch(array, branchTarget)) {

return true;

}

}

return false;

}

private static boolean binarySearch(int[] array, int target) {

return binarySearch(array, target, 0, array.length - 1);

}

private static boolean binarySearch(int[] array, int target, int startIndex, int endIndex) {

if (endIndex > startIndex) {

int middleIndex = (endIndex + startIndex) / 2;

if (target < array[middleIndex]) {

return binarySearch(array, target, startIndex, middleIndex);

} else if (target > array[middleIndex]) {

return binarySearch(array, target, middleIndex + 1, endIndex);

} else if (target == array[middleIndex]) {

return true;

}

}

return false;

}

}

算法测试

package tech.jarry.learning.test.algorithms.twosum;

public class TwoSumTest {

public static void main(String[] args) {

int[] testArray = new int[] {3, 5, 7, 2, 4, 1, 5, 0, 9, 18 ,12};

int target = 80;

System.out.println(TwoSum.twoSum(testArray, target));

}

}

结果输出

false

由于书中的demo只要求输出存在与否,而leetcode类似的题目,则要求返回两个元素的index。感兴趣的朋友,可以看我之前写的有关leetcode求两数之和解法的博客。

总结

其实,这章算法的demo还是比较容易实现的。更多的是找找实现算法的感觉吧。

如果代码存在什么问题,或者你们存在什么疑惑,可以私信或@我。

愿与诸君共进步。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/550616.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

java js websocket_js+java websocket记录

首先websocket后台需要在web服务器运行&#xff0c;所以先搭建一个web项目。websocket需要依赖其他的jar包&#xff0c;引入依赖javaxjavaee-api7.0provided新建一个类&#xff0c;该类建立一个服务端点 ServerEndpointimportjavax.websocket.*;importjavax.websocket.server.P…

java 写一个计算器_java编写一个计算器类

展开全部import java.awt.BorderLayout;import java.awt.Color;import java.awt.GridLayout;import java.awt.event.ActionEvent;import java.awt.event.ActionListener;import javax.swing.ImageIcon;import javax.swing.JButton;import javax.swing.JFrame;import javax.swin…

java 移位运算效率_java 简单认识移位运算符和位运算符

移位运算符和位运算符本质上都是操作二进制位&#xff0c;因为计算机存储的是二进制数据&#xff0c;运算效率相对较高。移位运算符:把整数的二进制位进行左移或右移 .左移一位,相当于这个数乘以2, 右移一位,相当于这个数除以2/*移位运算符把整数的二进制位进行左移或右移按位左…

java文本框背景_用Java编写小程序(包含组合框下拉和文本框)变换背景颜色

展开全部代码如下&#xff1a;import java.awt.Color;import java.awt.Dimension;import java.awt.FlowLayout;import java.util.HashMap;import java.util.Map;import javax.swing.JComboBox;import javax.swing.JFrame;import javax.swing.JTextField;public class App exten…

mysql call procedure into_mysql存储过程之创建(CREATE PROCEDURE)和调用(CALL)及变量...

本文实例讲述了mysql存储过程之创建(CREATE PROCEDURE)和调用(CALL)及变量创建(DECLARE)和赋值(SET)操作方法。分享给大家供大家参考&#xff0c;具体如下&#xff1a;存储过程创建(CREATE PROCEDURE)和调用(CALL)这个咱就不说什么定义啦术语啦啥的&#xff0c;直接看着实例来慢…

关于java_关于Java基础

一、方法的重载与重写方法重载(overload)是让类以统一的方式处理不同类型数据的一种手段。调用方法时通过传递给它们的不同个数和类型的参数来决定具体使用哪个方法&#xff0c;这就是多态性。所谓方法重载是指在一个类中&#xff0c;多个方法的方法名相同&#xff0c;但是参数…

java map 转string_java-将Map String,Object转换为Map String,String

java-将Map 转换为Map 如何将Map转换为Map&#xff1f;这不起作用&#xff1a;Map map new HashMap(); //Object is containing StringMap newMap new HashMap(map);11个解决方案37 votes现在我们有了Java 8 / streams&#xff0c;我们可以在列表中添加一个可能的答案&#xf…

java gui 线程_如何在java中将消息从工作线程传递给GUI

您必须使用SwingUtilities.invokeLater&#xff0c;因为只能从事件派发线程访问Swing组件。这个方法的javadoc有一个关于线程的Swing教程的链接。点击此链接。这是一个例子&#xff1a;public class SwingWithThread {private JLabel label;// ...public void startBackgroundT…

java for each 原理_Java for each实现机制代码原理解析

源测试代码如下public class ForEachTest {public void test4Iterate(Iterable strings) {for (String str : strings) {System.out.println(str);}}public void test4Array(String[] strings) {for (String str : strings) {System.out.println(str);}}}执行编译命令javac For…

java输出九九乘法口诀表_java编程打印“九九乘法口诀表”的程序

其实就这几行代码&#xff0c;我大体意思已经写明白了&#xff0c;你还想知道什么&#xff0c;还是你没看懂这几行代码&#xff1f;/** main方法&#xff0c;一个java程序要运行&#xff0c;必须有这个方法&#xff0c;写法是固定的&#xff0c;记住就行了 */public static voi…

学java交学费包分配_java 学生缴学费案例:

/**学生缴学费案例: 学生类(Student)有两个字段: name(名字)和isFee(是否交学费的状态),有一个方法:交学费(fees)。每 一个学生都是通过student类new出来的一个对象,现在创建一个数组存放多个学生对象,再判断数组 中的学生是否已经缴费,如果没有,则学生的交学费方法。 */ //描述…

java web删除文件_Web开发为了java端程序能删除文件的解决办法

Web开发为了java端程序能删除文件&#xff0c;必须在.../jre/lib/security/java.policy文件中添加以下两个中的任何一段代码进去。注&#xff1a;必须是服务器的本地文件&#xff0c;局域网内的其他机器文件根本不能访问。一&#xff0c;grant codeBase "file:/d:/-"…

如何用java对接口发送请求_7. 用Java做接口测试1-发送HTTP请求和接收HTTP响应

前面的文章介绍了使用Postman、Fiddler做接口测试&#xff0c;现在咱们进入代码阶段。我会使用Java做示例&#xff0c;讲解如何使用Java做接口自动化测试。所以&#xff0c;这一部分需要一些Java代码基础&#xff0c;需要掌握java的代码结构、类、对象、接口、List、Map、多态等…

solr java浏览器访问_solr解决访问安全

Tomcat7&#xff0c;solr3.6&#xff0c;mmseg1.81&#xff1a;环境的搭建1&#xff1a;解压tomcat&#xff0c;solr&#xff0c;mmseg4j2&#xff1a;复制dist文件夹下apache-solr.war到tomcat的webapp文件夹下&#xff0c;修改文件名为solr(不该也行&#xff0c;这里修改文件…

java heap buffer direct buffer_java NIO - DirectBuffer 和 HeapBuffer

问题 &#xff1a;DirectBuffer 属于堆外存&#xff0c;那应该还是属于用户内存&#xff0c;而不是内核内存&#xff1f;FileChannel 的read(ByteBuffer dst)函数,write(ByteBuffer src)函数中&#xff0c;如果传入的参数是HeapBuffer类型,则会临时申请一块DirectBuffer,进行数…

yii2 提供接口给java_yii2写api接口步骤

yii2写api接口步骤Yii2如何实现RESTful风格的API(推荐&#xff1a;《YII教程》 )1、建立单独的应用程序为了增加程序的可维护性&#xff0c;易操作性&#xff0c;我们选择新建一套应用程序&#xff0c;这也是为了和前台应用、后台应用区分开操作。在WEB前端(frontend)和后端(ba…

java caller_js中的caller和callee属性

caller返回一个对函数的引用&#xff0c;该函数调用了当前函数。functionName.callerfunctionName 对象是所执行函数的名称。如果一个函数f是在全局作用域内被调用的,则f.caller为null,相反,如果一个函数是在另外一个函数作用域内被调用的,则f.caller指向调用它的那个函数.该属…

mysql 备份 php_PHP备份/还原MySQL数据库的代码

一、备份数据库并下载到本地// 设置SQL文件保存文件名$filenamedate("Y-m-d_H-i-s")."-".$cfg_dbname.".sql";// 所保存的文件名header("Content-disposition:filename".$filename);header("Content-type:application/octetstrea…

java代码修改触发编译_gcc -O0仍然优化了“未使用”的代码 . 是否有一个编译标志来改变它?...

当我在this question中提出时&#xff0c;gcc正在移除(是的&#xff0c;使用 -O0 )一行代码 _mm_div_ss(s1, s2); &#xff0c;大概是因为结果未保存 . 但是&#xff0c;这应该触发浮点异常并引发SIGFPE&#xff0c;如果删除调用则不会发生这种情况 .Question &#xff1a;是否…

java第二部分项目_Java_第二次作业:项目构思与实现

写在最前&#xff1a;我我我我我我靠&#xff0c;以后再也不再ddl截止前1小时调试程序了&#xff01;之前在DDL前1小时修改程序&#xff0c;当我改完后&#xff0c;我想着&#xff0c;再把之前的测试样例跑一遍&#xff0c;如果都对就OK了。就在这时&#xff0c;问题出现了&…