LeetCode习题集 有些题可能直接略过了,整理一下之前刷leetcode

441. 排列硬币

你总共有 n 枚硬币,你需要将它们摆成一个阶梯形状,第 k 行就必须正好有 k 枚硬币。

给定一个数字 n,找出可形成完整阶梯行的总行数。

n 是一个非负整数,并且在32位有符号整型的范围内。

示例 1:

n = 5

硬币可排列成以下几行:

¤
¤ ¤
¤ ¤

因为第三行不完整,所以返回2. 示例 2:

n = 8

硬币可排列成以下几行:

¤
¤ ¤
¤ ¤ ¤
¤ ¤

因为第四行不完整,所以返回3.

根据数学公式,k(k+1) /2 = n,可以得到其正数解为:k = sqrt(2n+1/4) - 1/2。然后求整即可。
唯一的问题是,这里2n+1/4有可能会超出sqrt函数的参数范围。
于是,我们可以变换一下, k = sqrt(2) * sqrt(n+1/8) - 1/2,这样求平方根就不会超限了。
于是,我们就有了这么一行代码
class Solution {
    public int arrangeCoins(int n) {
      return (int) (-1 + Math.sqrt(1 + 8 * (long) n)) / 2;
    }
}

442. 数组中重复的数据

给定一个整数数组 a,其中1 ≤ a[i] ≤ n (n为数组长度), 其中有些元素出现两次而其他元素出现一次。

找到所有出现两次的元素。

你可以不用到任何额外空间并在O(n)时间复杂度内解决这个问题吗?

示例:

输入: [4,3,2,7,8,2,3,1]

输出: [2,3]

 * 这个题属于技巧题 首先仔细看输入的给定的数组值 该值的区间为 1 ≤ a[i] ≤ n
 * 这其实是这道题解题的关键点,好好利用这个信息。 某些元素出现了两次,
 * 而一些其他的元素只出现了1次,我们可以利用这些元素在出现次数上面的不一样做文章。
 *
 * 仔细观察发现1 ≤ a[i] ≤ n 这个条件,正好和我们数组的下标差1,我们可以按照数值
 * 来遍历数组,那么在数组中具有相同值的元素,会被经过两次,那么我们只要想出一种方式
 * 在这个遍历结束后可以区分,哪些元素被经过了多次即可,由于数组元素具有1 ≤ a[i] ≤ n
 * 这样的范围,那其实我们当每次经过一个元素时,给他加上n,当遍历结束时,我们再次遍历数组
 * 那些数值超过2n的元素索引+1,对应的就是我们的出现了两次的元素。
class Solution {
    
    public List<Integer> findDuplicates(int[] nums) {
        List<Integer> ret = new ArrayList<>();

        int n = nums.length;
        for(int i = 0; i < n; i++){
            nums[(nums[i] - 1) % n] += n;
        }

        for(int i = 0; i < n; i++){
            if(nums[i] > 2 * n) ret.add(i+1);
        }
        return ret;
    }
}

443. 压缩字符串

给定一组字符,使用原地算法将其压缩。

压缩后的长度必须始终小于或等于原数组长度。

数组的每个元素应该是长度为1 的字符(不是 int 整数类型)。

在完成原地修改输入数组后,返回数组的新长度。

进阶: 你能否仅使用O(1) 空间解决问题?

示例 1:

输入:

["a","a","b","b","c","c","c"]

输出: 返回6,输入数组的前6个字符应该是:["a","2","b","2","c","3"]

说明: "aa"被"a2"替代。"bb"被"b2"替代。"ccc"被"c3"替代。 示例 2:

输入: ["a"]

输出: 返回1,输入数组的前1个字符应该是:["a"]

说明: 没有任何字符串被替代。 示例 3:

输入:

["a","b","b","b","b","b","b","b","b","b","b","b","b"]

输出: 返回4,输入数组的前4个字符应该是:["a","b","1","2"]。

说明: 由于字符"a"不重复,所以不会被压缩。"bbbbbbbbbbbb"被“b12”替代。 注意每个数字在数组中都有它自己的位置。 注意:

所有字符都有一个ASCII值在[35, 126]区间内。 1 <= len(chars) <= 1000。

循环,如果有连续的,和上一个字符相同的就记录他,用count记录次数
等到和前面的不相等的时候,就把这些字符放进数组,
    把count字符化,每个位置只能放一个字符,如果count大于10的话,需要转成String,再换成字符放进数组
    循环后还要判断一下,防止最后一个位置的内容没放进去
class Solution {
   public int compress(char[] chars) {
        if (chars.length == 0) return 0;
        char temp = chars[0];
        int index = 0;
        int count = 1;
        for (int i = 1; i < chars.length; i++) {
            if (chars[i] != temp) {
                chars[index++] = temp;
                temp = chars[i];
                if (count > 1) {
                    for (char c : Integer.toString(count).toCharArray()) {
                        chars[index++] = c;
                    }
                }
                count = 1;
                continue;
            }
            count++;
        }
        chars[index++] = chars[chars.length - 1];
        if (count > 1) {
            for (char c : Integer.toString(count).toCharArray()) {
                chars[index++] = c;
            }
        }
        return index;
    }
}

445. 两数相加 II

给定两个非空链表来代表两个非负整数。数字最高位位于链表开始位置。它们的每个节点只存储单个数字。将这两数相加会返回一个新的链表。

你可以假设除了数字 0 之外,这两个数字都不会以零开头。

进阶:

如果输入链表不能修改该如何处理?换句话说,你不能对列表中的节点进行翻转。

示例:

输入: (7 -> 2 -> 4 -> 3) + (5 -> 6 -> 4) 输出: 7 -> 8 -> 0 -> 7

先把两个链表的值都分别拿出来,找个list集合存放(倒叙存放在list集合里面,)
集合前面是低位,后面是高位
每次相加都保留好进位,然后当前位为和%10
下一位相加的时候加上进位
如果一个空了还是照常加,默认值为0
如果循环完,进位不为0,说明还有一个进位

每次当前位置计算完都创建一个结点放入链表

/**
 * Definition for singly-linked list.
 * public class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode(int x) { val = x; }
 * }
 */
class Solution {
      public ListNode addTwoNumbers(ListNode l1, ListNode l2) {
        //栈
        LinkedList<Integer> s1 = new LinkedList<>();
        LinkedList<Integer> s2 = new LinkedList<>();
        while(l1 != null) {
            s1.addFirst(l1.val);
            l1 = l1.next;
        }
        while(l2 != null) {
            s2.addFirst(l2.val);
            l2 = l2.next;
        }
        int carry = 0;
        ListNode lastNode = null;
        while(!s1.isEmpty() || !s2.isEmpty()) {
            int a1 = 0, a2 = 0;
            if(!s1.isEmpty()) {
                a1 = s1.removeFirst();
            }
            if(!s2.isEmpty()) {
                a2 = s2.removeFirst();
            }
            ListNode curNode = new ListNode((a1 + a2 + carry) % 10);
            carry = (a1 + a2 + carry) / 10;
            curNode.next = lastNode;
            lastNode = curNode;
        }
        if(carry > 0) {
            ListNode curNode = new ListNode(carry);
            curNode.next = lastNode;
            lastNode = curNode;
        }
        return lastNode;
    }
}

446. 等差数列划分 II - 子序列

如果一个数列至少有三个元素,并且任意两个相邻元素之差相同,则称该数列为等差数列。

例如,以下数列为等差数列:

1, 3, 5, 7, 9 7, 7, 7, 7 3, -1, -5, -9 以下数列不是等差数列。

1, 1, 2, 5, 7

数组 A 包含 N 个数,且索引从 0 开始。该数组子序列将划分为整数序列 (P0, P1, ..., Pk),P 与 Q 是整数且满足 0 ≤ P0 < P1 < ... < Pk < N。

如果序列 A[P0],A[P1],...,A[Pk-1],A[Pk] 是等差的,那么数组 A 的子序列 (P0,P1,…,PK) 称为等差序列。值得注意的是,这意味着 k ≥ 2。

函数要返回数组 A 中所有等差子序列的个数。

输入包含 N 个整数。每个整数都在 -231 和 231-1 之间,另外 0 ≤ N ≤ 1000。保证输出小于 231-1。

示例:

输入:[2, 4, 6, 8, 10]

输出:7

解释:

所有的等差子序列为:
[2,4,6]
[4,6,8]
[6,8,10]
[2,4,6,8]
[4,6,8,10]
[2,4,6,8,10]
[2,6,10]
 
	等差数列,差相等,我for循环,把差当作map的key,数量当作value,然后暴力大法
        
        ```
```java
class Solution {
    public int numberOfArithmeticSlices(int[] A) {
        int n = A.length;
        long ans = 0;
        Map<Integer, Integer>[] cnt = new Map[n];
        for (int i = 0; i < n; i++) {
            cnt[i] = new HashMap<>(i);
            for (int j = 0; j < i; j++) {
                long delta = (long)A[i] - (long)A[j];
                if (delta < Integer.MIN_VALUE || delta > Integer.MAX_VALUE) {
                    continue;
                }
                int diff = (int)delta;
                //找到i为结束位置的等差数列的数量
                int sum = cnt[j].getOrDefault(diff, 0); 
                //找到j为结束位置的等差数列的数量
                int origin = cnt[i].getOrDefault(diff, 0);
                //当前i位置的等差数列的数量=以前的等差数列数量+j位置的等差数列的数量和等差diff能凑成i位置的新构成方法
                cnt[i].put(diff, origin + sum + 1);
                ans += sum;
            }
        }
        return (int)ans;        
    }
 
}

447. 回旋镖的数量

给定平面上 n 对不同的点,“回旋镖” 是由点表示的元组 (i, j, k) ,其中 i 和 j 之间的距离和 i 和 k 之间的距离相等(需要考虑元组的顺序)。

找到所有回旋镖的数量。你可以假设 n 最大为 500,所有点的坐标在闭区间 [-10000, 10000] 中。

示例:

输入: [[0,0],[1,0],[2,0]]

输出: 2

解释: 两个回旋镖为 [[1,0],[0,0],[2,0]] 和 [[1,0],[2,0],[0,0]]

这道题思路其实也比较简单,计算一点和其他点之间的距离,使用哈希表存储,
  若同一距离出现多次,则可以形成回旋镖。
  假设同一距离出现 n 次,由数字规律可推出回旋镖的数量 sum = n*(n-1) 。
  本人开始只能做到存储到哈希表,然后按该公式累加得到最后结果。
  参考了速度第一的答案,优化如下:
      假设当前同一距离的数量为 n, 回旋镖数量为 n*(n-1), 
      当再出现一个同一距离时,回旋镖的数量应为 (n+1)*n,与之前相差 (n+1)*n - n*(n-1) = 2*n, 
      所以只需要把最后答案加上 2*n, 最后 n+1 再存储到哈希表中。
  
class Solution {
 

     public int numberOfBoomerangs(int[][] points) {
        int len = points.length;
	int ans = 0;
	HashMap<Double, Integer> map = new HashMap<Double, Integer>();
	for(int i = 0; i < len; i++){
	     for(int j = 0; j < len; j++){
		if(i != j){
		    double dis = Math.pow(points[i][0] - points[j][0], 2)
				+ Math.pow(points[i][1] - points[j][1], 2);
		    if(!map.containsKey(dis)){
			map.put(dis, 1);
		    }else{
			int n = map.get(dis);
			ans += 2 * n;
			map.put(dis, 1+n);
		    }
		}
	    }
	    map.clear();
	}	
	return ans;
    }
}

448. 找到所有数组中消失的数字

给定一个范围在 1 ≤ a[i] ≤ n ( n = 数组大小 ) 的 整型数组,数组中的元素一些出现了两次,另一些只出现一次。

找到所有在 [1, n] 范围之间没有出现在数组中的数字。

您能在不使用额外空间且时间复杂度为O(n)的情况下完成这个任务吗? 你可以假定返回的数组不算在额外空间内。

示例:

输入: [4,3,2,7,8,2,3,1]

输出: [5,6] 通过次数29,638提交次数51,956

采用类似计数排序
    把出现的数,当作下标去处理,如果当前值存在,并且对应下标的值为正数,就把对应下标的值变为负数,
                即使第二次出现,对应值的数也变成负数了,不会在计算了
        循环整个数组,把大于零的位置的下标都添加进去,只要大于0说明对应下标没出现过
        这里因为是下标,用的时候是 值-1   所以下标变值也是 下标+1
class Solution {
    public List<Integer> findDisappearedNumbers(int[] nums) {
        List<Integer> results = new ArrayList<>();
        for (int i = 0; i < nums.length; i++) {
            if (nums[Math.abs(nums[i]) - 1] > 0) {
                nums[Math.abs(nums[i]) - 1] = - nums[Math.abs(nums[i]) - 1];
            }
        }
        for (int i = 0; i < nums.length; i++) {
            if (nums[i] > 0) {
                results.add(i + 1);
            }
        }
        return results;
    }
}

449. 序列化和反序列化二叉搜索树

序列化是将数据结构或对象转换为一系列位的过程,以便它可以存储在文件或内存缓冲区中,或通过网络连接链路传输,以便稍后在同一个或另一个计算机环境中重建。

设计一个算法来序列化和反序列化二叉搜索树。 对序列化/反序列化算法的工作方式没有限制。 您只需确保二叉搜索树可以序列化为字符串,并且可以将该字符串反序列化为最初的二叉搜索树。

编码的字符串应尽可能紧凑。

注意:不要使用类成员/全局/静态变量来存储状态。 你的序列化和反序列化算法应该是无状态的。

/**
 * Definition for a binary tree node.
 * public class TreeNode {
 *     int val;
 *     TreeNode left;
 *     TreeNode right;
 *     TreeNode(int x) { val = x; }
 * }
 */
public class Codec {
 public int i = 0;
 
    public String serialize(TreeNode root) {
        StringBuilder sb = new StringBuilder();
        doSerialize(root, sb);
        return sb.toString();
    }

    public void doSerialize(TreeNode root, StringBuilder sb) {
        if (root == null) {
            return;
        }
        sb.append((char) root.val);
        doSerialize(root.left, sb);
        doSerialize(root.right, sb);
    }
 
    public TreeNode deserialize(String data) {
        char[] arr = data.toCharArray();
        return doDescrialize(arr, Integer.MIN_VALUE, Integer.MAX_VALUE);
    }

    private TreeNode doDescrialize(char[] arr, int minValue, int maxValue) {
        if (i >= arr.length || (char) arr[i] > maxValue) {
            return null;
        }
        TreeNode root = new TreeNode((char) arr[i++]);
        root.left = doDescrialize(arr, minValue, root.val);
        root.right = doDescrialize(arr, root.val, maxValue);
        return root;
    }
}

// Your Codec object will be instantiated and called as such:
// Codec codec = new Codec();
// codec.deserialize(codec.serialize(root));

450. 删除二叉搜索树中的节点

给定一个二叉搜索树的根节点 root 和一个值 key,删除二叉搜索树中的 key 对应的节点,并保证二叉搜索树的性质不变。返回二叉搜索树(有可能被更新)的根节点的引用。

一般来说,删除节点可分为两个步骤:

首先找到需要删除的节点; 如果找到了,删除它。 说明: 要求算法时间复杂度为 O(h),h 为树的高度。

示例:

root = [5,3,6,2,4,null,7] key = 3

    5
   / \
  3   6
 / \   \
2   4   7

给定需要删除的节点值是 3,所以我们首先找到 3 这个节点,然后删除它。

一个正确的答案是 [5,4,6,2,null,null,7], 如下图所示。

    5
   / \
  4   6
 /     \
2       7

另一个正确答案是 [5,2,6,null,4,null,7]。

    5
   / \
  2   6
   \   \
    4   7
/**
 * Definition for a binary tree node.
 * public class TreeNode {
 *     int val;
 *     TreeNode left;
 *     TreeNode right;
 *     TreeNode(int x) { val = x; }
 * }
 */
class Solution {
    public TreeNode deleteNode(TreeNode root, int key) {
        if (root == null) {
            return null;
        }
        if (key < root.val) {
            // 待删除节点在左子树中
            root.left = deleteNode(root.left, key);
            return root;
        } else if (key > root.val) {
            // 待删除节点在右子树中
            root.right = deleteNode(root.right, key);
            return root;
        } else {
            // key == root.val,root 为待删除节点
            if (root.left == null) {
                // 返回右子树作为新的根
                return root.right;
            } else if (root.right == null) {
                // 返回左子树作为新的根
                return root.left;
            } else {
                // 左右子树都存在,返回后继节点(右子树最左叶子)作为新的根
                TreeNode successor = min(root.right);
                successor.right = deleteMin(root.right);
                successor.left = root.left;
                return successor;
            }
        }
    }

    private TreeNode min(TreeNode node) {
        if (node.left == null) {
            return node;
        }
        return min(node.left);
    }

    private TreeNode deleteMin(TreeNode node) {
        if (node.left == null) {
            return node.right;
        }
        node.left = deleteMin(node.left);
        return node;
    }
}