`
aa8945163
  • 浏览: 279331 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

一道让我知道算法忘记的题

阅读更多
原题如下:用1、2、2、3、4、5这六个数字,用java写一个main函数,打印出所有不同的排列,如:512234、412345等,要求:"4"不能在第三位,"3"与"5"不能相连.
我看了回贴都没有很好解决,主要是没有排除重复。
解决思路:强化题目,用1、2、2、3、4、5这六个数字排列“递增”序列。其他要求不变。
算法思路:显然是递归,初始序列122345,先从末两位(45)变化(45,54),然后末三位(345) ... 直到最后六位.怎样解决重复问题?很简单,由于是递增序列,每生成新序列可与前一生成序列比较,如<放弃当前序列。当然有更好效率,如预先预测。代码如下:
class test
{
// 当前固定部分
private String CurFixPart;
private String PreGenNum;

public static void main(String[] args)
{
test t=new test();
t.GenControll("122345");
}

// 调整字符串s位置pos字符到最前
private String shift(String s, int pos)
{
String newStr;
if (s.length()>pos+1)
newStr=s.substring(pos, pos+1)
+s.substring(0, pos)
+s.substring(pos+1);
else
newStr=s.substring(pos)
+s.substring(0, pos);
return newStr;
}

protected int Validate(String newNum)
{
String newGenNum=CurFixPart+newNum;
if (Integer.valueOf(newGenNum)<=Integer.valueOf(PreGenNum))
return 0;
if (newGenNum.substring(2,3).equals("4") ||
(newGenNum.indexOf("35")!=-1) || (newGenNum.indexOf("53")!=-1))
return 0;

PreGenNum=newGenNum;
System.out.println(newGenNum);
return 0;
}

public void GenControll(String Base)
{
PreGenNum="0";
CurFixPart="";
GenNext(Base, 0);
}

void GenNext(String varPart, int curPos)
{
if (varPart.length()==2)
{
Validate(varPart);
Validate(shift(varPart, 1));
return;
}
// Next Layer
String newGen=shift(varPart, curPos);
String SavedFixPart=CurFixPart;
CurFixPart=CurFixPart+newGen.substring(0,1);
GenNext(newGen.substring(1), 0);
CurFixPart=SavedFixPart;
// 同层递增
if (curPos==varPart.length()-1)
return;
GenNext(varPart, curPos+1);
}
}
序列122345测试通过。
有什么意见请大家多多提点。



1 把问题归结为图结构的遍历问题。实际上6个数字就是六个结点,把六个结点连接成无向连通图,对于每一个结点求这个图形的遍历路径,所有结点的遍历路径就是最后对这6个数字的排列组合结果集。
2 显然这个结果集还未达到题目的要求。从以下几个方面考虑:
1. 3,5不能相连:实际要求这个连通图的结点3,5之间不能连通, 可在构造图结构时就满足改条件,然后再遍历图。
2. 不能有重复: 考虑到有两个2,明显会存在重复结果,可以把结果集放在TreeSet中过滤重复结果
3. 4不能在第三位: 仍旧在结果集中去除满足此条件的结果。

采用二维数组定义图结构,最后的代码是:

import java.util.Iterator;
import java.util.TreeSet;

public class TestQuestion {

private String[] b = new String[]{"1", "2", "2", "3", "4", "5"};
private int n = b.length;
private boolean[] visited = new boolean[n];
private int[][] a = new int[n][n];
private String result = "";
private TreeSet set = new TreeSet();

public static void main(String[] args) {
new TestQuestion().start();
}

private void start() {

// Initial the map a[][]
for (int i = 0; i < n; i++) {
for (int j = 0; j < n; j++) {
if (i == j) {
a[i][j] = 0;
} else {
a[i][j] = 1;
}
}
}

// 3 and 5 can not be the neighbor.
a[3][5] = 0;
a[5][3] = 0;

// Begin to depth search.
for (int i = 0; i < n; i++) {
this.depthFirstSearch(i);
}

// Print result treeset.
Iterator it = set.iterator();
while (it.hasNext()) {
String string = (String) it.next();
// "4" can not be the third position.
if (string.indexOf("4") != 2) {
System.out.println(string);
}
}
}

private void depthFirstSearch(int startIndex) {
visited[startIndex] = true;
result = result + b[startIndex];
if (result.length() == n) {
// Filt the duplicate value.
set.add(result);
}
for(int j = 0; j < n; j++) {
if (a[startIndex][j] == 1 && visited[j] == false) {
depthFirstSearch(j);
} else {
continue;
}
}

// restore the result value and visited value after listing a node.
result = result.substring(0, result.length() -1);
visited[startIndex] = false;
}
}
3,5不能相连:实际要求这个连通图的结点3,5之间不能连通, 可在构造图结构时就满足改条件,然后再遍历图。
代码中请注意这几行:
// 3 and 5 can not be the neighbor.
a[3][5] = 0;
a[5][3] = 0;

只要这样定义图,根本不用在代码中写IF ELSE语句。
实际上基于图的算法好处在于,只要你能定义好满足题目要求的图结构,遍历的结果就是你要的结果,不用任何对遍历结果做任何处理。包括本题中的:4不能在第三位置,3,5不能相连,唯一
性要求,其实都可以在体现在构造的图形结构里,然后直接遍历图取得自己要的结果。而不用再次处理结果集。

只是说这里实际上对其它要求要体现在图结构里有困难(理论上是可以的),但起码3,5不能相接是很好构造的,就是上面的代码段来解释的。

关于图形数据结构建议先看看数据结构的书,主要是将如何利用二维数组描述图结构,再看看图的深度遍历实现原理。最后再应用到这个问题上来,自然就不难明白了。

分享到:
评论

相关推荐

    LeetCode算法题练习完整源代码分享给需要的同学

    但考虑到标题和描述中提到的内容是“LeetCode算法题练习完整源代码分享”,以下是基于这个主题的知识点梳理: LeetCode是一个在全球范围内广受欢迎的在线编程实践平台,主要用于帮助程序员提升算法和数据结构技能,...

    《最优化导论》习题答案

    通过对照"最优化导论 Solutions_Manual答案.pdf.zip",你可以针对每一道习题进行深度学习,不仅检验自己的理解,也能从解题思路中学习到更多。同时,不要忘记结合教材和课堂讲解,全方位提升对最优化理论和方法的...

    华育国际一期经典题

    【华育国际一期经典题】是一份专门针对华育国际第一阶段Java学习的考核资源,旨在帮助学员在考试前进行全面的复习和准备。这个压缩包包含的不仅是笔试题目,还有机试题目,确保覆盖了理论知识和实际操作的双重检验。...

    2024河北省对口高考计算机理论考试试题及答案

    2024年河北省对口高考计算机理论考试试题及答案是中职生迈进高等教育殿堂的一道重要门槛,它不仅需要考生有扎实的基础知识,也需要具备良好的解题能力和实际操作技能。通过对这些内容的系统学习与准备,学生将能够更...

    2021-2022计算机二级等级考试试题及答案No.1907.docx

    这是一道考察基础知识的选择题,旨在测试考生对Visual FoxPro命令的记忆和理解。 - **应用场景**:在实际项目中,打开数据库通常是开发流程的第一步,以便后续进行数据的查询、更新等操作。 #### 排序算法的性能...

    一道优雅面试题分析js中fn()和return fn()的区别

    本篇文章通过一道面试题深入解析这两种调用方式的不同。 首先,我们要明白函数在JavaScript中的默认行为。如果一个函数没有显式地使用`return`语句返回一个值,那么它会隐式地返回`undefined`。这是在控制台中看到`...

    C++-leetcode题解674. Longest Continuous Increasing Subsequence.cpp

    Longest Continuous Increasing Subsequence是一道典型的算法练习题,主要考察编程者对数组遍历和子序列概念的理解及其应用。该题目要求找出一个整型数组中连续递增的最长子序列的长度,其中子序列是原始数组中去掉...

    青岛版五四制五年级上册信息窗1第二课时.pdf

    【青岛版五四制五年级上册信息...总的来说,这节课以实际问题为切入点,结合丰富的练习题,深入浅出地教授了分数乘整数的意义和计算方法,旨在让学生在实践中理解和运用数学知识,提高他们的数学素养和解决问题的能力。

    data.zip_数据结构_Visual_C++_

    每一道题目都是一个实际问题的抽象,通过解决这些问题,你不仅可以巩固理论知识,还能提升编程技能。此外,解题过程中的错误分析和调试也是提升编程能力的重要途径。 在学习过程中,建议先掌握基本概念,然后通过...

Global site tag (gtag.js) - Google Analytics