LeetCode 1423.Maximum Points You Can Obtain from Cards(可获得的最大点数) 数学,滑动窗口/Medium


1.Description

几张卡牌 排成一行,每张卡牌都有一个对应的点数。点数由整数数组 cardPoints 给出。

每次行动,你可以从行的开头或者末尾拿一张卡牌,最终你必须正好拿 k 张卡牌。

你的点数就是你拿到手中的所有卡牌的点数之和。

给你一个整数数组 cardPoints 和整数 k,请你返回可以获得的最大点数。


2.Example

输入:cardPoints = [1,2,3,4,5,6,1], k = 3
输出:12
解释:第一次行动,不管拿哪张牌,你的点数总是 1 。但是,先拿最右边的卡牌将会最大化你的可获得点数。最优策略是拿右边的三张牌,最终点数为 1 + 6 + 5 = 12 

3.Solution

主要有两种解法:
1.遍历从左边和从右边一共取k个的所有情况,找出最大的
2.使用滑动窗口,找length-k个连续的数的最小的和:
在这里插入图片描述

1.

//先从左边取k个,然后左边一个一个少,右边一个一个多
class Solution {
    
    
        public static int maxScore(int[] cardPoints, int k) {
    
    
    	int sum = 0;
    	for(int i=0;i<k;i++) {
    
    
    		sum += cardPoints[i];
    	}
    	
    	int copy = sum;
    	for(int i=0;i<k;i++) {
    
    
    		copy -= cardPoints[k-1-i];
    		copy += cardPoints[cardPoints.length-1-i];
    		sum = Math.max(sum, copy);
    	}	
    	return sum;
    }
}

我一开始写的:把k均匀分配到左右两边,然后遍历左边和右边的(麻烦了)

class Solution {
    
    
    public static int maxScore(int[] cardPoints, int k) {
    
    
    	int sum = 0;
    	int oringin;
    	int left = k/2-1;
    	int right = cardPoints.length - k/2;
    	
    	//获得初始的sum
    	for(int i=0;i<k/2;i++) {
    
    
    		sum += cardPoints[i];
    		sum += cardPoints[cardPoints.length-1-i];
    	}
    	if(k%2==1) {
    
    
    		if(cardPoints[left+1]>=cardPoints[right-1]) {
    
    
    			left++;
    			sum += cardPoints[left];
    		}else {
    
    
    			right--;
    			sum += cardPoints[right];
    		}
    	}
    	
    	int copy = sum;
    	oringin = sum;
    	int leftcopy =left;//一开始分配的left和right还有用,不动,定义个copy
    	int rightcopy = right;
        
        //把左边的一个个到右边
        while (leftcopy>=0) {
    
    
			copy -= cardPoints[leftcopy];
			leftcopy--;
			copy += cardPoints[rightcopy-1];
			rightcopy--;
			if(copy>sum) {
    
    
				sum = copy;
			}
		}
        copy = oringin;
        leftcopy = left;
        rightcopy = right;
        
        //右边一个个到左边
        while (rightcopy<=cardPoints.length-1) {
    
    
			copy -= cardPoints[rightcopy];
			rightcopy++;
			copy += cardPoints[leftcopy+1];
			leftcopy++;
			if(copy>sum) {
    
    
				sum = copy;
			}
		}
    	return sum;
    }
}

2.


class Solution {
    
    
    public int maxScore(int[] cardPoints, int k) {
    
    
        int n = cardPoints.length;
        // 滑动窗口大小为 n-k
        int windowSize = n - k;
        // 选前 n-k 个作为初始值
        int sum = 0;
        for (int i = 0; i < windowSize; ++i) {
    
    
            sum += cardPoints[i];
        }
        int minSum = sum;
        for (int i = windowSize; i < n; ++i) {
    
    
            // 滑动窗口每向右移动一格,增加从右侧进入窗口的元素值,并减少从左侧离开窗口的元素值
            sum += cardPoints[i] - cardPoints[i - windowSize];
            minSum = Math.min(minSum, sum);
        }
        return Arrays.stream(cardPoints).sum() - minSum;
    }
}

猜你喜欢

转载自blog.csdn.net/weixin_45736160/article/details/113731643