Leetcode(easy Double pointer)

Leetcode(easy Double pointer)

Leetcode 双指针简单题目

26 删除排序数组中的重复项

class Solution{
	public int removeDuplicates(int[] nums){
		// step代表慢指针
		int step = 0;
		// 这里面的i就相当于快指针
		for(int i = 1;i<nums.length;i++){
			if(nums[i] != nums[step]){
				// 满指针后移 存放i指向的不同的元素
				step++;
				nums[step] = nums[i];
			}
		}
		return step+1;
	}
}

27 移除元素

class Solution{
	public int removeElement(int[] nums,int val){
		int step = 0;
		for(int i = 0;i<nums.length;i++){
			if(nums[i]!=val){
				nums[step] = nums[i];
				step++;
			}
		}
		return step;
	}
}

28 实现strStr()

实现 strStr() 函数。给定一个 haystack 字符串和一个 needle 字符串,在 haystack 字符串中找出 needle 字符串出现的第一个位置 (从0开始)。如果不存在,则返回 -1。

class Solution {
  public int strStr(String haystack, String needle) {
    int L = needle.length(), n = haystack.length();
    if (L == 0) return 0;

    int pn = 0;
    while (pn < n - L + 1) {
      while (pn < n - L + 1 && haystack.charAt(pn) != needle.charAt(0)) ++pn;
      int currLen = 0, pL = 0;
      while (pL < L && pn < n && haystack.charAt(pn) == needle.charAt(pL)) {
        ++pn;
        ++pL;
        ++currLen;
      }

      if (currLen == L) return pn - L;
      pn = pn - currLen + 1;
    }
    return -1;
  }
}

88 合并两个有序数组

给你两个有序整数数组 nums1 和 nums2,请你将 nums2 合并到 nums1 中,使 nums1 成为一个有序数组。

class Solution {
    public void merge(int[] nums1, int m, int[] nums2, int n) {
        int i = m-1;
        int j = n-1;
        int index = m+n-1;
        while(i>-1&&j>-1){
            if(nums1[i]<nums2[j]){
                nums1[index] = nums2[j];
                j--;
                index--;
            }else{
                nums1[index] = nums1[i];
                i--;
                index--;
            }
        }
        while(i>-1){
            nums1[index] = nums1[i];
            i--;
            index--;
        }
        while(j>-1){
            nums1[index] = nums2[j];
            j--;
            index--;
        }

    }
}

125 验证回文串

给定一个字符串,验证它是否是回文串,只考虑字母和数字字符,可以忽略字母的大小写。

class Solution {
    public boolean isPalindrome(String s) {
        int n = s.length();
        int left = 0, right = n - 1;
        while (left < right) {
            while (left < right && !Character.isLetterOrDigit(s.charAt(left))) {
                ++left;
            }
            while (left < right && !Character.isLetterOrDigit(s.charAt(right))) {
                --right;
            }
            if (left < right) {
                if (Character.toLowerCase(s.charAt(left)) != Character.toLowerCase(s.charAt(right))) {
                    return false;
                }
                ++left;
                --right;
            }
        }
        return true;
    }
}

141 环形链表

给定一个链表,判断链表中是否有环。

/**
 * Definition for singly-linked list.
 * class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode(int x) {
 *         val = x;
 *         next = null;
 *     }
 * }
 */
public class Solution {
    public boolean hasCycle(ListNode head) {
        if(head == null || head.next == null) return false;
        ListNode fast = head;
        ListNode slow = head;
        while(fast!=null && fast.next !=null && slow!=null){
            // 一定要把if(fast == slow) return 放在移动过指针的下面,不然的话初始状态肯定是相等的
            slow = slow.next;
            fast = fast.next.next;
            if(fast == slow) return true;
        }
        return false;

    }
}

167. 两数之和 II - 输入有序数组

给定一个已按照升序排列 的有序数组,找到两个数使得它们相加之和等于目标数。函数应该返回这两个下标值 index1 和 index2,其中 index1 必须小于 index2。

class Solution {
    public int[] twoSum(int[] numbers, int target) {
        // 设置两个指针
        int i = 0;
        int j = numbers.length - 1;
        boolean flag = true;
        while(flag){
            if(numbers[i] + numbers[j] > target) j--;
            else if(numbers[i] + numbers[j] < target) i++;
            else return new int[]{i+1,j+1};
        }
        return new int[2];
    }
}

234 回文链表

请判断一个链表是否为回文链表。

/**
 * Definition for singly-linked list.
 * public class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode(int x) { val = x; }
 * }
 */
class Solution {
    public boolean isPalindrome(ListNode head) {
        if(head == null || head.next == null) return true;
        ListNode pre = head;
        ListNode cur = head;
        int len = 0;
        while(pre != null){
            pre=pre.next;
            len++;
        }
        pre = head;
        int mid = (len%2)==1?(len/2+2):(len/2+1);
        while(mid-- >1) cur = cur.next;
        cur = reverse(cur);
        while(cur != null && pre != null){
            if(pre.val != cur.val) return false;
            pre = pre.next;
            cur = cur.next;
        }
        return true;

    }
    // 将链表翻转,并返回翻转之后的头结点
    public ListNode reverse(ListNode root){
        if(root == null || root.next == null) return root;
        ListNode dummy = new ListNode(0);
        dummy.next = null;
        ListNode pre = root;
        ListNode cur = root.next;
        while(pre != null){
            pre.next = dummy.next;
            dummy.next = pre;
            pre = cur;
            if(cur!=null) cur = cur.next;
        }

        return dummy.next;
    }
}

283. 移动零

给定一个数组 nums,编写一个函数将所有 0 移动到数组的末尾,同时保持非零元素的相对顺序。

class Solution {
    public void moveZeroes(int[] nums) {
        /**
        // 这种解法会导致非零元素的相对位置的改变
        int i = 0;
        int j = nums.length-1;
        while(i<j){
            // 正向找到0所在的下标
            while(i<j && nums[i]!=0) i++;
            // 反向找到不为0的数字所在的下标
            while(i<j && nums[j]==0) j--;
            int tmp = nums[i];
            nums[i] = nums[j];
            nums[j] = tmp;
            i++;
            j--;
        }
        */
		if(nums==null) {
			return;
		}
		//第一次遍历的时候,j指针记录非0的个数,只要是非0的统统都赋给nums[j]
		int j = 0;
		for(int i=0;i<nums.length;++i) {
			if(nums[i]!=0) {
				nums[j++] = nums[i];
			}
		}
		//非0元素统计完了,剩下的都是0了
		//所以第二次遍历把末尾的元素都赋为0即可
		for(int i=j;i<nums.length;++i) {
			nums[i] = 0;
		}
	}
}

344. 反转字符串

编写一个函数,其作用是将输入的字符串反转过来。输入字符串以字符数组 char[] 的形式给出。不要给另外的数组分配额外的空间,你必须原地修改输入数组、使用 O(1) 的额外空间解决这一问题。
你可以假设数组中的所有字符都是 ASCII 码表中的可打印字符。

class Solution {
    public void reverseString(char[] s) {
        // 设置两个指针
        int i = 0;
        int j = s.length-1;
        while(i<j){
            char tmp = s[i];
            s[i] = s[j];
            s[j] = tmp;
            i++;
            j--;
        }
    }
}

345. 反转字符串中的元音字母

编写一个函数,以字符串作为输入,反转该字符串中的元音字母

class Solution {
    public String reverseVowels(String s) {
        Set<Character> yuan = new HashSet<>();
        String str = "aeiouAEIOU";
        for(Character c:str.toCharArray()) yuan.add(c);
        // 设置两个指针,left和right,其中left指针从左向右查找元音字符,right从右向左查找元音字符,
        int left = 0;
        int right = s.length()-1;
        // 因为java无法对字符串进行直接的操作,这里我们将字符串s变为字符数组
        char[] s2c = s.toCharArray();
        // 遍历s2c
        while(left < right){
            while(left < right && !yuan.contains(s2c[left])) left++;
            while(left < right && !yuan.contains(s2c[right])) right--;
            char temp = s2c[left];
            s2c[left] = s2c[right];
            s2c[right] = temp;
            // 交换完毕之后,让指针移动
            left++;
            right--;
        }
        return String.valueOf(s2c);
    }
}

349 两个数组的交集

题目:给定两个数组,编写一个函数来计算它们的交集。 不允许重复的元素出现在最终结果中。

class Solution {
    public int[] intersection(int[] nums1, int[] nums2) {
        Set<Integer> union = new HashSet<>();
        Arrays.sort(nums1);
        Arrays.sort(nums2);
        // 设置两个指针 i和j 其中i指向nums1中的元素,j指向nums2中的元素
        int i = 0;
        int j = 0;
        //只要两个nums还没有被遍历完,就继续便利
        while(i < nums1.length && j < nums2.length){
            // 如果指向的元素相同,则将元素添加到union中
            if(nums1[i] == nums2[j]){
                union.add(nums1[i]);
                i++;
                j++;
            } 
            // 如果指向的元素不相同,则移动指针
            else{
                // 这时候移动指针也分为两种情况,第一种是nums1[i] > nums2[j]
                if(nums1[i] > nums2[j]) j++;
                else i++;
            }
        }
        int[] res = new int[union.size()];
        int index = 0;
        for(int k:union) res[index++] = k; 
        return res;
    }
}

350 两个数组的交集2

题目:给定两个数组,编写一个函数来计算它们的交集。 允许重复的元素出现在最终结果中。

class Solution {
    public int[] intersect(int[] nums1, int[] nums2) {
        List<Integer> union = new ArrayList<>();
        Arrays.sort(nums1);
        Arrays.sort(nums2);
        // 设置两个指针 i和j 其中i指向nums1中的元素,j指向nums2中的元素
        int i = 0;
        int j = 0;
        //只要两个nums还没有被遍历完,就继续便利
        while(i < nums1.length && j < nums2.length){
            // 如果指向的元素相同,则将元素添加到union中
            if(nums1[i] == nums2[j]){
                union.add(nums1[i]);
                i++;
                j++;
            } 
            // 如果指向的元素不相同,则移动指针
            else{
                // 这时候移动指针也分为两种情况,第一种是nums1[i] > nums2[j]
                if(nums1[i] > nums2[j]) j++;
                else i++;
            }
        }
        int[] res = new int[union.size()];
        for(int k = 0;k<union.size();k++) res[k] = union.get(k); 
        return res;

    }
}

844 比较含退格的字符串

题目:给定 S 和 T 两个字符串,当它们分别被输入到空白的文本编辑器后,判断二者是否相等,并返回结果。 # 代表退格字符。

// 虽然是可以用双指针来写这道题,不过相对而言,使用额外的数据结构更容易理解以及更容易实现。使用stack
class Solution {
    public boolean backspaceCompare(String S, String T) {
        return toString(S).equals(toString(T));
    }
    public String toString(String str){
        Stack<Character> stack = new Stack<>();
        for(Character c:str.toCharArray()){
            if(c=='#' && !stack.isEmpty()) stack.pop();
            else if(c=='#') continue;
            else{
                stack.push(c);
            }
        }
        String res = "";
        while(!stack.isEmpty()) res+=stack.pop();
        return res;
    }
}

925 长键按入

题目:你的朋友正在使用键盘输入他的名字 name。偶尔,在键入字符 c 时,按键可能会被长按,而字符可能被输入 1 次或多次。你将会检查键盘输入的字符 typed。如果它对应的可能是你的朋友的名字(其中一些字符可能被长按),那么就返回 True。

class Solution {
    public boolean isLongPressedName(String name, String typed) {
        // 设置两个指针,i,j 其中i指针指向的是name中的字符,j指针指向的是typed中的字符
        int i = 0;
        int j = 0;
        // 只要typed还有字符,就一直向后遍历
        while(j<typed.length()){
            // 如果i指针指向的name中的字符与j指针指向的typed中字符相同的话,两个指针均向后移动
            if(i<name.length()&&name.charAt(i) == typed.charAt(j)){
                i++;
                j++;
            }else if(j>0 && typed.charAt(j) == typed.charAt(j-1)){
                //如果i指针指向的name中的字符与j指针指向的typed中的字符不相等并且typed中的j指向的字符与上一个j-1指向的字符相同,
                j++;
            }else{
                return false;
            }
        }
        return i == name.length();
    }
}

977. 有序数组的平方

给定一个按非递减顺序排序的整数数组 A,返回每个数字的平方组成的新数组,要求也按非递减顺序排序。

class Solution {
    public int[] sortedSquares(int[] A) {
        // 循环便利,将A中的元素替换成为原是数子的平方
        for(int i =0;i<A.length;i++) A[i] = A[i]*A[i];
        quickSort(A,0,A.length-1);
        return A;
    }
    // 手动实现快速排序
    public void quickSort(int[] nums,int l,int r){
        if(l<r){
            int i = l;
            // 选取基准值
            int pivot = nums[l];
            int j = r;
            while(i<j){
                // 从右到左找到比pivot小的值
                while(i<j && nums[j]>=pivot) j--;
                // 找到了比pivot小的值,按照快速排序的思想,该值应该出现在基准值的左边,所以交换位置
                if(i<j) nums[i] =  nums[j];
                // 从左到右找到比pivot大的值
                while(i<j && nums[i]<=pivot) i++;
                if(i<j) nums[j] = nums[i];
            }
            //交换完之后,将pivot放到他本来应该在的位置
            nums[i] = pivot;
            // 然后递归处理基准值左边的以及基准值右边的子数列
            quickSort(nums,l,i-1);
            quickSort(nums,i+1,r);
        }
    }
}

剑指 Offer 04. 二维数组中的查找

在一个 n * m 的二维数组中,每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序。请完成一个函数,输入这样的一个二维数组和一个整数,判断数组中是否含有该整数。

class Solution {
    public boolean findNumberIn2DArray(int[][] matrix, int target) {
        // 从数组的右上角开始查找,如果当前元素比target小就向下移动,如果当前元素比target大就向左移动
        if(matrix.length == 0 || matrix == null) return false;
        // 获取matrix的行数和列数
        int m = matrix.length;
        int n = matrix[0].length;
        // 设计两个指针,i和j 其中i指向当前所在的行数,j指向当前所在的列数,初始化i=0,j=n-1
        int i = 0;
        int j = n-1;
        while(j>-1 && i<m){
            if(matrix[i][j] > target) j--;
            else if(matrix[i][j] < target) i++;
            else return true;
        }
        return false;
    }
}

剑指 Offer 22. 链表中倒数第k个节点

/**
 * Definition for singly-linked list.
 * public class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode(int x) { val = x; }
 * }
 */
class Solution {
    public int kthToLast(ListNode head, int k) {
        // 因为题目中说了k保证是有效的,所以不用先计算整个链表的长度
        // 定义两个指针,一个快指针,一个慢指针,快指针在满指针后k个,当快指针为空的时候,满指针就到了目的元素的位置
        ListNode slow = head;
        ListNode fast = head;
        while(k-->0) fast = fast.next;
        while(fast!=null){
            slow = slow.next;
            fast = fast.next;
        }
        return slow.val;
    }
}

面试题 02.02. 返回倒数第 k 个节点

实现一种算法,找出单向链表中倒数第 k 个节点。返回该节点的值。

/**
 * Definition for singly-linked list.
 * public class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode(int x) { val = x; }
 * }
 */
class Solution {
    public int kthToLast(ListNode head, int k) {
        // 因为题目中说了k保证是有效的,所以不用先计算整个链表的长度
        // 定义两个指针,一个快指针,一个慢指针,快指针在满指针后k个,当快指针为空的时候,满指针就到了目的元素的位置
        ListNode slow = head;
        ListNode fast = head;
        while(k-->0) fast = fast.next;
        while(fast!=null){
            slow = slow.next;
            fast = fast.next;
        }
        return slow.val;
    }
}

面试题 10.01. 合并排序的数组

给定两个排序后的数组 A 和 B,其中 A 的末端有足够的缓冲空间容纳 B。 编写一个方法,将 B 合并入 A 并排序。初始化 A 和 B 的元素数量分别为 m 和 n。

class Solution {
    public void merge(int[] A, int m, int[] B, int n) {
        // 一看就是倒着来嘛
        int aLen = A.length-1;
        // 设置两个指针分别指向A,B中的元素,i指向A,j指向B
        int i = m-1;
        int j = n-1;
        while(i>-1 && j>-1){
            if(A[i] > B[j]){
                A[aLen] = A[i];
                aLen--;
                i--;
            }else{
                A[aLen] = B[j];
                aLen--;
                j--;
            }
        }
        while(i>-1){
            A[aLen] = A[i];
            aLen--;
            i--;
        }
        while(j>-1){
            A[aLen] = B[j];
            aLen--;
            j--;
        }

    }
}
posted @ 2020-11-05 18:27  BOTAK  阅读(94)  评论(0编辑  收藏  举报