array专题6

78. Subsets

思路1:深度优先搜索:每个位置都有选,和 不选两种选择。退出条件是下标出界。

    public List<List<Integer>> subsetsV3(int[] nums) {List<List<Integer>> result = new ArrayList<List<Integer>>();List<Integer> list = new ArrayList<Integer>();dfs(result, list, nums, 0);return result;}private void dfs(List<List<Integer>> result, List<Integer> list, int[] nums, int idx) {if(idx==nums.length){result.add(new ArrayList<Integer>(list));return;}dfs(result, list, nums, idx+1);list.add(nums[idx]);dfs(result, list, nums, idx+1);list.remove(list.size()-1);}

思路2:使用位操作。有一个二进制数组,对应数组中每个元素是不是选择。例如数组[1,2,3],有一个二进制数组[0,0,0]表示一个元素都不选择;[0,0,1]选择nums[0]元素;[0,1,0]选择nums[1]元素;依次….

    public List<List<Integer>> subsetsV2(int[] nums) {List<List<Integer>> result = new ArrayList<List<Integer>>();int n = nums.length;int total = (1 << n) - 1;// 2^n-1for (int i = 0; i <= total; i++) {List<Integer> list = new ArrayList<Integer>();for(int j=0;j<n;j++){if(((i>>j) &1)==1){list.add(nums[j]);}}result.add(list);}return result;}

代码

90. Subsets II

思路:与78的深度优先搜索思路相同,只是要自己试试看哪些情况下出现重复的情况,在深度优先搜索过程中去掉这些情况。
代码

729 My Calendar I

思路:挨个比较
学习:放在map中,floorKey(key):比key小的最大值,ceilingKey(key):比key大的最小的值
代码

611 Valid Triangle Number

思路:这道题目首先会想到用深度优先搜索。每个位置的元素可以选择或者不选择。这样肯定能解决,问题是会超时。这时候肯定是需要观察,找规律。很遗憾,我没找到规律。
学习:条件是:三角形任意两边之和>第三边。如果两个小的边的和>最长的边,那一定符合要求。先确定最大的边:数组排序,下标最大的就是最大的边。
例如数组[2,2,3,4]。如果i=3,接着使用两个指针l=0,r=2;如果nums[0]+nums[2]>nums[3],那么,保持r不变,不断增加l,直到l=r,那这个过程中所有的nums[l]+nums[r]>nums[i]。所以这中间的组合有r-l个。具体来说就是:因为nums[0]+nums[2]>nums[3]符合要求,那么nums[1]+nums[2]>nums[3]肯定符合要求,所以组合数有r-l=2-0=2个。
代码

526 Beautiful Arrangement

思路:深度优先搜索,对于第i个位置,枚举可能使用的数值。可能使用的数值需要满足两个条件: 1 还未被使用;2 能够使得 数值%i=0或者 i%数值=0
学习:思路相同,但是迭代次数更少。
技巧一: backtracing从后面开始,这样每个搜索在失败之前都不会太深,因为较小的数字有更大的机会在他们自己之间被分割。解释是这样的。如果start从小到大,假如 start=1,nums[1]=4 ,那么4%1=0成立;start=4,nums[4]=3,那么4%3=0 或者 3%3=0都不成立;如果start从大到小,假如start=4, nums[4]=3,那么4%3=0 或者 3%3=0都不成立;则就不可能再去判断 nums[4]=3情况下: start=3,2,1的情况了,这样就会减少调用次数。
技巧二:Also I don’t use “visited” boolean array but use swap of an array of 1~N to avoid duplication.第start位能够选择的数字在nums[1~start]这些位置上的数字选择,每当选择一个,就把这个数字放在nums[start]位置上。这样别的位置就不能再选择这个数字了。聪明。

    public int countArrangementV2(int N) {if (N == 0)return 0;int[] nums = new int[N + 1];for (int i = 0; i <= N; i++)nums[i] = i;helper(nums, N);return count;}private void helper(int[] nums, int start) {if (start == 0) {count++;return;}for (int i = start; i > 0; i--) {swap(nums, start, i);if (nums[start] % start == 0 || start % nums[start] == 0)helper(nums, start - 1);swap(nums, i, start);}}private void swap(int[] nums, int i, int j) {int tmp = nums[i];nums[i] = nums[j];nums[j] = tmp;}

代码

667 Beautiful Arrangement II

思路:深度优先搜索,为每个位置上取值,再判断是否符合条件。超时。继续观察。想到差值最小是1,最大是n-1.如果 k<n1k<n−1怎么处理?我想到的是随机取。然后就思考不下去了。
学习:如果 k<n1k<n−1,那就先取k-1个差值:n-1,n-2….,最后的一个差值取1。
1,n,2,n-1,3,n-2,4… ==> Diff: n-1, n-2, n-3, n-4, n-5… k个数有k-1个不同的差值。
例如n=9,k=5,则取数据:1 9 2 8 3 4 5 6 7
例如n=9,k=6,则取数据:1 9 2 8 3 7 6 5 4
代码

238 Product of Array Except Self

思路:这道题目如果允许除法,很简单;如果是O(n^2)也很简单。所有数相乘,只是不乘以对应下标中nums[i]这个数。但是当要求不用除法,O(n)时间复杂度,就没有思路了。
给定数组nums={2,3,4,5}。
结果应该是:

Number2345
结果1*3*4*52*1*4*52*3*1*52*3*4*1

我思考到这里,就不知道该怎么做了。

学习:学习了一个思路,太鬼才了。在上面的表格中按1分隔,分为从左边乘和从右边乘。
考虑下标i=2,结果应该是2*3*1*5。这个结果可以看做是左边的2*3,和右边的5相乘得到的结果。

Number2345
左边22*32*3*4
右边5*4*35*45

把空的地方填上1

Number2345
左边122*32*3*4
右边5*4*35*451

通过两次循环实现。
如果能看到2*3*4=4*3*2也许能想到从右向左。如果能看到以1分隔,分别是nums[i]左边的数字和右边的数字相乘,也许能想到解决方法把。

    public int[] productExceptSelf(int[] nums) {int[] result = new int[nums.length];int left = 1;for (int i = 1; i <= nums.length; i++) {result[i-1] = left;left *= nums[i - 1];}int right = 1;for (int i = nums.length - 1; i >= 0; i--) {if(i<nums.length-1){right = right * nums[i+1];}result[i] = result[i] * right;}return result;}

代码

565 Array Nesting

思路:像环的检测, Floyd环之前做过。最简单:把每个i遍历一次,比较长度 。做完之后打印一下环,发现一个环内的元素无论从哪个元素开始,得到的长度都相同。例如nums=[5,4,0,3,1,6,2]。对于{5, 6, 2, 0}这样一个环,从下标0开始,长度是4;从下标2开始,长度还是4。所以需要去重,访问过的就不再访问了。提交后发现超时。
学习:这个时候我又在想我的思路错了?是不是还有更好的思路。看了讨论,发现是在去重那里优化。不需要用额外的空间存储是否访问过;访问过的可以标记为-1,因为初始每个元素是在[0,N-1]之间的。
改进的时候有两种:第一计算方式需要改进,这个时候需要重新观察数据。例如238。第二种是去重方式可以优化。例如本题。
学习2:本题还有用并查集解决的。
nums=[5,4,0,3,1,6,2]

index0123456
nums5403162
parent0123456
rank0000000

union(0,5):5的父节点是0:parent[5]=0;

index0123456
nums5403162
parent0123406
rank1000000

union(1,4):4的父节点是1:parent[4]=1;

index0123456
nums5403162
parent0123106
rank1100000

union(2,0):0的父节点是2:rootP=2,rootQ=0,rank[rootQ] > rank[rootP]=>parent[rootP] = rootQ;

index0123456
nums5403162
parent0103106
rank1100000

union(3,3):3的父节点是3:parent[3]=3;

index0123456
nums5403162
parent0103106
rank1100000

union(4,1):1的父节点是4:parent[1]=4,parent[4]=1 return;

index0123456
nums5403162
parent0103106
rank1100000

union(5,6):6的父节点是5:parent[6]=6,parent[5]=0 rank[rootQ] < rank[rootP] =>parent[6]=0

index0123456
nums5403162
parent0103100
rank1100000

union(6,2): rootP=0,rootQ=0

index0123456
nums5403162
parent0103100
rank1100000

最终有4处的parent都是0,所以长度为4,是最长的环。

[代码]

(https://github.com/Emma123456/leetcodepractice/blob/master/src/array/ArrayNesting565.java)

769 Max Chunks To Make Sorted

思路:刷leetcode,简直就是在找虐。我想到了,题目要求返回最多的分区数量。那如果每个都可以单独分开是最好的。但是不可以。例如nums=[1,0,2,3,4],0和1分开的话,排序就不对了,所以0和1必须在一起。那就是说,如果在当前位置i,如果arr[i+1]以及之后的数字都大于0到i位置的元素,那么就能分隔。这样的话,需要O(n^2)的复杂度。应该可以简化。
学习1:因为所有左边的元素 < 右边元素的时候可以形成新的分区,也就是说如果对于i下标:左边元素的最大值 <<<script type="math/tex" id="MathJax-Element-8"><</script> (i+1)到(n-1)位置的最小元素 =>可以形成新的分区。学习到的思路就是:用maxOfLeft数组,记录当当前位置最大值;而不是拿一个变量找全局最大值。

public int maxChunksToSorted(int[] arr) {int n = arr.length;int[] maxOfLeft = new int[n];maxOfLeft[0] = arr[0];for (int i = 1; i < n; i++) {maxOfLeft[i] = Math.max(arr[i], maxOfLeft[i - 1]);}int[] minOfRight = new int[n];minOfRight[n - 1] = arr[n - 1];for (int i = n - 2; i >= 0; i--) {minOfRight[i] = Math.min(arr[i], minOfRight[i + 1]);}int chunksCount = 0;for (int i = 0; i < n - 1; i++) {if (maxOfLeft[i] <= minOfRight[i + 1])chunksCount++;}return chunksCount + 1;}

学习2:用一个max数组保存到当前位置最大的元素,每个位置的最大元素和已经排序好的数组中的元素比较,如果max[i] = sorted[i],那在这个位置就可以分割。

nums:10234
max:11234
排序好的数组:01234
下标:01234

在题目中说明arr[i] 在 [0,arr.length - 1]所以排序后的数组一定是0,1,2,….n-1。就可以进一步省略max数组,只需要max变量。这里就是重点理解 max==i。max是到目前位置的最大值;i是排序好以后i位置元素的值。

    public int maxChunksToSortedV2(int[] arr) {int count = 0;int max = -1;for (int i = 0; i < arr.length; i++) {max = Math.max(max, arr[i]);if (max == i) {count++;}}return count;}

代码

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

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

相关文章

[Leetcode][第657题][JAVA][机器人能否返回原点][数组][HashMap]

【问题描述】[简单] 【解答思路】 遍历方向 看是否回到原点 或者 “上下” “左右”两个方向的数量是否相等 1. 方向 时间复杂度&#xff1a;O(N) 空间复杂度&#xff1a;O(1) class Solution {public boolean judgeCircle(String moves) {int x 0,y 0;int len moves.len…

网站公共部分的复用

一个网站的公共部分因为是一样的&#xff0c;可以通过公共部分的复用来减少代码的重复&#xff0c;同时也利于代码的后期维护。 关于公共部分复用的方法有很多&#xff0c;这里提供一些方法&#xff08;持续更新&#xff09; 利用ajax请求组件&#xff08;单个HTML&#xff09;…

第三十五期:当我们在讨论CQRS时,我们在讨论些神马?

thz 6月18日 当我写下这个标题的时候&#xff0c;我就有些后悔了&#xff0c;题目有点大&#xff0c;不太好控制。但我还是打算尝试一下&#xff0c;通过这篇内容来说清楚CQRS模式&#xff0c;以及和这个模式关联的其它东西。希望我能说得清楚&#xff0c;你能看得明白&#x…

【数据结构与算法】【算法思想】回溯算法

贪心算法 回溯算法 分治算法 动态规划 回溯算法思想应用广泛&#xff0c;除了用来指导深度优先搜索这种经典算法设计之外&#xff0c;还可以用在如正则表达式匹配&#xff0c;编译原理中的语法分析等。 除此之外&#xff0c;很多经典的数学问题都可以用回溯算法解决&#xff…

90 Subsets II

90 Subsets II leetcode第90题&#xff0c;用深度优先搜索的思路写出了三种不同的代码。一道题目&#xff0c;同样是深度优先搜索的问题&#xff0c;因为看问题角度不同&#xff0c;思路不同&#xff0c;代码也不一样 /*** [1,2,2,3] [] [1] [1,2] [1,2](重复) 情况&#xff1a…

2019 DDCTF 部分writeup

网上的wp已经很多了&#xff0c;但wp普遍很简略。我尽量写的详细一点。 一、WEB 滴~ 拿到题目后首先右键查看源代码&#xff0c;发现图片是以base64传送的 而且看url发现里面应该是包含了文件名&#xff0c;并且用了某个编码。测试过后是转16进制ascii码后两层bases64 &#xf…

第三十六期:如果把线程当作一个人来对待,所有问题都瞬间明白了

7月8日 以下文章来源于编程新说 &#xff0c;作者编程新说李新杰 多线程的问题都曾经困扰过每个开发人员&#xff0c;今天将从全新视角来解说&#xff0c;希望读者都能明白。 强烈建议去运行下文章中的示例代码&#xff0c;自己体会下。 问题究竟出在哪里&#xff1f; 一个…

[Leetcode][第322题][JAVA][零钱兑换][回溯][记忆化搜索][动态规划]

【问题描述】[中等] 【解答思路】 1. 递归&#xff08;超时&#xff09; class Solution {int res Integer.MAX_VALUE;public int coinChange(int[] coins, int amount) {if(coins.length 0){return -1;}findWay(coins,amount,0);// 如果没有任何一种硬币组合能组成总金额&…

用BST解决729. My Calendar I 731. My Calendar II 732. My Calendar III

My Calendar的book方法实现指定开始时间、结束时间&#xff0c;在重叠次数要求不同的情况下怎么实现。 729 My Calendar I 要求任意两个事件之间不能重叠。如果要插入的事件和已经插入的事件不重叠&#xff0c;则插入&#xff1b;否则不插入。 731 MyCalendar II 要求任意三个…

[转载]抓大放小,要事为先

最近看到了swing框架&#xff0c;细节颇多。细细看来效率很低&#xff0c;偶然找到了这篇文章&#xff0c;恩确实有道理&#xff1b;以下&#xff1b; 对待人生的任何事情都要&#xff1a;抓大放小&#xff0c;要事为先 对于一个以前从来没有接触过java的人&#xff0c;java无疑…

第三十七期:如果你这样回答“什么是线程安全”,面试官都会对你刮目相看

6月12日 以下文章来源于编程新说 &#xff0c;作者编程新说李新杰 有读者跟我说&#xff0c;喜欢看我的文章&#xff0c;说很容易读&#xff0c;我确实在易读性上花费的心思不亚于在内容上。因为我不喜欢一上来就堆很多东西&#xff0c;而且把简单的东西搞得复杂人人都会&…

第三十八期:如何在Windows 10上使用Windows Update目录驱动程序安装打印机

尽管Win10能够自动设置大多数打印机&#xff0c;但有时Windows Update可能会在安装设备驱动程序时遇到问题-尤其是在添加较旧的打印机时。 作者&#xff1a;佚名来源&#xff1a;Win10系统之家 图片来源&#xff1a;伏天氏(m.futianshuwu.com) 伏天书屋(futianshuwu.com)10月…

MySQL 表和列的注释

像代码一样&#xff0c;可以为表以及表中的列添加注释&#xff0c;方便其他人知晓其功能。对于一些字段&#xff0c;在经过一定时间后&#xff0c;创建者未必也能想起其具体的含意&#xff0c;所以注释显得尤为重要。 注释的添加 注释的添加是通过在定义表或列的时候在末尾加上…

376 Wiggle Subsequence 贪心解法以及证明

376. Wiggle Subsequence 题目理解 给定一个数组&#xff0c;相邻两个数计算差值。差值排成的序列是正负相间的&#xff0c;那这个数组就是一个wiggle 数组。例如数组[1,7,4,9,2,5]&#xff0c;差值序列是(6,-3,5,-7,3)。原数组用坐标轴表示如下。 思路是&#xff1a;在一段…

【数据结构与算法】【算法思想】动态规划

贪心算法 回溯算法 分治算法 动态规划 贪心&#xff1a;一条路走到黑&#xff0c;就一次机会&#xff0c;只能哪边看着顺眼走哪边 回溯&#xff1a;一条路走到黑&#xff0c;无数次重来的机会&#xff0c;还怕我走不出来 (Snapshot View) 动态规划&#xff1a;拥有上帝视角&am…

第六十七期:Python爬虫44万条数据揭秘:如何成为网易音乐评论区的网红段子手

获取数据,其实逻辑并不复杂&#xff1a;爬取歌单列表里的所有歌单url、进入每篇歌单爬取所有歌曲url&#xff0c;去重、进入每首歌曲首页爬取热评&#xff0c;汇总。 作者&#xff1a;嗨学python来源&#xff1a;今日头条 获取数据 其实逻辑并不复杂&#xff1a; 爬取歌单列…

第一阶段SCRUM冲刺 08

昨天的成就&#xff1a;实现任务查找模块。感觉到了硬件支持对软件编程的重要性。 遇到的难题&#xff1a;电脑出现卡顿&#xff0c;编程工具十分卡&#xff0c;重启电脑好几次解决这个问题。电脑也是需要休息的。 今天的任务&#xff1a;进行资料上传模块&#xff1b;发布任务…

array专题7

714 Best Time to Buy and Sell Stock with Transaction Fee 思路 首先是暴力枚举。考虑在第idx天能做的操作&#xff1a;买&#xff1f;卖&#xff1f;不操作&#xff1f; /*** 暴力枚举* * param prices* param fee* return*/public int maxProfitV99(int[] prices, int f…

第三十九期:收藏 | 第一次有人把“分布式事务”讲的这么简单明了

不知道你是否遇到过这样的情况&#xff0c;去小卖铺买东西&#xff0c;付了钱&#xff0c;但是店主因为处理了一些其他事&#xff0c;居然忘记你付了钱&#xff0c;又叫你重新付。 作者&#xff1a;咖啡拿铁来源 又或者在网上购物明明已经扣款&#xff0c;但是却告诉我没有发…

【小技巧】字符char与整型int的相互转换

char转int char与int的相互转化&#xff0c;联想ASCII码&#xff0c;字符‘0’对应的值为48&#xff0c;所以不能直接加减‘ ’ char ch9; int ch_intch-0;//此时ch_int9int转char int i9&#xff1b; char i_chi0;//此时i_ch9必须牢记的ASCII