算法习题之动态规划
创始人
2024-05-31 11:23:59
0

动态规划

  • 习题1 打印n层汉诺塔从最左边移动到最右边的全部过程
  • 习题2 给你一个栈,请你逆序这个栈,不能申请额外的数据结构,只能使用递归函数。 如何实现?
  • 习题3 打印一个字符串的全部子序列,打印一个字符串的全部子序列,要求不要出现重复字面值的子序列
  • 习题4 打印一个字符串的全部排列,要求不要出现重复的排列

1.什么暴力递归可以继续优化?

有重复调用同一个子问题的解,这种递归可以优化

如果每一个子问题都是不同的解,无法优化也不用优化

2.暴力递归和动态规划的关系

某一个暴力递归,有解的重复调用,就可以把这个暴力递归优化成动态规划

任何动态规划问题,都一定对应着某一个有重复过程的暴力递归

但不是所有的暴力递归,都一定对应着动态规划

3.如何找到某个问题的动态规划方式?

1)设计暴力递归:重要原则+4种常见尝试模型!重点!

2)分析有没有重复解:套路解决

3)用记忆化搜索 -> 用严格表结构实现动态规划:套路解决

4)看看能否继续优化:套路解决

4.面试中设计暴力递归过程的原则

1)每一个可变参数的类型,一定不要比int类型更加复杂

2)原则1)可以违反,让类型突破到一维线性结构,那必须是单一可变参数

3)如果发现原则1)被违反,但不违反原则2),只需要做到记忆化搜索即可

4)可变参数的个数,能少则少

5.常见的4种尝试模型

1)从左往右的尝试模型

2)范围上的尝试模型

3)多样本位置全对应的尝试模型

4)寻找业务限制的尝试模型

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

public static void hanoi1(int n) {leftToRight(n);}// 请把1~N层圆盘 从左 -> 右public static void leftToRight(int n) {if (n == 1) { // base caseSystem.out.println("Move 1 from left to right");return;}leftToMid(n - 1);System.out.println("Move " + n + " from left to right");midToRight(n - 1);}// 请把1~N层圆盘 从左 -> 中public static void leftToMid(int n) {if (n == 1) {System.out.println("Move 1 from left to mid");return;}leftToRight(n - 1);System.out.println("Move " + n + " from left to mid");rightToMid(n - 1);}public static void rightToMid(int n) {if (n == 1) {System.out.println("Move 1 from right to mid");return;}rightToLeft(n - 1);System.out.println("Move " + n + " from right to mid");leftToMid(n - 1);}public static void midToRight(int n) {if (n == 1) {System.out.println("Move 1 from mid to right");return;}midToLeft(n - 1);System.out.println("Move " + n + " from mid to right");leftToRight(n - 1);}public static void midToLeft(int n) {if (n == 1) {System.out.println("Move 1 from mid to left");return;}midToRight(n - 1);System.out.println("Move " + n + " from mid to left");rightToLeft(n - 1);}public static void rightToLeft(int n) {if (n == 1) {System.out.println("Move 1 from right to left");return;}rightToMid(n - 1);System.out.println("Move " + n + " from right to left");midToLeft(n - 1);}public static void hanoi2(int n) {if (n > 0) {func(n, "left", "right", "mid");}}public static void func(int N, String from, String to, String other) {if (N == 1) { // baseSystem.out.println("Move 1 from " + from + " to " + to);} else {func(N - 1, from, other, to);System.out.println("Move " + N + " from " + from + " to " + to);func(N - 1, other, to, from);}}public static class Record {public boolean finish1;public int base;public String from;public String to;public String other;public Record(boolean f1, int b, String f, String t, String o) {finish1 = false;base = b;from = f;to = t;other = o;}}public static void hanoi3(int N) {if (N < 1) {return;}Stack stack = new Stack<>();stack.add(new Record(false, N, "left", "right", "mid"));while (!stack.isEmpty()) {Record cur = stack.pop();if (cur.base == 1) {System.out.println("Move 1 from " + cur.from + " to " + cur.to);if (!stack.isEmpty()) {stack.peek().finish1 = true;}} else {if (!cur.finish1) {stack.push(cur);stack.push(new Record(false, cur.base - 1, cur.from, cur.other, cur.to));} else {System.out.println("Move " + cur.base + " from " + cur.from + " to " + cur.to);stack.push(new Record(false, cur.base - 1, cur.other, cur.to, cur.from));}}}}public static void main(String[] args) {int n = 3;hanoi1(n);System.out.println("============");hanoi2(n);
//		System.out.println("============");
//		hanoi3(n);}

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

public static void reverse(Stack stack) {if (stack.isEmpty()) {return;}int i = f(stack);reverse(stack);stack.push(i);}// 栈底元素移除掉// 上面的元素盖下来// 返回移除掉的栈底元素public static int f(Stack stack) {int result = stack.pop();if (stack.isEmpty()) {return result;} else {int last = f(stack);stack.push(result);return last;}}public static void main(String[] args) {Stack test = new Stack();test.push(1);test.push(2);test.push(3);test.push(4);test.push(5);reverse(test);while (!test.isEmpty()) {System.out.println(test.pop());}}

习题3 打印一个字符串的全部子序列,打印一个字符串的全部子序列,要求不要出现重复字面值的子序列

// s -> "abc" ->public static List subs(String s) {char[] str = s.toCharArray();String path = "";List ans = new ArrayList<>();process1(str, 0, ans, path);return ans;}// str 固定参数// 来到了str[index]字符,index是位置// str[0..index-1]已经走过了!之前的决定,都在path上// 之前的决定已经不能改变了,就是path// str[index....]还能决定,之前已经确定,而后面还能自由选择的话,// 把所有生成的子序列,放入到ans里去public static void process1(char[] str, int index, List ans, String path) {if (index == str.length) {ans.add(path);return;}// 没有要index位置的字符process1(str, index + 1, ans, path);// 要了index位置的字符process1(str, index + 1, ans, path + String.valueOf(str[index]));}public static List subsNoRepeat(String s) {char[] str = s.toCharArray();String path = "";HashSet set = new HashSet<>();process2(str, 0, set, path);List ans = new ArrayList<>();for (String cur : set) {ans.add(cur);}return ans;}public static void process2(char[] str, int index, HashSet set, String path) {if (index == str.length) {set.add(path);return;}String no = path;process2(str, index + 1, set, no);String yes = path + String.valueOf(str[index]);process2(str, index + 1, set, yes);}public static void main(String[] args) {String test = "acccc";List ans1 = subs(test);List ans2 = subsNoRepeat(test);for (String str : ans1) {System.out.println(str);}System.out.println("=================");for (String str : ans2) {System.out.println(str);}System.out.println("=================");}

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

public static List permutation1(String s) {List ans = new ArrayList<>();if (s == null || s.length() == 0) {return ans;}char[] str = s.toCharArray();ArrayList rest = new ArrayList();for (char cha : str) {rest.add(cha);}String path = "";f(rest, path, ans);return ans;}public static void f(ArrayList rest, String path, List ans) {if (rest.isEmpty()) {ans.add(path);} else {int N = rest.size();for (int i = 0; i < N; i++) {char cur = rest.get(i);rest.remove(i);f(rest, path + cur, ans);rest.add(i, cur);}}}public static List permutation2(String s) {List ans = new ArrayList<>();if (s == null || s.length() == 0) {return ans;}char[] str = s.toCharArray();g1(str, 0, ans);return ans;}public static void g1(char[] str, int index, List ans) {if (index == str.length) {ans.add(String.valueOf(str));} else {for (int i = index; i < str.length; i++) {swap(str, index, i);g1(str, index + 1, ans);swap(str, index, i);}}}public static List permutation3(String s) {List ans = new ArrayList<>();if (s == null || s.length() == 0) {return ans;}char[] str = s.toCharArray();g2(str, 0, ans);return ans;}public static void g2(char[] str, int index, List ans) {if (index == str.length) {ans.add(String.valueOf(str));} else {boolean[] visited = new boolean[256];for (int i = index; i < str.length; i++) {if (!visited[str[i]]) {visited[str[i]] = true;swap(str, index, i);g2(str, index + 1, ans);swap(str, index, i);}}}}public static void swap(char[] chs, int i, int j) {char tmp = chs[i];chs[i] = chs[j];chs[j] = tmp;}public static void main(String[] args) {String s = "acc";List ans1 = permutation1(s);for (String str : ans1) {System.out.println(str);}System.out.println("=======");List ans2 = permutation2(s);for (String str : ans2) {System.out.println(str);}System.out.println("=======");List ans3 = permutation3(s);for (String str : ans3) {System.out.println(str);}}

相关内容

热门资讯

巴基斯坦总理与巴各政党领导人通... 【环球网快讯】印巴局势升级引发广泛关注。据巴基斯坦《黎明报》10日援引巴基斯坦国家电视台PTV最新消...
这一天!邯郸市中心医院…… 转自:邯郸新闻网邯郸道 天使情——邯郸市中心医院的特色护士节活动编者按:在国际护士节来临之际,邯郸市...
泰国4月份通胀率现负增长 转自:中国新闻网  中新社曼谷5月10日电 据泰国《民族报》10日报道,泰国4月份通货膨胀率为-0....
5月12日现场对接!辽宁省工信... 原标题:需求清单,向全球发布!  需求是市场,是合作的开始。近日,省工业信息化厅围绕重点产业链条、工...
特朗普高兴早了!首位美籍罗马教... 当地时间5月8日,枢机主教罗伯特·普雷沃斯特当选第267任天主教罗马教皇,称为利奥十四世,成为首位美...
马光远谈中国文化IP:中国每一...   2025世界IP经济发展大会暨全球IP授权博览会将于5月10日-5月12日在广州举行。著名经济学...
媒体评:学术近亲繁殖伤害教育公... 【媒体评:#学术近亲繁殖伤害教育公平#】近日,因发表14篇SCI论文,并获得3项国家发明专利,重庆大...
熟练到脑子跟不上嘴!护理专业男... 【熟练到脑子跟不上嘴!#护理专业男生丝滑操作婴儿急救#[666]】5月9日,四川一护理专业男同学,在...
深化改革开放再发力 国务院部署... 深化对外开放,国务院又有新部署。5月9日,国务院常务会议听取推动自贸试验区建设提质增效工作汇报,并研...
46岁章子怡,突传喜讯! 今天,章子怡工作室发文官宣好消息:《卧虎藏龙》25周年之际,章子怡在GoldGala出席奥斯卡博物馆...
面板双虎4月营收持平,Q2电视... 来源:Wit Display面板双虎友达光电今天公布4月合并营收新台币231.4亿元新台币,较3月减...
16楼“飞”出电竞椅!原因查明... 据四川成都市公安局高新技术产业开发区分局消息近日成都高新区某小区发生了一起高空抛物事件一男子情绪失控...
富采Q1亏1亿! 来源:Wit DisplayLED大厂富采5月9日举行法说会,公布2025年第1季财报。尽管单季仍呈...
小市值、反转因子表现较好   炒股就看金麒麟分析师研报,权威,专业,及时,全面,助您挖掘潜力主题机会! 因子IC跟踪    ...
全球专家共话科技与产业融合的脑... 5月10日,“2025浦江创新论坛”第七届神经科技国际创新论坛暨中国神经科学学会神经调控基础与转化分...
璀璨夺目!这场焰火展演点亮南昌... 转自:南昌发布5月9日晚,随着暮色降临,“星焰焕新·城启璀璨”——中国南昌烟花爆竹产业博览会城市焰火...
5月楼市开局冷热分化 公积金利... (转自:地产红榜)  5月1日国家新版《住宅项目规范》正式实施。从“五一”假期楼市表现来看,分化明显...
清华大学夏清:新型电力系统的构...   5月10日消息,远东控股集团40周年庆暨企业家论坛今天在宜兴举行。清华大学电机系教授、国家能源互...
十二载从无到有 海军舰载航空兵... 今天(5月10日)是人民海军首支舰载航空兵部队成立12周年的日子。12年前,舰载航空兵部队在渤海湾畔...
今天,该如何看尖岗山? 来源:苗头有朋友打来电话,说完全同意我的意见,全款买了一套万科未来之光80平方米的房子。有时候,你不...