代码随想录刷题笔记

目录

四数相加2(Leetcode454)

赎金信(Leetcode383)

三数之和 (Leetcode15)

四数之和(Leetcode18)

反转字符串(Leetcode344)

反转字符串2(Leetcode541)

替换数字

反转字符串里面的单词 



四数相加2(Leetcode454)

public int fourSumCount(int[] nums1, int[] nums2, int[] nums3, int[] nums4) {
			int cnt = 0; //用来计数
			Map<Integer,Integer> map = new HashMap();
			/*
			 * 这一步用来存 数组1和数组2里面值的和,然后再用来和数组3和数组4和的值进行比较,如果相加为0则满足条件
			 * a+b+c+d=0  我们先将a+b的值还有出现次数存起来 
			 * 这里要注意值的合可以重复,所以不去重
			 */
			//这里是求a+b和的个数
			for(int i = 0;i<nums1.length;i++) {
				for(int j = 0;j<nums2.length;j++) {
					int temp = nums1[i]+nums2[j];//求两个数组的值
					map.put(temp, map.getOrDefault(temp, 0)+1); //存temp出现的次数,默认为0,之前存过则+1;
				}
			}
			//这里求0-(c+d)=temp,求map中是否有满足a+b=temp 有的话需要加上次数。
			for(int i = 0;i<nums3.length;i++) {
				for(int j = 0;j<nums4.length;j++) {
					int temp = nums3[i]+nums4[j]; 
					if(map.get(0-temp)!=null) { //map中找是否有和0减去数组3和数组4之后相等的值
						cnt+=map.get(0-temp);
					}
				}
			}
			return cnt;
    }

赎金信(Leetcode383)

	//1、统计magazine里面有多种字符,每种字符的个数
	//2、遍历ransomNode,若是magazine里面的值为0,则代表没有或者不够用,返回false
	public boolean canConstruct(String ransomNote, String magazine) {
		int[] arr = new int[26];
		for(int i = 0;i<magazine.length();i++) {
			arr[magazine.charAt(i)-97]++;
		}
		for(int i = 0;i<ransomNote.length();i++) {
			arr[ransomNote.charAt(i)-97]--;  //magazine里面匹配一次就减1
			if(arr[ransomNote.charAt(i)-97]<0) { //减完要判断是否小于0,小于0代表magazine里面缺少字符
				return false;
			}
		}
		return true;	
    }

三数之和 (Leetcode15)

/*
	 * 注意点:每个三元组下标不重复,但是不同元组下标可以重复,只要和为0即可
	 * 存的是不同坐标下的值,而不是坐标
	 */
//直接提交第一个函数就可以了 
public List<List<Integer>> threeSum(int[] nums) {
    	List<List<Integer>> result = new ArrayList<>();
    	//quickSort(nums,0,nums.length-1);
    	Arrays.sort(nums); //先排序,方便指针还有减枝操作
    	for(int i = 0;i<nums.length;i++) {  //遍历每个i
    		if(nums[i]>0) {  //如果nums[i]>0,因为是从小到大排,如果最小的数都大于0,后面的数相加也必定大于0
    			return result;
    		}
    		if(i>0&&nums[i]==nums[i-1]) {  //去重,这里如果是用nums[i]==nums[i+1]的话会去掉[-1,-1,2]的情况,i>0防止为空
    			continue;
    		}
    		int left = i+1;  
    		int right = nums.length-1;
    		while(right>left) { //如果right>=left  相当于出现存在一个坐标对应的值被取了2次的情况
    			if(nums[i]+nums[left]+nums[right]>0) {   //>0证明和太大,所以right--,让和变小
    				right--;
    			}else if(nums[i]+nums[left]+nums[right]<0) {//<0证明和太小,所以left++,让和变大
    				left++;
    			}else {
    				result.add(Arrays.asList(nums[i],nums[left],nums[right])); //Arrays.asList:将多个数组成数组
    				while(right>left&&nums[right]==nums[right-1]) right--;  //还可以继续寻找,去除重复值
                    while(right>left&&nums[left]==nums[left+1]) left++;
    				left++;
    				right--;
    			}
    		}
    	}
    	return result;
    }
    //快排
    public static void quickSort(int[] nums,int l,int r) {
    	while(l<r) {
    		int p1 = l;
    		int p2 = r;
    		int temp = nums[l];
    		while(p1<p2) {
    			if(nums[p2]<temp) { //p2找比temp小的 p2先找
    				if(nums[p1]>temp) { //p1找比temp大的
    					int flag = nums[p2];
    					nums[p2] = nums[p1];
    					nums[p1] = flag;
    				}else {
    					p1++;
    				}
    			}else {
    				p2--;
    			}
    		}
    		nums[l] = nums[p1];
    		nums[p1] = temp;
    		quickSort(nums,l,p1);  //分段再排
    		quickSort(nums,p1+1,r); //分段再排
    	}
    }

四数之和(Leetcode18)

public static List<List<Integer>> fourSum(int[] nums, int target) {
    	List<List<Integer>> result = new ArrayList<>();
    	Arrays.sort(nums);
    	for(int a = 0;a<nums.length;a++) {
    		if(nums[a]>0&&nums[a]>target) {  //这里要注意大于0
    			return result;
    		}
    		if(a>0&&nums[a]==nums[a-1]) {  //与三数之和一样去重
    			continue;
    		}
    		for(int b = a+1;b<nums.length;b++) {
    			if(nums[a]+nums[b]>=0&&nums[a]+nums[b]>target){//这里要break不能return 例如[-3,-2,-1,0,0,1,2,3] target=0 当a指向-1,b指向3时return的话会缺少-1,0,0,1
                    break;									
                }
    			 if(b>a+1&&nums[b]==nums[b-1]){ //与三数之和一样去重
                     continue;
                 }
    			int c = b+1;
    			int d = nums.length-1;
    			while(c<d) {
    				if(nums[a]+nums[b]+nums[c]+nums[d]<target) {
    					c++;
    				}else if(nums[a]+nums[b]+nums[c]+nums[d]>target) {
    					d--;
    				}else {
    					result.add(Arrays.asList(nums[a],nums[b],nums[c],nums[d]));
    					while(c<d&&nums[c]==nums[c+1]) c++;
    					while(c<d&&nums[d]==nums[d-1]) d--;
    					c++;
    					d--;
    				}
    			}
    		}
    	}
    	return result;
    }

反转字符串(Leetcode344)

	//值得注意的是原地修改数组,不能多开数组
	public void reverseString(char[] s) {
		int l = 0;
		int r = s.length-1;
		while(l<r) {
			char temp = s[l];
			s[l] = s[r];
			s[r] = temp;
			l++;
			r--;
		}
	}

反转字符串2(Leetcode541)

//条件一  每计数2k则反转前k个  
	//条件二  剩余少于k则全部反转
	//条件三  小于2k但是大于等于k,反转前k个
	//其实条件1和3可以合成一个条件,每次反转k个,
	public String reverseStr(String s, int k) {
		char[] result = s.toCharArray();
		int i = 0;
		while(i<result.length) {
			int end = Math.min(result.length-1, i+k-1); //其实每次就去判断在一个区间内是否可以反转k个,若是没办法的话则全部反转,也这时也就是在末尾了
			swapStr(result,i,end);
			i += 2*k;
		}
		return new String(result);
    }
	//直接在数组上交换
	public  void swapStr(char[] arr,int start,int end) {
		while(start<end) {
			char temp = arr[start];
			arr[start] = arr[end];
			arr[end] = temp;
			start++;
			end--;
		}
	}}

替换数字

public static void main(String[] args) {
		Scanner scanner = new Scanner(System.in);
		String str = scanner.next();
		System.out.print(swap(str));
		scanner.close();
	}
	//1、先统计数字个数
	//2、创建新数组
	//3、将老数组赋值给新数组
	//4、开始赋值
	public static String swap(String str) {
		int Osize = str.length();
		int count=0;
		for(int i = 0;i<Osize;i++) {
			if(Character.isDigit(str.charAt(i))) {
				count++;
			}
		}
		char[] newS = new char[Osize+count*5];
		System.arraycopy(str.toCharArray(), 0, newS, 0, Osize);
		int Nsize = newS.length;
		for(int i = Osize-1,j=Nsize-1;i<j;i--,j--) {
			if(!Character.isDigit(newS[i])) {
				newS[j] = newS[i];
			}else {
				newS[j] = 'r';
                newS[j - 1] = 'e';
                newS[j - 2] = 'b';
                newS[j - 3] = 'm';
                newS[j - 4] = 'u';
                newS[j - 5] = 'n';
                j -= 5;
			}
		}
		return new String(newS);
	}

反转字符串里面的单词 (Leetcode151)

package 字符串;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;

public class 反转字符串里面的单词 {

	public static void main(String[] args) {
		
		System.out.print(reverseWords("  Bob    Loves  Alice   "));
	}
	//1、去掉多余空格
	//2、整体反转
	//3、单个单词反转
	public static String reverseWords(String s) {
		s = s.trim();//去掉前后空格
		char[] chars = s.toCharArray();
		System.out.println(Arrays.toString(chars));
		int fast = 0;
		int slow = 0;
		//去掉中间出现空格的情况
		while(fast!=chars.length) {
			if(chars[fast]!=' ') {
				chars[slow] = chars[fast];
				fast++;
				slow++;
			}else {
				while(chars[fast]==' ') {
					fast++;
				} 
				chars[slow]=' ';  //注意   不然  Bob    Loves  Alice   过不去
				slow++;
			}
		}
		System.out.println(Arrays.toString(chars));
		//复制到新数组里面
		char[] result = new char[slow];
		System.arraycopy(chars, 0, result, 0, slow);
		reverse(result,0,result.length-1);
		System.out.println(Arrays.toString(result));
		//反转单词
		for(int i = 0,j=0;i<result.length;i++) {
			if(result[i]==' ') {
				reverse(result,j,i-1);
				j = i+1;
			}else if(i==result.length-1) {
				reverse(result,j,i);
			}
		}
		return new String(result);
		
    }
	//实现反转功能
	public static void reverse(char[] str,int l,int r) {
		while(l<r) {
			char temp = str[l];
			str[l] = str[r];
			str[r] = temp;
			l++;
			r--;
		}
	}
	

}

右旋字符串 

	//方法一直接截取就行,下面是方法一
	//方法二 abcdefg 2
	//1、 gfedcba 2、fgedcba 3、fgabced
	public static String return_(int d,String str) {
		String temp = str.substring(str.length()-d, str.length());
		return temp+str.substring(0,str.length()-d);
		
	}

找出字符串中第一个匹配项的下标(Leetcode28)

//转数组
	//遍历haystack
	//判断 如果存在和needle第一个开头元素,进入判断
	//如果遍历完haystack都没有,则返回负一
	public static int strStr(String haystack, String needle) {
		//needle长度大于haystack肯定不匹配
		if(needle.length()>haystack.length()) {
			return -1;
		}
		int k = -1;
		int flag = -1;
		char[] first = haystack.toCharArray();
		char[] second = needle.toCharArray();
		for(int i = 0;i<first.length;i++) {
			if(first[i]==second[0]) {
				flag = i; //记录i值
				k = i; 
				//防止越界 当前+needle的长度超过了haystack的长度则代表无法匹配
				if(i+needle.length()>first.length) {
					flag=-1;
					break;
				}
				for(int j=0;j<second.length;j++) {
					if(first[k]==second[j]) {
						k++;
					}else {
						flag=-1;
						break;
					}
				}
			}
			if(flag!=-1) {
				break;
			}
		}
		return flag;
    }

重复的子字符串(Leetcode459)

	public static boolean repeatedSubstringPattern(String s) {
		for(int i = 1;i<s.length();i++) {
			int flag = -1;
			String temp = s.substring(0, i);
			for(int j = 0;j<s.length();j+=temp.length()) {
				if((j+temp.length())>s.length()) {  //越界则不满足 abcv的情况
					flag = 0;
					break;
				}
				if(!s.substring(j, j+temp.length()).equals(temp)) { //注意substring中(j,j+temp.length) 不是(j,temp.length) 报错了后面发现是这里的问题细节还是很重要的
					flag = 0;
					break;
				}
			}
			if(flag==-1) {
				return true;
			}
		
		}
		return false;
    }
//公式  len%(len-(next[len-1]+1))==0代表有重复子串
 public boolean repeatedSubstringPattern(String s) {
        int[] next = new int[s.length()];
        int j = -1;
        next[0] = j;
        //求最长相等前后缀
        for(int i = 1;i<s.length();i++){
            while(j>=0&&s.charAt(i)!=s.charAt(j+1)){
                j = next[j];
            }
            if(s.charAt(i)==s.charAt(j+1)){
                j++;
            }
            next[i] = j;
        }
        if(next[s.length()-1]!=-1&&(s.length()%(s.length()-(next[s.length()-1]+1)))==0){
            return true;
        }
        return false;
    }

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值