数据结构与算法--其他算法

数据结构与算法--其他算法

  • 1 汉诺塔问题
  • 2 字符串的全部子序列
  • 3 字符串的全排列
  • 4 纸牌问题
  • 5 逆序栈问题
  • 6 数字和字符串转换问题
  • 7 背包问题
  • 8 N皇后问题

暴力递归就是尝试
1,把问题转化为规模缩小了的同类问题的子问题
2,有明确的不需要继续进行递归的条件(base case)
3,有当得到了子问题的结果之后的决策过程
4,不记录每一个子问题的解

 


 

1 汉诺塔问题

 

打印n层汉诺塔从最左边移动到最右边的全部过程

如下图所示,把 A 上的方块从移动到 C 上,要求 在移动的过程中 ,小的块在大的块上面

假设有三个点 : from to other,有 i 个方块
问题可以转换成 将 1 ~ i个圆盘从from点移动到to 点,过程如下 :

  1. 1 ~ (i - 1)个方块从 from 点移动到 other
  2. 将 第i个方块从 from点移动到 to
  3. i - 1个方块从 other点移动到 to

如下所示

1)
在这里插入图片描述
 

2)
在这里插入图片描述

 

3)
在这里插入图片描述

coding

public class HanoiTest {public static void main(String[] args) {hanoi(20);System.out.println(iCount);}public static int iCount = 0;public static void hanoi(int n) {if (n > 0) {func(n, "A", "B", "C");}}public static void func(int i, String start, String end, String other) {if (i == 1) {System.out.println("move 1 from " + start + " to " + end);iCount ++;} else {func(i - 1, start, other, end);iCount ++;System.out.println("move " + i + " from " + start + " to " + end);func(i - 1, other, end, start);}}
}

 


 

2 字符串的全部子序列

 

假设有字符串 abc ,要求打印出 abc的全部子序列

可根据包不包含每个字符 构建如图所示的二叉树
在这里插入图片描述

coding

public class PrintAllSubSequenceTest {public static void main(String[] args) {List<String> list = processIteration("abc".toCharArray());for (String s : list) {System.out.println(s);}}public static List<String> characterList = new ArrayList<>();public static List<String> getAllSubSequence(String parentStr) {characterList.clear();process(parentStr.toCharArray(), 0);return characterList;}/*** 当前来到了 i 位置 子序列中要和不要改字符走两条路** @param chars* @param i*/public static void process(char[] chars, int i) {// 到了最后一个位置if (i == chars.length) {characterList.add(String.valueOf(chars));return;}process(chars, i + 1);// 要 i 位置字符的路char temp = chars[i];chars[i] = 0;process(chars, i + 1);// 要 i 位置字符的路chars[i] = temp; //把字符串还原}/*** 使用迭代的方法** @param chars*/public static List<String>  processIteration(char[] chars) {List<String> retList = new ArrayList<>();for (int i = 0; i < chars.length;i++) {String str = "";for (int j = i ; j < chars.length;j++){str += chars[j];retList.add(str);}}return retList;}
}

 


 

3 字符串的全排列

 

打印一个字符串的全部排列

public class StringPermutationTest {public static void main(String[] args) {List<String> strings = getAllSubPermutation("abc");for (String s : strings) {System.out.println(s);}}public static List<String> getAllSubPermutation(String str){List<String> subPermutationList = new ArrayList<>();if (str == null || str.length() == 0){return subPermutationList;}process(0,str.toCharArray(),subPermutationList);return subPermutationList;}/*** 当前来到的是 i 位置* str[0..i-1]是之前做的选择* @param i* @param str* @param retList*/public static void process(int i,char[] str,List<String> retList){if (i == str.length){retList.add(String.valueOf(str));}for (int j = i; j < str.length;j++){swap(str,i,j);process(i + 1,str,retList);// 交换回来swap(str,i,j);}}public static void swap(char[] chars,int i,int j){char temp = chars[i];chars[i] = chars[j];chars[j] = temp;}
}

打印一个字符串的全部排列,要求不能出现重复的排列

public class StringPermutationTest {public static void main(String[] args) {List<String> strings = getAllSubPermutation("abc");for (String s : strings) {System.out.println(s);}}public static List<String> getAllSubPermutation(String str){List<String> subPermutationList = new ArrayList<>();if (str == null || str.length() == 0){return subPermutationList;}process(0,str.toCharArray(),subPermutationList);return subPermutationList;}/*** 当前来到的是 i 位置* str[0..i-1]是之前做的选择* @param i* @param str* @param retList*/public static void process(int i,char[] str,List<String> retList){if (i == str.length){retList.add(String.valueOf(str));}boolean[] bVisited = new boolean[26];for (int j = i; j < str.length;j++){//字符串没有被试过 才进行处理 if (!bVisited[str[j] - 'a']) {bVisited[str[j] - 'a'] = true;swap(str,i,j);process(i + 1,str,retList);// 交换回来swap(str,i,j);}}}public static void swap(char[] chars,int i,int j){char temp = chars[i];chars[i] = chars[j];chars[j] = temp;}
}

 


 

4 纸牌问题

 

给定一个整型数组arr,代表数值不同的纸牌排成一条线。玩家A和玩家B依次拿走每张纸牌,规定玩家A先拿,玩家B后拿,但是每个玩家每次只能拿走最左或最右的纸牌,玩家A和玩家B都绝顶聪明。请返回最后获胜者的分数

例如 :
arr=[1,2,100,4]。
开始时,玩家A只能拿走1或4。如果开始时玩家A拿走1,则排列变为[2,100,4],接下来玩家 B可以拿走2或4,然后继续轮到玩家A… 如果开始时玩家A拿走4,则排列变为[1,2,100],接下来玩家B可以拿走1或100,然后继续轮到玩家A… 玩家A作为绝顶聪明的人不会先拿4,因为拿4之后,玩家B将拿走100。所以玩家A会先拿1,
让排列变为[2,100,4],接下来玩家B不管怎么选,100都会被玩家 A拿走。玩家A会获胜,分数为101。所以返回101。

arr=[1,100,2]。
开始时,玩家A不管拿1还是2,玩家B作为绝顶聪明的人,都会把100拿走。玩家B会获胜,分数为100。所以返回100。

ooding

public class CardInLineTest {public static void main(String[] args) {int[] arr = {1,2,100,4};int iWinScore = win(arr);System.out.println(iWinScore);}/*** 先手函数* 在 L..R范围上先手拿牌 返回最大值* @param arr* @param L* @param R* @return*/public static int first(int[] arr, int L, int R) {// 如果只有一个数 直接拿走if (L == R) {return arr[L];}// 返回一个最大值return Math.max(arr[L] + second(arr, L + 1, R), // 先手拿走最左侧的数 后手在 (L + 1)..R范围上arr[R] + second(arr, L, R - 1)// 先手拿走最右侧的数 后手在 L..(R - 1)范围上);}/*** 后手函数** @param arr* @param L* @param R* @return*/public static int second(int[] arr, int L, int R) {// 因为是后手 L == R时 被别人拿走 因此直接返回if (L == R){return 0;}// 因为是别人决定的  因此会别人会把最小的留给自己return Math.min(first(arr,L + 1,R),// 别人拿走了L上 先手在 (L + 1)..R范围上first(arr,L,R -1));}public static int win(int[] arr){if (arr == null || arr.length == 0){return 0;}return Math.max(first(arr,0,arr.length - 1),second(arr,0,arr.length -1));}
}

 


 

5 逆序栈问题

 

给你一个栈,请你逆序这个栈,不能申请额外的数据结构,只能使用递归函数。
如何实现?

coding

public class ReverseStackNoRecurTest {public static void main(String[] args) {Stack<Integer> stack = new Stack<>();stack.push(3);stack.push(2);stack.push(1);reverseStack(stack);while (!stack.isEmpty()){System.out.println(stack.pop());}}/*** 反转栈* @param stack*/public static void reverseStack(Stack<Integer> stack){if (stack.isEmpty()){return;}// 每次调用都从栈中移除栈底元素int bottomEle = getBottomEle(stack);// 继续反转栈reverseStack(stack);stack.push(bottomEle);}/*** 从栈中移除栈底的元素* @param stack* @return*/public static int getBottomEle(Stack<Integer> stack){int ret = stack.pop();if (stack.isEmpty()){return ret;} else {int last = getBottomEle(stack);stack.push(ret);return last;}}
}

 


 

6 数字和字符串转换问题

 
规定1和A对应、2和B对应、3和C对应… 那么一个数字字符串比如"111",就可以转化为"AAA"、“KA"和"AK”。
给定一个只有数字字符组成的字符串str,返回有多少种转化结果

public class ConvertToLetterStringTest {public static void main(String[] args) {String str = "111";int count = convertToLetterString(str);System.out.println(count);}public static int convertToLetterString(String str){if (str == null || str.length() == 0){return 0;}return process(str.toCharArray(),0);}/*** [0..index-1]位置的字符已经做过决定* 当前来到 index位置* @param chr* @param index* @return*/public static int process(char[] chr,int index){if (index == chr.length){ //来到字符串的最后位置return 1;}if (chr[index] == '0') { // 出现 0 则无效 返回 0return 0;}if (chr[index] == '1'){int ret = process(chr,index + 1);// index 位置自己作为单独的部分 后续有多少种if (index + 1 < chr.length){ret += process(chr,index + 2);// (index 和 index + 1)作为单独的部分 后续有多少种}return ret;}if (chr[index] == '2'){int ret = process(chr,index + 1); //  index 位置自己作为单独的部分 后续有多少种if (index + 1 < chr.length && (chr[index + 1] <= '6' && chr[index] >= '0')) {ret += process(chr,index + 2);// (index 和 index + 1)作为单独的部分 后续有多少种}return ret;}// index位置的字符为 3..9的情况return process(chr,index + 1);}
}

 


 

7 背包问题

 

给定两个长度都为N的数组weightsvaluesweights[i]values[i]分别代表
i号物品的重量和价值。给定一个正数bag,表示一个载重bag的袋子,你装的物
品不能超过这个重量。返回你能装下最多的价值是多少?

coding

public class KnapsackQuesTest {/*** index... 之后的货物随意选择,形成的最大价值返回* 重量不能超过 bagWeight* @param weights index号货物的价值* @param values index 号货物的重量* @param index* @param alreadyWeight 之前做的决定 所达到的重量* @param bagWeight 背包载重* @return*/public static int process(int[] weights,int[] values,int index,int alreadyWeight,int bagWeight){if (alreadyWeight > bagWeight){return 0;}if (index == weights.length){return 0;}return Math.max(// 不要index位置的货物process(weights, values, index + 1, alreadyWeight, bagWeight),// 要index位置的货物values[index] + process(weights, values, index + 1, weights[index] + alreadyWeight,bagWeight));}
}

 


 

8 N皇后问题

 

N皇后问题是指在N*N的棋盘上要摆N个皇后,要求任何两个皇后不同行、不同列,也不在同一条斜线上。
给定一个整数n,返回n皇后的摆法有多少种。

n=1,返回1

n=232皇后和3皇后问题无论怎么摆都不行,返回0

n=8,返回92。

coding

public class NQueuesQues {public static void main(String[] args) {System.out.println(num(8));}/**** @param n 皇后的个数* @return*/public static int num(int n){if (n < 1){return 0;}int[] rec = new int[n]; // rec[index] index行的皇后放在了第几列return process(0,rec,n);}/**** @param index 当前来到的行* @param rec index 行放在了哪一列* @param n 行数* @return*/public static int process(int index,int[] rec,int n){if (index == n){ // 到最后一行的下一行  则说明之前有一种选择是正确的 找到了一种摆放的方法return 1;}int ret = 0;// 每一列进行尝试for (int col = 0; col < n;col++){if (isValid(rec,index,col)){// 有效 则将 index 行的皇后放在 col列rec[index] = col;//继续处理 下一行ret += process(index + 1,rec,n);}}return ret;}/***  判断 r 行的皇后 放在 c 列 是否有效 只需要判断 rec[0..r-1]即可* @param rec rec[]* @param r* @param c* @return*/public static boolean isValid(int[] rec,int r,int c){for (int k = 0; k < r;k++){// 共列if (rec[k] == c){return false;}// 共斜线// Math.abs(r - k) 竖直方向 r行到 k行的距离// rec[k] - c 水平方向 c行到 rec[k]列的距离if (Math.abs(r - k) == Math.abs(rec[k] - c)) {return false;}}return true;}
}

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

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

相关文章

设计模式之是简单工厂模式

分类 设计模式一般分为三大类&#xff1a;创建型模式、结构型模式、行为型模式。 创建型模式&#xff1a;用于创建对象&#xff0c;共五种&#xff0c;包括单例模式、简单工厂模式、工厂方法模式、抽象工厂模式、建造者模式、原型模式。结构型模式&#xff1a;用于处理类或对…

超火爆的6 个必学持续集成工具,测试人的福音

开发人员喜欢把写的代码当成自己的孩子&#xff0c;他们会被当成艺术品一样呵护。作为家长&#xff0c;总是会认为自己的孩子是最好的&#xff0c;也会尽全力给自己的孩子最好的&#xff0c;就算有时候会超出自己的能力范围。 最终&#xff0c;孩子会走出去&#xff0c;和其他…

android studio检测不到真机

我的情况是&#xff1a; 以前能检测到&#xff0c;有一天我使用无线调试&#xff0c;发现调试有问题&#xff0c;想改为USB调试&#xff0c;但是半天没反应&#xff0c;我就点了手机上的撤销USB调试授权&#xff0c;然后就G了。 解决办法&#xff1a; 我这个情况比较简单&…

Leetcode刷题笔记--Hot61-70

1--课程表&#xff08;207&#xff09; 主要思路&#xff1a; 用 in 记录每一门课程剩余的先修课程个数&#xff0c;当剩余先修课程个数为0时&#xff0c;将该课程加入到队列q中。 每修队列q中的课程&#xff0c;以该课程作为先修课程的所有课程&#xff0c;其剩余先修课程个数…

vue elementui的select组件实现滑到底部分页请求后端接口

vue elementui的select组件实现滑到底部分页请求后端接口 1.实现效果2.实现原理 1.实现效果 老规矩&#xff0c;直接上最后的实现效果 2.实现原理 直接上代码 <el-form-item class"diagmosisItem" label"诊断" v-scroll"handleScroll">…

✔ ★【备战实习(面经+项目+算法)】 10.15学习时间表

✔ ★【备战实习&#xff08;面经项目算法&#xff09;】 坚持完成每天必做如何找到好工作1. 科学的学习方法&#xff08;专注&#xff01;效率&#xff01;记忆&#xff01;心流&#xff01;&#xff09;2. 每天认真完成必做项&#xff0c;踏实学习技术 认真完成每天必做&…

【微信小程序开发】基础语法篇

&#x1f973;&#x1f973;Welcome Huihuis Code World ! !&#x1f973;&#x1f973; 接下来看看由辉辉所写的关于小程序的相关操作吧 目录 &#x1f973;&#x1f973;Welcome Huihuis Code World ! !&#x1f973;&#x1f973; 一.视图层 1.数据绑定 wxml js 2 .列…

Python学习六

前言&#xff1a;相信看到这篇文章的小伙伴都或多或少有一些编程基础&#xff0c;懂得一些linux的基本命令了吧&#xff0c;本篇文章将带领大家服务器如何部署一个使用django框架开发的一个网站进行云服务器端的部署。 文章使用到的的工具 Python&#xff1a;一种编程语言&…

AnolisOS升级SSH,不升级SSL

由于ssh有漏洞需要升级&#xff0c;但是为了最小化升级不影响ssl&#xff0c;因为ssl里面带了加密库&#xff0c;系统中很多核心服务的加密都是用ssl进行加密的&#xff08;像网络服务&#xff0c;系统用户登录等&#xff09;&#xff0c;如果ssl升级出现不兼容&#xff0c;就可…

Java实现B树

1.介绍 B树是一种自平衡的搜索树数据结构&#xff0c;常用于数据库和文件系统中的索引结构。它具有以下好处和功能&#xff1a; 高效的查找操作&#xff1a;B树的特点是每个节点可以存储多个关键字&#xff0c;并且保持有序。通过在节点上进行二分查找&#xff0c;可以快速定位…

260. 只出现一次的数字 III

给你一个整数数组 nums&#xff0c;其中恰好有两个元素只出现一次&#xff0c;其余所有元素均出现两次。 找出只出现一次的那两个元素。你可以按 任意顺序 返回答案。 你必须设计并实现线性时间复杂度的算法且仅使用常量额外空间来解决此问题。 示例 1&#xff1a; 输入&…

计算机基础知识32

Socket抽象层(socket编程) # Socket是在应用层和传输层之间的一个抽象层&#xff0c;它把TCP/IP层复杂的操作抽象为几个简单 的接口供应用层调用已实现进程在网络中通信 socket () 对象 bind () 函数来绑定 listen () 监听&#xff0c;等别人电话 accept&#xff08;&#…

AI电销机器人好不好用关键是什么?

影响AI电销机器人是否好用的两个因素分别是&#xff0c;识别系统以及线路。 有很多电销企业都想找一个好用的AI电销机器人&#xff0c;可是什么样的机器人才是好用的机器人呢?有哪些因素会影响AI电销机器人好不好用呢? 添加图片注释&#xff0c;不超过 140 字&#xff08;可选…

qt中json类

目录 QJsonValue QJsonObject QJsonArray QJsonDocument 案例&#xff1a; Qt 5.0开始提供了对Json的支持&#xff0c;我们可以直接使用Qt提供的Json类进行数据的组织和解析&#xff0c;下面介绍4个常用的类。 QJsonValue 该类封装了JSON支持的数据类型。 布尔类型&#xf…

【Power BI】Power BI 入门指南:版本、下载和报表创建的步骤

文章目录 一、前言二、了解 Power BI 版本三、下载 Power BI Desktop四、如何开始使用 Power BI Desktop五、在 Power BI Desktop 中创建报表六、文末总结 一、前言 Power BI 是微软于 2013 年推出的产品&#xff0c;为一款商业智能与数据可视化工具。它通过引人注目的视觉效果…

[Linux 基础] Linux编辑器Vim,你值得拥有

文章目录 1、Linux 软件包管理器 yum1.1 什么是软件包1.2 如何安装软件1.3 如何卸载软件 2、vim的使用2.1 vim的安装和配置2.2 vim的基本概念2.3 vim的基本操作 3、vim正常模式命令集4、vim注释与去注释5、Liunx编辑器-gcc/g使用5.1 如何使用gcc编译c程序5.2 gcc的翻译过程5.2.…

Python-pyecharts和pandas库

目录 pyecharts库 pandas库 示例1 示例2 pyecharts库 pyecharts是一个基于Python的交互式数据可视化库&#xff0c;旨在帮助用户轻松地创建各种类型的图表和可视化效果。该库是在Echarts开源项目的基础上开发的&#xff0c;Echarts是一款由百度开发的优秀的数据可视化工具。…

计算机网络基础(三):IPv4编址方式、子网划分、IPv4通信的建立与验证及ICMP协议

**IPv4地址是一个32位长的二进制数。**而这个32位二进制数又通常会表示为4个用点隔开的十进制数。那么&#xff0c;这个32位二进制数要如何通过4个十进制数表示出来呢&#xff1f; 我们在配置IPv4地址时&#xff0c;同时配置的“掩码”又有何用途&#xff1f; 1.IPv4编址方式…

第 367 场 LeetCode 周赛题解

A 找出满足差值条件的下标 I 模拟 class Solution { public:vector<int> findIndices(vector<int> &nums, int indexDifference, int valueDifference) {int n nums.size();for (int i 0; i < n; i)for (int j 0; j < i; j)if (i - j > indexDiffe…

软件测试的调用接口怎么调用,逻辑是什么?

一、什么是接口测试&#xff1f; 接口测试是测试系统组件之间接口的测试。接口主要用于检测外部系统和内部子系统之间的交互点。测试的重点是检查数据交换、传输、控制和管理过程&#xff0c;以及系统之间的相互逻辑依赖。 二、为什么要做接口测试&#xff1f; 在淘宝系统的历…