刷题碎碎念---力扣001-006

001--两数之和

比较简单,暴力法就能过,可以用HashMap优化一下

class Solution {
    public int[] twoSum(int[] nums, int target) {
       HashMap<Integer,Integer> maps = new HashMap<Integer,Integer>();

       for(int i=0; i<nums.length; i++){
           if(maps.containsKey(target - nums[i])){
               int j = maps.get(target-nums[i]);
               return new int[]{i,j};
           }

           maps.put(nums[i],i);
       }

       return new int[0];
    }
}

002---两数相加

比较简单吧,就是一个链表的遍历,然后手动进行一个加法,没啥难度,就是注意一下最后那个进位就行了。

/**
 * Definition for singly-linked list.
 * public class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode() {}
 *     ListNode(int val) { this.val = val; }
 *     ListNode(int val, ListNode next) { this.val = val; this.next = next; }
 * }
 */
class Solution {
    public ListNode addTwoNumbers(ListNode l1, ListNode l2) {
        ListNode head = new ListNode();

        ListNode node;
        node = head;

        int flag = 0;
        while(l1 != null || l2 != null){
            
            int i = l1 != null ? l1.val : 0;
            int j = l2 != null ? l2.val : 0;

            int num = i + j + flag;
            flag = 0; // 进位标志
            if (num >= 10){
                flag = 1;
                num %= 10;
            }
            node.next = new ListNode(num);
            node = node.next;
            l1 = l1 != null ? l1.next : null;
            l2 = l2 != null ? l2.next : null;
        }

        if(flag == 1) node.next = new ListNode(flag);

        return head.next;
    }
}

003 --- 最长子串


用了一个类似dp的思想,就是从第一位开始计算最长的无重复子序列,然后第二位可以在第一位的基础上进行计算,减少计算次数。

class Solution {
    public int lengthOfLongestSubstring(String s) {
        if (s.length() <= 1) return s.length();

        Set<Character> occ = new HashSet<Character>();
        int maxLen = 0; // 最大长度
        int index = 1; // 下标
        occ.add(s.charAt(0));
        for(int i=1; i<s.length(); i++){
            while(index < s.length() && ! occ.contains(s.charAt(index))){
                occ.add(s.charAt(index++));
            }
            occ.remove(s.charAt(i-1));
            maxLen = Math.max(maxLen,index - i + 1);
            if(index == s.length()) return maxLen;
        }

        return maxLen;
    }
}

004 --- 寻找两个正序数组的中位数


用了一个类似于二分查找一样的算法,就是将完全不可能的数据给舍弃掉就行了。建立了一个递归方法,用于寻找第k位的数,然后因为两个数列是有序的,所以可以对k/2-1处的数据大小进行判断,然后如果谁更小就肯定不可能是第k位,然后可以完全舍弃掉。接着进行递归。

代码中的递归方法是我自己琢磨出来的,感觉还算优雅。

class Solution {
    public double findMedianSortedArrays(int[] nums1, int[] nums2) {
        int k = nums1.length + nums2.length;
        if(k % 2 == 0){
            return ((findKSortedArrays(nums1,nums2,0,0,k / 2 + 1) + findKSortedArrays(nums1,nums2,0,0,k / 2)) / 2);
        }else{
            return findKSortedArrays(nums1,nums2,0,0,k / 2 + 1);
        }
        
    }

    public double findKSortedArrays(int[] nums1, int[] nums2,int start1,int start2,int k) {
        if(start1 >= nums1.length) return nums2[start2 + k-1];
        if(start2 >= nums2.length) return nums1[start1 + k-1];
        if (k <= 1){
            return Math.min(nums1[start1],nums2[start2]); 
        }
        
        int index1 = Math.min(k/2 - 1,nums1.length - start1 - 1);
        int index2 = Math.min(k/2 - 1,nums2.length - start2 - 1);

        if(nums1[start1 + index1] <= nums2[start2 + index2]){
            k -= (index1 + 1);
            start1 += (index1 + 1);
        }else{
            k -= (index2 + 1);
            start2 += (index2 + 1);
        }

        return findKSortedArrays(nums1,nums2,start1,start2,k);
    }
}

005 --- 最长回文子串


典型的dp题,没啥好说的

class Solution {
    public String longestPalindrome(String s) {
        if(s.length() == 1) return s;
        int x=0,y=0;
        boolean[][] dp = new boolean[s.length()][s.length()];
        for(int i=1; i<=s.length(); i++){
            for(int j=0; j<=s.length()-i; j++){
                if(i == 1){
                    dp[j][j] = true;
                }
                else if(i == 2 && s.charAt(j) == s.charAt(j+1)){
                    dp[j][j+1] = true;
                    x = j;
                    y = j + 1;
                }
                else{
                    if(dp[j+1][j+i-2] == true && s.charAt(j) == s.charAt(j+i-1)){
                        dp[j][j+i-1] = true;
                        x = j;
                        y = j+i-1;      
                    }
                }
            }
        }
        return s.substring(x,y + 1);
    }
}

006 --- N字形变换


就是一个标记问题,例如:为4时,进行1234321,然后将同类型的放在同一个字符串中即可
代码中的flag和mark的使用是我第一次这么用。

class Solution {
    public String convert(String s, int numRows) {
        if(numRows < 2) return s;
        StringBuilder[] rows = new StringBuilder[numRows];

        int flag = 1;
        int mark = 0;
        for(int i=0; i<numRows; i++) rows[i] = new StringBuilder();
        for(int i=0; i< s.length(); i++) {
            rows[mark].append(s.charAt(i));
            mark += flag;
            if(mark == 0 || mark == numRows-1) flag = -flag;
        }

        StringBuilder result = new StringBuilder();
        for(StringBuilder row : rows){
            result.append(row);
        }

        return result.toString();
    }
}
posted @ 2023-02-07 11:30  竹林取剑  阅读(19)  评论(0编辑  收藏  举报