You are given an array of integers stones where stones[i] is the weight of the ith stone.
We are playing a game with the stones. On each turn, we choose any two stones and smash them together. Suppose the stones have weights x and y with x <= y. The result of this smash is:
x == y, both stones are destroyed, andx != y, the stone of weight x is destroyed, and the stone of weight y has new weight y - x.At the end of the game, there is at most one stone left.
Return the smallest possible weight of the left stone. If there are no stones left, return 0.
Example 1:
Input: stones = [2,4,1,5,6,3]
Output: 1Explanation:
Example 2:
Input: stones = [4,4,1,7,10]
Output: 2Constraints:
1 <= stones.length <= 301 <= stones[i] <= 100class Solution:
def lastStoneWeightII(self, stones: List[int]) -> int:
stoneSum = sum(stones)
target = (stoneSum + 1) // 2
def dfs(i, total):
if total >= target or i == len(stones):
return abs(total - (stoneSum - total))
return min(dfs(i + 1, total), dfs(i + 1, total + stones[i]))
return dfs(0, 0)Where is the number of stones and is the sum of the weights of the stones.
class Solution:
def lastStoneWeightII(self, stones: List[int]) -> int:
stoneSum = sum(stones)
target = (stoneSum + 1) // 2
dp = {}
def dfs(i, total):
if total >= target or i == len(stones):
return abs(total - (stoneSum - total))
if (i, total) in dp:
return dp[(i, total)]
dp[(i, total)] = min(dfs(i + 1, total), dfs(i + 1, total + stones[i]))
return dp[(i, total)]
return dfs(0, 0)Where is the number of stones and is the sum of the weights of the stones.
class Solution:
def lastStoneWeightII(self, stones: List[int]) -> int:
stoneSum = sum(stones)
target = stoneSum // 2
n = len(stones)
dp = [[0] * (target + 1) for _ in range(n + 1)]
for i in range(1, n + 1):
for t in range(target + 1):
if t >= stones[i - 1]:
dp[i][t] = max(dp[i - 1][t], dp[i - 1][t - stones[i - 1]] + stones[i - 1])
else:
dp[i][t] = dp[i - 1][t]
return stoneSum - 2 * dp[n][target]Where is the number of stones and is the sum of the weights of the stones.
class Solution:
def lastStoneWeightII(self, stones: List[int]) -> int:
stoneSum = sum(stones)
target = stoneSum // 2
dp = [0] * (target + 1)
for stone in stones:
for t in range(target, stone - 1, -1):
dp[t] = max(dp[t], dp[t - stone] + stone)
return stoneSum - 2 * dp[target]Where is the number of stones and is the sum of the weights of the stones.
class Solution:
def lastStoneWeightII(self, stones: List[int]) -> int:
stoneSum = sum(stones)
target = stoneSum // 2
dp = {0}
for stone in stones:
new_dp = set(dp)
for val in dp:
if val + stone == target:
return stoneSum - 2 * target
if val + stone < target:
new_dp.add(val + stone)
dp = new_dp
return stoneSum - 2 * max(dp)Where is the number of stones and is the sum of the weights of the stones.
class Solution:
def lastStoneWeightII(self, stones: List[int]) -> int:
stoneSum = sum(stones)
target = stoneSum // 2
dp = 1
for stone in stones:
dp |= dp << stone
for t in range(target, -1, -1):
if dp & (1 << t):
return stoneSum - 2 * tWhere is the number of stones and is the sum of the weights of the stones.