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

news2025/1/24 15:17:59

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

  • 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 则无效 返回 0
            return 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.coloradmin.cn/o/1094387.html

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

相关文章

yolov5+车辆重识别【附代码】

本篇文章主要是实现的yolov5和reid结合的车辆重识别项目。是在我之前实现的yolov5_reid行人重识别的代码上修改实现的baseline模型。 目录 相关参考资料 数据集说明 环境说明 项目使用说明 vehicle reid训练 yolov5车辆重识别 从视频中获取想要检测的车(待检测车辆) 车…

Attention Is All You Need原理与代码详细解读

文章目录 前言一、Transformer结构的原理1、Transform结构2、位置编码公式3、transformer公式4、FFN结构 二、Encode模块代码解读1、编码数据2、文本Embedding编码3、位置position编码4、Attention编码5、FFN编码 三、Decode模块代码解读1、编码数据2、文本Embedding与位置编码…

centos6/7 SOCKS5 堆溢出漏洞修复(RPM方式)curl 8.4 CVE-2023-38545 CVE-2023-38546

引用 https://darkdark.top/update-curl.html centos6 rpm 升级包下载&#xff1a;https://download.csdn.net/download/sinat_24092079/88425840 yum update libcurl-8.4.0-1.el6.1.x86_64.rpm curl-8.4.0-1.el6.1.x86_64.rpmcentos7 rpm 升级包下载&#xff1a;https://down…

ELF和静态链接:为什么程序无法同时在Linux和Windows下运行?

目录 疑问 编译、链接和装载&#xff1a;拆解程序执行 ELF 格式和链接&#xff1a;理解链接过程 小结 疑问 既然我们的程序最终都被变成了一条条机器码去执行&#xff0c;那为什么同一个程序&#xff0c;在同一台计算机上&#xff0c;在 Linux 下可以运行&#xff0c;而在…

Linux | 关于入门Linux你有必要了解的指令

目录 前言 1、ls指令 2、pwd指令 3、cd指令 4、touch指令 5、stat指令 6、mkdir指令 7、rmdir 与 rm指令 8、man指令 9、cp指令 10、mv指令 11、cat指令 &#xff08;1&#xff09;输入重定向 &#xff08;2&#xff09;输出重定向与追加重定向 12、less指令 1…

多模态模型文本预处理方式

句子级别 句子级别的表征编码一整个句子到一个特征中。如果一个句子有多个短语&#xff0c;提取这些短语丢弃其他的单词。 缺点&#xff1a;这种方式会丢失句子中细粒度的信息。 单词级别 将句子中的类别提取出来&#xff0c;结合成一个句子。 缺点&#xff1a;会在类别之…

【数据结构】线性表的抽象数据类型

&#x1f984;个人主页:修修修也 &#x1f38f;所属专栏:数据结构 ⚙️操作环境:Visual Studio 2022 线性表抽象数据类型(LinearListAbstractDataType&#xff0c;简称 ADT)是一种非常重要的抽象数据类型&#xff0c;它是一种使用抽象的方式表示和实现一组数据元素的集合以及与…

宝塔面板服务器内存使用率高的三招解决方法

卸载多余PHP版本。假若安装了多个PHP版本&#xff0c;甚至把 php 5.3、5.4、7.0、7.3 全都安装上了&#xff0c;就会严重增加系统负载和内存使用率。 安装memcached 缓存组件&#xff0c;建议在宝塔面板后台直接安装。 卸载不常用软件。如&#xff1a;宝塔运维、宝塔一键安装…

php如何查找地图距离

要在PHP中使用高德地图、百度地图或腾讯地图获取位置信息&#xff0c;您可以使用它们的相应API服务。以下是获取位置信息的一般步骤&#xff1a; 思路: 获取API密钥&#xff1a;首先&#xff0c;您需要注册并获取相应地图服务提供商的API密钥。这将允许您访问他们的API以获取位…

CSS的美化(文字、背景) Day02

一、文字控制属性 分为&#xff1a;字体样式属性 、文本样式属性 1.1 CSS字体样式属性 1.color定义元素内文字颜色2.font-size 字号大小3 font-family 字体4 font-weight 字体粗细5.font-style 字体风格6.font 字体综合属性 1.1.1 > 文字颜色 color 属性名: color color …

Yakit工具篇:简介和安装使用

简介(来自官方文档) 基于安全融合的理念&#xff0c;Yaklang.io 团队研发出了安全领域垂直语言Yaklang&#xff0c;对于一些无法原生集成在Yak平台中的产品/工具&#xff0c;利用Yaklang可以重新编写 他们的“高质量替代”。对于一些生态完整且认可度较高的产品&#xff0c;Y…

C# CodeFormer 图像修复

效果 项目 代码 using Microsoft.ML.OnnxRuntime; using Microsoft.ML.OnnxRuntime.Tensors; using OpenCvSharp; using System; using System.Collections.Generic; using System.Drawing; using System.Drawing.Imaging; using System.Windows.Forms;namespace 图像修复 {p…

高校教务系统登录页面JS分析——南京邮电大学

高校教务系统密码加密逻辑及JS逆向 本文将介绍南京邮电大学教务系统的密码加密逻辑以及使用JavaScript进行逆向分析的过程。通过本文&#xff0c;你将了解到密码加密的基本概念、常用加密算法以及如何通过逆向分析来破解密码。 本文仅供交流学习&#xff0c;勿用于非法用途。 一…

C++标准模板(STL)- 类型支持 (数值极限,min_exponent10,max_exponent,max_exponent10)

数值极限 std::numeric_limits 定义于头文件 <limits> 定义于头文件 <limits> template< class T > class numeric_limits; numeric_limits 类模板提供查询各种算术类型属性的标准化方式&#xff08;例如 int 类型的最大可能值是 std::numeric_limits&l…

多个Python包懒得import,那就一包搞定!

使用Python时&#xff0c;有的代码需要依赖多个框架或库者来完成&#xff0c;代码开头需要import多次&#xff0c;比如&#xff0c; import pandas as pd from pyspark import SparkContext from openpyxl import load_workbook import matplotlib.pyplot as plt import seabo…

Java Day2(Java基础语法)

Java基础 Java基础语法1. 注释、关键字、标识符1.1 Java中的注释1.2 关键字1.3 标识符 2. 数据类型&#xff08;1&#xff09;基本类型&#xff08;primitive type&#xff09;a.字节b.进制c. 浮点数拓展d. 字符拓展 &#xff08;2&#xff09; 引用类型(Reference type ) 3. 类…

【软件测试】总结

文章目录 一. 测试用例1. 常见设计测试用例(1)非软件题型(2)软件题型(3)代码型题(4)关于个人项目设计测试用例 2. 万能公式和具体的方法如何理解(1)万能公式(2)Fiddler实现弱网模式(3)针对公交卡设计测试用例 3. 进阶设计测试用例 二. 自动化1. 什么是自动化以及为什么要做自动…

杀死僵尸进程ZooKeeperMain

关闭Hadoop后jps发现还有个进程ZooKeeperMain没有关闭&#xff0c;使用kill -9 <>也没有用&#xff0c;这种就是僵尸进程&#xff0c;需要用父进程ID来杀死 解决方法 话不多说&#xff0c;直接上解决方案&#xff0c; 1. 第一步 清楚需要关闭的进程ID&#xff0c;我…

CentOS-7下安装及配置vsftpd详细步骤(可匿名访问)

第一步安装vsftpd&#xff1a; yum -y install vsftpd 第二步修改ftp主目录所属用户为用户ftp&#xff1a; chown ftp /var/ftp/pub 第三步备份及配置ftp&#xff1a; cp /etc/vsftpd/vsftpd.conf ~/vsftpd.conf.bakvim /etc/vsftpd/vsftpd.conf 配置如下图&#xff1a;…

《AWD特训营》CTF/AWD竞赛的速胜指南!全面提升安全技术

&#x1f3ac; 鸽芷咕&#xff1a;个人主页 &#x1f525; 个人专栏:《粉丝福利》 《C语言进阶篇》 ⛺️生活的理想&#xff0c;就是为了理想的生活! 文章目录 前言一、AWD竞赛的由来《AWD特训营&#xff1a;技术解析、赛题实战与竞赛技巧》1.1介绍&#xff1a; 《AWD特训营》…