Skip to content

C++按tag刷leetcode,主要是算法思想的分类解答。原仓库@CyC2018@h2pl,并增加了若干题目

Notifications You must be signed in to change notification settings

Jing-Xiaolong/leetcode-tag

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

37 Commits
 
 
 
 
 
 

Repository files navigation

Leetcode算法思想分类解答

目录



算法思想

排序

七大基于比较的排序算法

冒泡排序

// O(n^2),稳定,从后往前遍历时发现逆序即立刻交换
void bubbleSort(vector<int> &ivec){
    int len = ivec.size();
    for(int i = 0; i < len; ++i)
        for(int j = 0; j < len - i - 1; ++j)
            if(ivec[j] > ivec[j + 1])	
                swap(ivec[j], ivec[j + 1]);
}

选择排序

// O(n^2),不稳定,第i次遍历时,选择下标[i, n)中数值最小的数放在i处
void selectionSort(vector<int> &ivec){
    int len = ivec.size();
    for(int i = 0; i < len; ++i){
        int minIdx = i;
        for(int j = i + 1; j < len; ++j)
            if(ivec[j] < ivec[i])	
                minIdx = j;
        swap(ivec[i], ivec[minIdx]);
    }
}

插入排序

// O(n^2),稳定,遍历第i个数时,一直将它往前交换直到不再比前一个数更大
void insertionSort(vector<int> &ivec){
    for(int i = 1; i < ivec.size(); ++i){
	int j = i - 1;
	while(j >= 0 && ivec[j + 1] < ivec[j])
	    swap(ivec[j + 1], ivec[j]),	--j;
    }
}

归并排序

// O(nlogn),稳定,不断分成更小的数组进行归并
void merge(vector<int> &ivec, int low, int mid, int high){
    vector<int> helper;
    int L = low, R = mid + 1;
    while(L <= mid && R <= high)
        ivec[L] < ivec[R] ? helper.push_back(ivec[L++]) : helper.push_back(ivec[R++]);
    while(L <= mid)
        helper.push_back(ivec[L++]);
    while(R <= high)
        helper.push_back(ivec[R++]);
    for(int i = low; i <= high; ++i)
        ivec[i] = helper[i - low];
}
void mergeSort(vector<int> &ivec, int low, int high){
    if(low >= high)
        return;
    int mid = (high - low) / 2 + low;
    mergeSort(ivec, low, mid);
    mergeSort(ivec, mid + 1, high);
    merge(ivec, low, mid, high);
}
// 排序入口
void mergeSort(vector<int> &ivec){
    mergeSort(ivec, 0, ivec.size() - 1);
}

快速排序

//O(nlogn),不稳定,找到第i个数,使[0,i)中均比它小且(i,n)中均比它大,再对[0,i)和(i,n)进行快排
void quickSort(vector<int> &ivec, int low, int high){
    if(low >= high)
	return;
    // 找到pivot_value满足上述条件
    int pivot_value = ivec[low];
    int pivot = low;
    for(int i = low + 1; i <= high; ++i)
	if(ivec[i] < pivot_value){
            ++pivot;
            swap(ivec[i], ivec[pivot]);
	}
    swap(ivec[low], ivec[pivot]);
    // 对[i,pivot),(pivot, n)继续快排
    quickSort(ivec, low, pivot - 1);
    quickSort(ivec, pivot + 1, high);
}
// 排序入口
void quickSort(vector<int> &ivec){
    quickSort(ivec, 0, ivec.size() - 1);
}

堆排序

// 自己网上找资料看吧
void heapDown(vector<int> &ivec, int beg, int end){	// 调整 ivec[beg...end] 使之满足大顶堆要求
    for(int cur = beg, child = beg * 2 + 1; child <= end; cur = child, child = child * 2 + 1){
        if(child < end && ivec[child] < ivec[child + 1])
            ++child;
        if(ivec[cur] > ivec[child])	// 调整父节点和子节点的关系
            break;
        else
            swap(ivec[cur], ivec[child]);
    }
}
// 排序入口
void heapSort(vector<int> &ivec){
    // 构造大顶堆
    for(int i = ivec.size() / 2 - 1; i >= 0; --i)
        heapDown(ivec, i, ivec.size() - 1);
    // 交换堆顶元素,并再次调整为大顶堆
    for(int i = ivec.size() - 1; i > 0; --i){
        swap(ivec[0], ivec[i]);
        heapDown(ivec, 0, i - 1);
    }
}

利用排序思想的算法

leetcode.215 数组中的第K个最大元素 medium(或前K个最大的元素)

在未排序的数组中找到第 k 个最大的元素。请注意,你需要找的是数组排序后的第 k 个最大的元素,而不是第 k 个不同的元素。

示例:

输入: [3,2,1,5,6,4] 和 k = 2
输出: 5

输入: [3,2,3,1,2,4,5,5,6] 和 k = 4
输出: 4
// 思路1:利用堆排序,第K次弹出大顶堆的堆顶即为第K大的元素。其中heapDown函数见堆排序
int findKthLargest(vector<int>& nums, int k) {
    // 构造大顶堆
    for(int i = nums.size() / 2 - 1; i >= 0; --i)
        heapDown(nums, i, nums.size() - 1);
    // 前k-1次换到数组末尾后,堆顶nums[0]即为第k大
    int cnt = 0;
    while(++cnt < k){	// 注意:条件 (++cnt < k) 进行 k-1 次操作, (cnt++ < k) 进行 k 次操作
        swap(nums[0], nums[nums.size() - cnt]);
        heapDown(nums, 0, nums.size() - cnt - 1);
    }
    return nums[0]; 	// k-1次操作之后的堆顶即为第k大的元素
}

// 思路2:利用快速排序,找到第k-1个枢点,此时0~k-1是最小的k个值
//			 枢点pivot<k => 搜索[pivot+1, high]。 枢点pivot>k => 搜索[left, pivot-1]
// 			 此方法的渐进时间复杂度理论上是log(n)


剑指offer 数组中的逆序对

在数组中的两个数字,如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对。输入一个数组,求出这个数组中的逆序对的总数P。并将P对1000000007取模的结果输出。 即输出P%1000000007

题目保证输入的数组中没有的相同的数字, 数据范围:对于%50的数据,size<=10^4, 对于%75的数据,size<=10^5, 对于%100的数据,size<=2*10^5

示例:

输入:1,2,3,4,5,6,7,0
输出:7
// 思路:在归并排序的过程中对逆序对进行统计
int InversePairs(vector<int> nums) {
    int cnt = 0;
    mergeCount(nums, 0, nums.size() - 1, cnt);
    return cnt;
}

void mergeCount(vector<int> &nums, int left, int right, int &cnt){
    if(left >= right)
        return;
    int mid = (right - left) / 2 + left;
    mergeCount(nums, left, mid, cnt);
    mergeCount(nums, mid + 1, right, cnt);
    merge(nums, left, mid, right, cnt);
}
void merge(vector<int> &nums, int left, int mid, int right, int &cnt){
    vector<int> tmp;
    int l = left, r = mid + 1;
    while(l <= mid && r <= right)
        if(nums[l] <= nums[r])
            tmp.push_back(nums[l++]);
    else{
        cnt = (cnt + mid - l + 1) % 1000000007;
        tmp.push_back(nums[r++]);
    }
    while(l <= mid)
        tmp.push_back(nums[l++]);
    while(r <= right)
        tmp.push_back(nums[r++]);
    for(int i = left; i <= right; ++i)
        nums[i] = tmp[i - left];
}

leetcode. 280 摆动排序 medium

给定一个无序的数组 nums,将它重新排列成 nums[0] <= nums[1] >= nums[2] <= nums[3]... 的顺序。

示例:

输入: [3, 5, 2, 1, 6, 4]
输出:一个可能的答案是[1, 6, 2, 5, 3, 4]
// 思路1: 先进行排序,然后交换1-2, 3-4, 5-6  ==> O(nlogn)
void wiggleSort(vector<int> &nums) {
    sort(nums.begin(), nums.end());
    if(nums.size() <= 2)
        return;
    for(int i = 2; i < nums.size(); i += 2)
        swap(nums[i], nums[i - 1]);
}

// 思路:由题意的,下标为奇数的大于等于两侧,下标为偶数的小于等于两侧,不满足的话将该数与左侧交换即可 O(n)
void wiggleSort(vector<int> &nums){
    if(nums.size() <= 1)
        return;
    for(int i = 1; i < nums.size(); ++i)
        if((i % 2 == 1 && nums[i] < nums[i - 1]) 
            || (i % 2 == 0 && nums[i] > nums[i - 1]))
            swap(nums[i], nums[i - 1]);
}

leetcode.324 摆动序列II medium

给定一个无序的数组 nums,将它重新排列成 nums[0] < nums[1] > nums[2] < nums[3]... 的顺序。

示例 :

输入: nums = [1, 5, 1, 1, 6, 4]
输出: 一个可能的答案是 [1, 4, 1, 5, 1, 6]

输入: nums = [1, 3, 2, 2, 3, 1]
输出: 一个可能的答案是 [2, 3, 1, 3, 1, 2]

说明: 你可以假设所有输入都会得到有效的结果。

进阶: 你能用 O(n) 时间复杂度和 / 或原地 O(1) 额外空间来实现吗?

// 思路:先进行排序,不能交换相互挨着的两个数,可能相等
// 		将排序后的数组分为前后两半,分别从前半部分和后半部分取出最后一个数,可以保证不等
void wiggleSort(vector<int>& nums) {
    vector<int> temp = nums;
    sort(temp.begin(), temp.end());

    // left_back取中间,right_back取末尾
    int left_back = (nums.size() % 2 == 1) ? nums.size() / 2 : nums.size() / 2 - 1;
    int right_back = nums.size() - 1;

    for(int i = 0; i < nums.size(); ++i)
        nums[i] = (i % 2 == 0) ? temp[left_back--] : temp[right_back--];
}

二分查找

leecode.69 x的平方根 easy

描述:实现 int sqrt(int x) 函数。计算并返回 x 的平方根,其中 x 是非负整数。由于返回类型是整数,结果只保留整数的部分,小数部分将被舍去。

示例 :

输入: 4
输出: 2

输入: 8
输出: 2
说明: 8 的平方根是 2.82842..., 
由于返回类型是整数,小数部分将被舍去。
int mySqrt(int x) {
    if(x <= 1)
        return x;
    int low = 1, high = x;
    while(low <= high){
        int mid = (high - low) / 2 + low;
        if(x / mid == mid)
            return mid;
        else if(x / mid < mid)
            high = mid - 1;
        else
            low = mid + 1;
    }
    return high;
}


leetcode.367 有效的完全平方数 easy

给定一个正整数 num,编写一个函数,如果 num 是一个完全平方数,则返回 True,否则返回 False。

说明:不要使用任何内置的库函数,如 sqrt。

示例 :

输入:16
输出:True

输入:14
输出:False
bool isPerfectSquare(int num) {
    if(num < 1)
        return false;
    int low = 1, high = num;
    while(low <= high){
        int mid = (high - low) / 2 + low;
        if(num / mid == mid && num % mid == 0)
            return true;
        else if(num / mid < mid)
            high = mid - 1;
        else
            low = mid + 1;
    }
    return false;
}


leetcode.441 排列硬币 easy

你总共有 n 枚硬币,你需要将它们摆成一个阶梯形状,第 k 行就必须正好有 k 枚硬币。给定一个数字 n,找出可形成完整阶梯行的总行数。n 是一个非负整数,并且在32位有符号整型的范围内。

示例 :

n = 5
硬币可排列成以下几行:
¤
¤ ¤
¤ ¤
因为第三行不完整,所以返回2.
int arrangeCoins(int n) {
    if(n < 1)
        return 0;
    int low = 1, high = n;
    while(low <= high){
        int mid = (high - low) / 2 + low;
        long long total = (mid + 1) * (long long)mid / 2;
        if(total == n)
            return mid;
        else if(total < n)
            low = mid + 1;
        else
            high = mid - 1;
    }
    return high;
}


leetcode.50 Pow(x,n) medium

实现 pow(x, n) ,即计算 x 的 n 次幂函数。

示例:

输入: 2.00000, 10
输出: 1024.00000

输入: 2.00000, -2
输出: 0.25000
解释: 2-2 = 1/22 = 1/4 = 0.25
// 思路:对x依次做1,2,4,...次幂,得到x, x*x, x*x*x*x,直到指数次数为不大于n的最大值
// 保存x*x*x*x...,并从n中减去该指数次数
// 重复操作,将所有x*x...连续乘起来即为所求
double myPow(double x, int n) {
    bool sign = n < 0;
    long long exp = abs((long long)n);  // 先转化为longlong再abs,防止出现n=INT_MIN
    double res = 1.0;
    while(exp > 0){
        long long e = 1;// 指数  1 -> 2   -> 4 ...
        double mul = x; // 结果  x -> x*x -> x*x*x*x ...
        while((e*2) < exp){   // 直到e*2仍小于exp
            e = e << 1;
            mul = mul * mul;
        }
        exp -= e;
        res *= mul;
    }
    return sign ? 1.0/res : res;
}


剑指offer 数字在排序数组中出现的次数

统计一个数字在排序数组中出现的次数。

// 思路:二分搜索num+0.5,num-0.5的位置,它们区间内个数即为所求
int biSearch(const vector<int> &data, double num){
    // 对数组data = 1 2 3a 3b 3c 4 4 5
    // num = 2.5 -> 返回2所在下标
    // num = 3.5 -> 返回3c所在下标
    int low = 0, high = data.size() - 1;
    while(low <= high){
        int mid = (high - low) / 2 + low;
        if(data[mid] < num)
            low = mid + 1;
        else
            high = mid - 1;
    }
    return high;
}
int GetNumberOfK(vector<int> data ,int k) {
    return biSearch(data, k + 0.5) - biSearch(data, k - 0.5);
}


leetcode.162 寻找峰值 medium

峰值元素是指其值大于左右相邻值的元素。给定一个输入数组 nums,其中 nums[i] ≠ nums[i+1],找到峰值元素并返回其索引。数组可能包含多个峰值,在这种情况下,返回任何一个峰值所在位置即可。你可以假设 nums[-1] = nums[n] = -∞。要求O(logN)时间复杂度

示例 1:

输入: nums = [1,2,3,1]
输出: 2
解释: 3 是峰值元素,你的函数应该返回其索引 2。

输入: nums = [1,2,1,3,5,6,4]
输出: 1 或 5 
解释: 你的函数可以返回索引 1,其峰值元素为 2;或者返回索引 5, 其峰值元素为 6。

// 思路:采用二分查找法,left = 0, right = nums.size() - 1为起点
// nums[mid] < nums[mid + 1],局部上升,右侧必然有峰值       (注意,mid+1必然是存在的, mid-1不一定)
// nums[mid] > nums[mid + 1],局部下降,左侧必然有峰值
int findPeakElement(vector<int>& nums) {
    int left = 0, right = nums.size() - 1;
    while(left < right){
        int mid = (right - left) / 2 + left;
        if(nums[mid] < nums[mid + 1])	// 是否取等,后面if-else-都需要仔细调整
            left = mid + 1;
        else
            right = mid;
    }
    return right;        
}

leetcode.240 搜索二维矩阵 II medium剑指offer 二维数组中的查找

编写一个高效的算法来搜索 m x n 矩阵 matrix 中的一个目标值 target。该矩阵具有以下特性:

每行的元素从左到右升序排列。 每列的元素从上到下升序排列。

示例: 现有矩阵 matrix 如下:

[
[1,   4,  7, 11, 15],
[2,   5,  8, 12, 19],
[3,   6,  9, 16, 22],
[10, 13, 14, 17, 24],
[18, 21, 23, 26, 30]
]

给定 target = 5,返回 true。

给定 target = 20,返回 false。

// 思路:从左下角、或者右上角开始,进行二分搜索
bool searchMatrix(vector<vector<int>>& matrix, int target) {
    if(matrix.empty() || matrix[0].empty())
        return false;
    int i = matrix.size() - 1, j = 0;
    while(i >= 0 && j < matrix[0].size())
        if(matrix[i][j] == target)
            return true;
        else if(matrix[i][j] < target)
            ++j;
        else
            --i;
    return false;
}

贪心思想

leetcode.445 分发饼干 easy

假设你是一位很棒的家长,想要给你的孩子们一些小饼干。但是,每个孩子最多只能给一块饼干。对每个孩子 i ,都有一个胃口值 gi ,这是能让孩子们满足胃口的饼干的最小尺寸;并且每块饼干 j ,都有一个尺寸 sj 。如果 sj >= gi ,我们可以将这个饼干 j 分配给孩子 i ,这个孩子会得到满足。你的目标是尽可能满足越多数量的孩子,并输出这个最大数值。

注意:

你可以假设胃口值为正。 一个小朋友最多只能拥有一块饼干。

示例 1:

输入: [1,2,3], [1,1]
输出: 1
解释: 
你有三个孩子和两块小饼干,3个孩子的胃口值分别是:1,2,3。
虽然你有两块小饼干,由于他们的尺寸都是1,你只能让胃口值是1的孩子满足。
所以你应该输出1。
// 思路:从胃口最小的孩子开始,从最小的饼干开始喂直到能满足为止
int findContentChildren(vector<int>& g, vector<int>& s) {
    sort(g.begin(), g.end());
    sort(s.begin(), s.end());
    int cnt = 0;
    int i = 0, j = 0;
    // 从胃口最小的孩子喂饼干,从最小的饼干开始喂
    while(i < g.size() && j < s.size()){
        if(g[i] <= s[j]){
            ++cnt;
            ++i;
            ++j;
        }else
            ++j;
    }
    return cnt;
}


leetcode.452 用最少数量的箭引爆气球 medium

在二维空间中有许多球形的气球。对于每个气球,提供的输入是水平方向上,气球直径的开始和结束坐标。由于它是水平的,所以y坐标并不重要,因此只要知道开始和结束的x坐标就足够了。开始坐标总是小于结束坐标。平面内最多存在104个气球。

一支弓箭可以沿着x轴从不同点完全垂直地射出。在坐标x处射出一支箭,若有一个气球的直径的开始和结束坐标为 xstart,xend, 且满足 xstart ≤ x ≤ xend,则该气球会被引爆。可以射出的弓箭的数量没有限制。 弓箭一旦被射出之后,可以无限地前进。我们想找到使得所有气球全部被引爆,所需的弓箭的最小数量。

示例:

输入:
[[10,16], [2,8], [1,6], [7,12]]
输出:
2
解释:
对于该样例,我们可以在x = 6(射爆[2,8],[1,6]两个气球)和 x = 11(射爆另外两个气球)。
// 思路:左往右依次射箭,每次保证引爆最多的气球
int findMinArrowShots(vector<vector<int>>& points) {
    if(points.size() <= 1)
        return points.size();
    sort(points.begin(), points.end());
    int cnt = 1;
    int rightEdge = points[0][1];   // 第一个圆右侧为右边界
    for(int idx = 1; idx < points.size(); ++idx){
        // 圆在边界内,需更新边界为它们右侧的较小值
        if(points[idx][0] <= rightEdge)  
            rightEdge = min(rightEdge, points[idx][1]);
        // 圆不在边界内,引爆气球++cnt,并更新右边界为新的右侧值
        else{   
            ++cnt;
            rightEdge = points[idx][1];
        }
    }
    return cnt;
}


leetcode.135 分发糖果 hard

老师想给孩子们分发糖果,有 N 个孩子站成了一条直线,老师会根据每个孩子的表现,预先给他们评分。

你需要按照以下要求,帮助老师给这些孩子分发糖果:

每个孩子至少分配到 1 个糖果。 相邻的孩子中,评分高的孩子必须获得更多的糖果。 那么这样下来,老师至少需要准备多少颗糖果呢?

示例:

输入: [1,0,2]
输出: 5
解释: 你可以分别给这三个孩子分发 2、1、2 颗糖果。

输入: [1,2,2]
输出: 4
解释: 你可以分别给这三个孩子分发 1、2、1 颗糖果。
第三个孩子只得到 1 颗糖果,这已满足上述两个条件。
// 思路:
// 步骤1. 先每个人发一颗糖果
// 步骤2. 从左往右遍历,i比i-1表现好就在i-1基础上+1糖果
// 步骤3. 从右往左遍历,i比i+1表现好就在i+1基础上+1糖果(需判断已有糖果是否已经比i+1多)
int candy(vector<int>& ratings) {
    vector<int> candies(ratings.size(), 1);

    for(int i = 1; i < candies.size(); ++i)
        if(ratings[i] > ratings[i - 1])
            candies[i] = candies[i - 1] + 1;

    for(int i = candies.size() - 2; i >= 0; --i)
        if(ratings[i] > ratings[i + 1])
            candies[i] = max(candies[i], candies[i + 1] + 1);

    return accumulate(candies.begin(), candies.end(), 0);
}


leetcode.122 买卖股票的最佳时机 II easy

给定一个数组,它的第 i 个元素是一支给定股票第 i 天的价格。设计一个算法来计算你所能获取的最大利润。你可以尽可能地完成更多的交易(多次买卖一l意:你不能同时参与多笔交易(你必须在再次购买前出售掉之前的股票)

// 思路:只要p[i] > p[i - 1],就在i-1买入并在i卖出。某天既买入又卖出可视为没有做任何操作
int maxProfit(vector<int>& prices) {
    if(prices.size() <= 1)
        return 0;
    int prof = 0;
    for(int i = 1; i < prices.size(); ++i)
        prof += (prices[i] > prices[i - 1]) ? prices[i] - prices[i - 1] : 0;
    return prof;
}


leetcode.605 种花问题 easy

给定一个花坛(表示为一个数组包含0和1,其中0表示没种植花,1表示种植了花),和一个数 n 。能否在不打破种植规则的情况下种入 n 朵花?能则返回True,不能则返回False。

// 思路:直接遍历,满足条件就种下。注意两类特殊情况和花坛两端
bool canPlaceFlowers(vector<int>& bed, int n) {
    int seeds = 0;
    if(n == 0)
        return true;
    if(bed.size() == 1)
        return bed[0] == 0;

    if(bed.size() >= 2 && bed[0] == 0 && bed[1] == 0){
        ++seeds;
        bed[0] = 1;
    }
    for(int i = 1; i < bed.size() - 1; ++i)
        if(bed[i - 1] == 0 && bed[i] == 0 && bed[i + 1] == 0){
            ++seeds;
            bed[i] = 1;
        }
    if(bed.size() >= 2 && bed[bed.size() - 2] == 0 && bed.back() == 0)
        ++seeds;

    return seeds >= n;
}


leetcode.665 非递减数列 easy

给定一个长度为 n 的整数数组,你的任务是判断在最多改变 1 个元素的情况下,该数组能否变成一个非递减数列。我们是这样定义一个非递减数列的: 对于数组中所有的 i (1 <= i < n),满足 array[i] <= array[i + 1]。

示例 1:

输入: [4,2,3]
输出: True
解释: 你可以通过把第一个4变成1来使得它成为一个非递减数列。

输入: [4,2,1]
输出: False
解释: 你不能在只改变一个元素的情况下将其变为非递减数列。
// 思路:对逆序的数对进行计数,同时需要确认是否可以修正
// 找到nums[i] > nums[i + 1]的位置,再对nums[i] or nums[i + 1]进行更改
bool checkPossibility(vector<int>& nums) {
    int cnt = 0;
    for(int i = 0; i < nums.size() - 1; ++i)
        if(nums[i] > nums[i + 1]){
            ++cnt;
            // 优先考虑更改nums[i]为它能接受的最小值,即nums[i - 1]
            if(i == 0)
                continue;
            if(nums[i - 1] <= nums[i + 1])
                nums[i] = nums[i - 1];	// 2 4(i) 2 5 => 改4为2
            else
                nums[i + 1] = nums[i];	// 3 4(i) 2 5 => 不能改4
        }
    return cnt <= 1;
}


leetcode.392 判断子序列 medium

给定字符串 s 和 t ,判断 s 是否为 t 的子序列。你可以认为 s 和 t 中仅包含英文小写字母。字符串 t 可能会很长(长度 ~= 500,000),而 s 是个短字符串(长度 <=100)。字符串的一个子序列是原始字符串删除一些(也可以不删除)字符而不改变剩余字符相对位置形成的新字符串。(例如,"ace"是"abcde"的一个子序列,而"aec"不是)。

// 思路:直接比对每一个字母即可
bool isSubsequence(string s, string t) {
    if(s.empty())
        return true;
    int p = 0;
    for(const auto &c:t)
        if(s[p] == c)
            if(++p == s.size())
                return true;
    return false;
}


leetcode.763 划分字母区间 medium

字符串 S 由小写字母组成。我们要把这个字符串划分为尽可能多的片段,同一个字母只会出现在其中的一个片段。返回一个表示每个字符串片段的长度的列表。

示例 1:

输入: S = "ababcbacadefegdehijhklij"
输出: [9,7,8]
解释:
划分结果为 "ababcbaca", "defegde", "hijhklij"。每个字母最多出现在一个片段中。
像 "ababcbacadefegde", "hijhklij" 的划分是错误的,因为划分的片段数较少
// 思路: 第一遍遍历得到每个字母最后出现的位置map
//       第二遍通过left,right指示区间,cur指示当前遍历到的位置
//       	使用right = max(map[s[cur]], right)扩张右侧边界
//       	cur==right时说明区间内的字母未出现在其它区域
vector<int> partitionLabels(string str) {
    map<char, int> map;
    for(int i = 0; i < str.size(); ++i)
        map[str[i]] = i;

    int left = 0, right = map[str[0]], cur = 0;
    vector<int> res;
    while(right < str.size()){
        right = max(right, map[str[cur]]);	// 更新右边界
        if(right == cur){				   // 此时[left,right]之间的字母未出现在其它位置
            res.push_back(right - left + 1);
            left = right + 1;
            right = (cur < str.size() - 1) ? map[str[cur + 1]] : str.size();
        }
        ++cur;
    }

    return res;
}


leetcode.56 合并区间 medium

给出一个区间的集合,请合并所有重叠的区间。

示例 1:

输入: [[1,3],[2,6],[8,10],[15,18]]
输出: [[1,6],[8,10],[15,18]]
解释: 区间 [1,3] 和 [2,6] 重叠, 将它们合并为 [1,6].

输入: [[1,4],[4,5]]
输出: [[1,5]]
解释: 区间 [1,4] 和 [4,5] 可被视为重叠区间。
// 思路:
// 将所有区间从小到大排序
// 选择第一个区间右端点作为界限,不断向右扩展其势力范围内的其它区间
vector<vector<int>> merge(vector<vector<int>>& intervals) {
    if(intervals.size() <= 1)
        return intervals;
    sort(intervals.begin(), intervals.end());
    vector<vector<int>> res;
    auto tmp = intervals[0];
    for(int i = 1; i < intervals.size(); ++i){
        if(intervals[i][0] <= tmp[1])
            tmp[1] = max(tmp[1], intervals[i][1]);
        else{
            res.push_back(tmp);
            tmp = intervals[i];
        }
    }
    res.push_back(it0);	// 最后一个tmp由于没有进入到for循环中的else语句,需要额外加入到res中
    return res;
}


leetcode.406 根据身高重建队列 medium

假设有打乱顺序的一群人站成一个队列。 每个人由一个整数对(h, k)表示,其中h是这个人的身高,k是排在这个人前面且身高大于或等于h的人数。 编写一个算法来重建这个队列。

注意:总人数少于1100人。

示例

输入:
[[7,0], [4,4], [7,1], [5,0], [6,1], [5,2]]
输出:
[[5,0], [7,0], [5,2], [6,1], [4,4], [7,1]]
// 思路
// 将所有人(h,k)按身高h降序,k升序进行排列得 [7,0],[7,1],[6,1],[5,0],[5,2],[4,4]
// 此时所有人前面的人均不低于它,只需将(hi,ki)往前swap直到ki = i为止
bool comp(const vector<int> &p1, const vector<int> &p2){
    return p1[0] > p2[0] || (p1[0] == p2[0] && p1[1] < p2[1]);
}
vector<vector<int>> reconstructQueue(vector<vector<int>>& people) {
    sort(people.begin(), people.end(), comp);

    for(int i = 0; i < people.size(); ++i)
        for(int k = i; k > people[k][1]; --k)
            swap(people[k], people[k - 1]);

    return people;
}


leetcode.621 任务调度器 medium

给定一个用字符数组表示的 CPU 需要执行的任务列表。其中包含使用大写的 A - Z 字母表示的26 种不同种类的任务。任务可以以任意顺序执行,并且每个任务都可以在 1 个单位时间内执行完。CPU 在任何一个单位时间内都可以执行一个任务,或者在待命状态。然而,两个相同种类的任务之间必须有长度为 n 的冷却时间,因此至少有连续 n 个单位时间内 CPU 在执行不同的任务,或者在待命状态。你需要计算完成所有任务所需要的最短时间。

示例 1:

输入: tasks = ["A","A","A","B","B","B"], n = 2
输出: 8
执行顺序: A -> B -> (待命) -> A -> B -> (待命) -> A -> B.
// 思路:统计每个任务出现次数,假设为AAAABBBBCCC,n=3
// 放A: |A---|A---|A---|A 
// 放B:|AB--|AB--|AB--|AB   => 执行时间 = (4 - 1) * (n + 1) + 2
// 放C:|ABC-|ABC-|ABC-|AB	 其中4是最多的次数,2是最多次数的任务
// 另外,当出现AAABBBCCDD,n=2时,总能通过尽量稀疏分布使得不需要等待时间
// 放A:|A--|A--|A--|
// 放B: |AB-|AB-|AB-|
// 放C:|ABC|AB-|ABC|		(这里是关键)
// 放D:|ABC|ABD|ABC|D
int leastInterval(vector<char>& tasks, int n) {
    vector<int> stat(26, 0);
    for(const auto &c:tasks)
        ++stat[c - 'A'];
    sort(stat.begin(), stat.end());

    int max_task = stat[25], max_cnt = 0;
    for(const auto &cnt:stat)
        max_cnt += cnt == stat[25] ? 1 : 0;

    int time = (max_task - 1) * (n + 1) + max_cnt;
    time = time > tasks.size() ? time : tasks.size();

    return time;
}

leetcode.861 翻转矩阵后的得分 medium

有一个二维矩阵 A 其中每个元素的值为 0 或 1 。移动是指选择任一行或列,并转换该行或列中的每一个值:将所有 0 都更改为 1,将所有 1 都更改为 0。在做出任意次数的移动后,将该矩阵的每一行都按照二进制数来解释,矩阵的得分就是这些数字的总和。返回尽可能高的分数。

示例:

输入:[[0,0,1,1],[1,0,1,0],[1,1,0,0]]
输出:39
解释:转换为 [[1,1,1,1],[1,0,0,1],[1,1,1,1]] => 0b1111 + 0b1001 + 0b1111 = 15 + 9 + 15 = 39
// 思路:首先每行翻转,保证第一位是1, 1000 > 0111
//      然后第二列开始,每列翻转保证该列1数量比0多
void reverseRow(vector<vector<int>> &A, int row){
    for(auto &c:A[row])
        c ^= 1;
}
void reverseCol(vector<vector<int>> &A, int col){
    for(int i = 0; i < A.size(); ++i)
        A[i][col] ^= 1;
}
int matrixScore(vector<vector<int>>& A) {
    if(A.empty() || A[0].empty())
        return 0;
    // 每行变换,保证每行第一位是1
    for(int row = 0; row < A.size(); ++row)
        if(A[row][0] == 0)
            reverseRow(A, row);
    // 列变换,保证每列1数量比0多
    for(int col = 1; col < A[0].size(); ++col){
        int ones = 0, zeros = 0;
        for(int i = 0; i < A.size(); ++i)
            A[i][col] == 0 ? ++ones : ++zeros;
        if(ones > zeros)
            reverseCol(A, col);
    }
    int sum = 0;	// 计算最终结果
    for(const auto &row:A){
        int sum_row = 0;
        for(const auto &i:row)
            sum_row = (sum_row << 1) + i;	// 移位运算符优先级很低,需要打上括号
        sum += sum_row;
    }
    return sum;
}

leetcode. 853 车队 medium

N 辆车沿着一条车道驶向位于 target 英里之外的共同目的地。

每辆车 i 以恒定的速度 speed[i] (英里/小时),从初始位置 position[i] (英里) 沿车道驶向目的地。

一辆车永远不会超过前面的另一辆车,但它可以追上去,并与前车以相同的速度紧接着行驶。

此时,我们会忽略这两辆车之间的距离,也就是说,它们被假定处于相同的位置。

车队 是一些由行驶在相同位置、具有相同速度的车组成的非空集合。注意,一辆车也可以是一个车队。

即便一辆车在目的地才赶上了一个车队,它们仍然会被视作是同一个车队。

会有多少车队到达目的地?

示例:

输入:target = 12, position = [10,8,0,5,3], speed = [2,4,1,1,3]
输出:3
解释:
从 10 和 8 开始的车会组成一个车队,它们在 12 处相遇。
从 0 处开始的车无法追上其它车,所以它自己就是一个车队。
从 5 和 3 开始的车会组成一个车队,它们在 6 处相遇。
请注意,在到达目的地之前没有其它车会遇到这些车队,所以答案是 3。

提示:

0 <= N <= 10 ^ 4 0 < target <= 10 ^ 6 0 < speed[i] <= 10 ^ 6 0 <= position[i] < target 所有车的初始位置各不相同。

// 计算每个车到达终点的所花时间, 并将车队按起始位置进行排序
// 从最后一个车开始开始遍历, 时间比最后一个车还短的车必然是同一个车队
int carFleet(int target, vector<int>& position, vector<int>& speed) {
    vector<pair<int,double>> times;     // 每辆车到达终点花费的时间
    for(int i = 0; i < position.size(); ++i)
        times.push_back({position[i], (target - position[i]) * 1.0 / speed[i]}); 
    sort(times.begin(), times.end());

    int count = 0, idx = times.size() - 1;
    while(idx >= 0){
        ++count;                        // 车队+1
        double t = times[idx].second;   // 最后一个车的花费的时间
        --idx;
        while(idx >= 0 && times[idx].second <= t)
            --idx;                      // 跳过所有花费时间比最后一个车还短的
    }
    return count;
}

双指针思想

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

// 思路:两个下标分别指向首、尾,同时向中间靠拢以搜索两数
vector<int> twoSum(vector<int>& nums, int target) {
    int left = 0, right = nums.size() - 1;
    while(left <= right){
        int sum = nums[left] + nums[right];
        if(sum == target)
            return { left + 1, right + 1 };
        else if(sum < target)
            ++left;
        else
            --right;
    }
    return {0,0};	// 题目必有解,不会执行到这里,但没有这句话无法通过编译
}


leetcode.345 翻转字符串中的元音字母 easy

const set<char> sc = { 'a','e','i','o','u' };
string reverseVowels(string s) {
    int left = 0, right = s.size() - 1;
    while(left <= right){
        while(left < s.size() && sc.count(tolower(s[left])) == 0)
            ++left;	// 左往右定位到元音
        while(right >= 0 && sc.count(tolower(s[right])) == 0)
            --right;// 右往左定位到元音
        if(left <= right)
            swap(s[left++], s[right--]);
    }
    return s;
}


leetcode.633 平方数之和 easy

bool judgeSquareSum(int c) {
    long long left = 0, right = sqrt(c);
    while(left <= right){
        long long num = left * left + right * right;
        if(num < c)
            ++left;
        else if(num > c)
            --right;
        else if(num == c)
            return true;                
    }
    return false;
}


leetcode.680 验证回文字符串 II easy

// 思路:定位到第一个非回文的位置,刨除左字母或右字母后验证剩余部分是否为回文串
bool valid(const string &s, int left, int right){
    while(left <= right)
        if(s[left++] != s[right--])
            return false;
    return true;
}
bool validPalindrome(string s) {
    int left = 0, right = s.size() - 1;
    while(left <= right){
        if(s[left] != s[right])	// 定位后验证
            return valid(s, left + 1, right) || valid(s, left, right - 1);
        ++left;
        --right;
    }
    return true;
}


leetcode.88 合并两个有序数组 easy

给定两个有序整数数组 nums1nums2,将 nums2 合并到 nums1 中*,*使得 num1 成为一个有序数组。

示例:

输入:
nums1 = [1,2,3,0,0,0], m = 3
nums2 = [2,5,6],       n = 3
输出: [1,2,2,3,5,6]
// 思路:从num1的m+n-1位置处往前填充数字
void merge(vector<int>& nums1, int m, vector<int>& nums2, int n) {
    int idx = m + n - 1;
    int p1 = m - 1, p2 = n - 1;
    while(idx >= 0)
        if(p1 >= 0 && p2 >= 0){ // 在num1,nums2中都还有未融合的数才进行
            nums1[idx--] = max(nums1[p1], nums2[p2]);
            nums1[p1] > nums2[p2] ? --p1 : --p2;
        }else
            break;
    // p2>=0说明nums2中存在数没融合,将它放到nums1中; p1>=0,此时nums1中已经排好无需再管
    while(p2 >= 0)
        nums1[idx--] = nums2[p2--];
}


leetcode.283 移动零 easy

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

示例:

输入: [0,1,0,3,12]
输出: [1,3,12,0,0]
// 与leetcode.88类似,不过这里从前往后填充
void moveZeroes(vector<int>& nums) {
    int p1 = 0, p2 = 0;
    while(p2 < nums.size()){
        if(nums[p2] != 0)
            nums[p1++] = nums[p2];
        ++p2;
    }
    while(p1 < nums.size())
        nums[p1++] = 0;
}

// 写得更紧凑一些
void moveZeros(vector<int> &nums){
    int p = 0;
    for(auto num:nums)
        if(num != 0)
            nums[p++] = num;
    while(p < nums.size())
        nums[p++] = 0;
}


leetcode.19 删除链表的倒数第N个节点 medium

// 思路:快慢指针,快指针先走n次,然后快、慢一起走,快指针到达nullptr时慢指针即倒数第n个
// 注意:题目保证n有效,即1 <= n <= 总节点数
ListNode* removeNthFromEnd(ListNode* head, int n) {
    ListNode *fast = head, *slow = head;
    int cnt = 0;
    while(cnt++ < n)
        fast = fast->next;
    // n == 总节点数,倒数第n个节点即头结点
    if(fast == nullptr){
        ListNode *tmp_head = head->next;
        delete head;
        return tmp_head;
    }
    // 其余情况:条件fast->next != nullptr可将slow定位到待删节点前一个
    //          条件fast != nullptr可将slow定位到待删节点。    注意区分使用
    while(fast->next != nullptr){
        fast = fast->next;
        slow = slow->next;
    }
    auto tmp = slow->next->next;
    delete slow->next;
    slow->next = tmp;
    return head;
}


leetcode.141 环形链表 easy

// 思路:快慢指针法,若有环则必然在某个点相遇
// 另外:也可以用set或者map对已出现的节点进行记录,需要额外空间O(n)
bool hasCycle(ListNode *head) {
    ListNode *fast = head, *slow = head;
    while(fast != nullptr && fast->next != nullptr){
        fast = fast->next->next;
        slow = slow->next;
        if(fast == slow)
            return true;
    }
    return false;
}


剑指Offer - 链表中环的入口节点

// __a__.______ b  a,b,c分为为各段长度
//     c \_____*  <---- 快慢指针相遇点
//  若有环,快慢指针在图中 * 处相遇,存在关系 fast = a+b+(b+c)*k = 2*slow =2*(a+b), k为快指针圈数
//  解得: a = (b + c)*(k - 1) + c
//  即:从头结点和相遇点分别出发,必然在入口节点相遇
ListNode* EntryNodeOfLoop(ListNode* head){
    ListNode* fast = head, *slow = head;
    while(fast != nullptr && fast->next != nullptr){
        fast = fast->next->next;
        slow = slow->next;
        if(slow == fast){	// 在*相遇
            fast = head;	// fast从头出发,slow从相遇点出发,一次跳一格
            while(fast != slow){
                fast = fast->next;
                slow = slow->next;
            }
            return fast;	// 入口接单相遇后返回fast或slow都可以
        }
    }
    return nullptr;
}


leetcode.3 无重复字符的最长子串 medium

给定一个字符串,请你找出其中不含有重复字符的 最长子串 的长度。

示例 1:

输入: "abcabcbb"
输出: 3 
解释: 因为无重复字符的最长子串是 "abc",所以其长度为 3。

输入: "bbbbb"
输出: 1
解释: 因为无重复字符的最长子串是 "b",所以其长度为 1。

输入: "pwwkew"
输出: 3
解释: 因为无重复字符的最长子串是 "wke",所以其长度为 3。
     请注意,你的答案必须是 子串 的长度,"pwke" 是一个子序列,不是子串。
// 思路:用left, right表示无重复字符的窗口,用一个map记录right之前所有字符最后出现的位置loc
// 依次遍历每个字符的位置作为右端right,判断s[right]最后出现的位置是否在[left, right]内
int lengthOfLongestSubstring(string s) {
    int left = 0, maxLen = 0;
    map<char, int> loc; // 记录已遍历字符最后出现的位置
    for(int right = 0; right < s.size(); ++right){
        if(loc.count(s[right]) > 0 && loc[s[right]] >= left)	// 判断
            left = loc[s[right]] + 1;
        loc[s[right]] = right;
        maxLen = max(maxLen, right - left + 1);
    }
    return maxLen;
}


leetcode.524 通过删除字母匹配到字典里最长单词 medium

给定一个字符串和一个字符串字典,找到字典里面最长的字符串,该字符串可以通过删除给定字符串的某些字符来得到。如果答案不止一个,返回长度最长且字典顺序最小的字符串。如果答案不存在,则返回空字符串。

示例 1:

输入:s = "abpcplea", d = ["ale","apple","monkey","plea"]
输出: "apple"

输入:s = "abpcplea", d = ["a","b","c"]
输出: "a"
// 思路:为字典里每个字符串分配一个指针idx, 表示该字符串已经匹配到了idx
//      遍历给定字符串str每个字母c,看c是否能与字典中字符串target_s[idx]进行匹配,匹配则idx+1
//      遍历完成后,看字典中每个字符串的idx==target_s.size()与否,相等则表示已匹配到
string findLongestWord(string str, vector<string>& vs) {
    unordered_map<string, int> dict;
    for(const auto &item:vs)
        dict[item] = 0;		// 分配指针
    
    for(const auto &c:str) // 比对
        for(auto &m:dict)  // 注意,这里一定要 auto &,否则更改后不会作用到m上
            if(c == m.first[m.second])
                ++m.second;

    string res = "";
    int maxLen = 0;
    for(const auto &m:dict)
        if(m.second == m.first.size()){ // 匹配成功的
            if(m.first.size() > maxLen){
                res = m.first;
                maxLen = m.first.size();
            }else if(m.first.size() == maxLen)
                res = res < m.first ? res : m.first;    // 字典序更小的
        }

    return res;
}

leetcode.713 乘积小于K的子数组 medium

给定一个正整数数组 nums。

找出该数组内乘积小于 k 的连续的子数组的个数。

示例 :

输入: nums = [10,5,2,6], k = 100
输出: 8
解释: 8个乘积小于100的子数组分别为: [10], [5], [2], [6], [10,5], [5,2], [2,6], [5,2,6]。
需要注意的是 [10,5,2] 并不是乘积小于100的子数组。

说明: 0 < nums.length <= 50000, 0 < nums[i] < 1000, 0 <= k < 10^6

// 思路:由于nums中全都是正数,因此可以使用双指针
int numSubarrayProductLessThanK(vector<int>& nums, int k) {
    if(nums.empty() || k == 0)
        return 0;

    int prod = 1;
    int left = 0, right = 0;
    int cnt = 0;
    while(right < nums.size()){
        prod *= nums[right];
        while(left <= right && prod >= k){
            prod /= nums[left];
            ++left;
        }
        cnt += prod < k ? right - left + 1 : 0;
        ++right;
    }
    return cnt;
}

搜索

广度优先搜索BFS、深度优先搜索DFS是最长用的两种搜索方法,广泛应用在图、二维数组、树的搜索和遍历中。它们最本质的区别:BSF是先入先出的遍历过程,DFS是先入后出的遍历过程;因此,在搜索过程中,BFS一般借助于队列,DFS一般借助于栈,这一点要非常明确!

广度优先搜索BFS

leetcode.102 二叉树的层次遍历

二叉树的层次遍历是及其典型的广度优先搜索。广度优先即:一层一层向下,距离根同一距离的节点遍历完后再遍历深一层的节点,例如:

给定二叉树: [3,9,20,null,null,15,7],
    3
   / \
  9  20
 /  \
15   7
层次遍历结果为:[[3], [9,20], [15,7]]
// 注意:返回结果是每一层的节点作为一个vector,所有层再作为一个vector
vector<vector<int>> levelOrder(TreeNode* root) {
    if(root == nullptr)
        return {};
    vector<vector<int>> res;
    queue<TreeNode*> q;
    q.push(root);
    while(!q.empty()){
        int cnt = q.size(); // 每层的节点数量
        vector<int> level_nums;
        while(cnt-- > 0){	
            auto tmp = q.front();
            q.pop();
            level_nums.push_back(tmp->val);
            if(tmp->left != nullptr)
                q.push(tmp->left);
            if(tmp->right != nullptr)
                q.push(tmp->right);
        }
        res.push_back(level_nums);
    }
    return res;
}

// 如果整体返回一个 vector<int>,则写法如下:
vector<int> levelOrder(TreeNode* root) {
    if(root == nullptr)
        return {};
    vector<int> res;
    queue<TreeNode*> q;
    q.push(root);
    while(!q.empty()){
        auto tmp = q.front();
        q.pop();
        res.push_back(tmp->val);
        if(tmp->left != nullptr)
            q.push(tmp->left);
        if(tmp->right != nullptr)
            q.push(tmp->right);
    }
    return res;
}


leetcode.199 二叉树的右视图

给定一棵二叉树,想象自己站在它的右侧,按照从顶部到底部的顺序,返回从右侧所能看到的节点值。

示例:

输入: [1,2,3,null,5,null,4]
输出: [1, 3, 4]
解释:
   1            <---
 /   \
2     3         <---
 \     \
  5     4       <---

思路:使用层次遍历,每层遍历最后一个节点时,保存节点的值

注:使用深度优先搜索也可以求解,查看数据结构-树-树的遍历、或深度优先搜索DFS


leetcode.101 对称二叉树

给定一个二叉树,检查它是否是镜像对称的。例如,二叉树 [1,2,2,3,4,4,3] 是对称的。

    1
   / \
  2   2
 / \ / \
3  4 4  3

但是下面这个 [1,2,2,null,3,null,3] 则不是镜像对称的:

  1
 / \
2   2
 \   \
  3    3

说明:

如果你可以运用递归和迭代两种方法解决这个问题,会很加分。

// 递归法,递归不需要借助queue
bool isSymmetric(TreeNode* root) {
    if(root == nullptr)
        return true;
    return check(root->left, root->right);
}
bool check(TreeNode* l, TreeNode *r){
    if(l == nullptr && r == nullptr)
        return true;
    if(l == nullptr || r == nullptr || l->val != r->val)
        return false;
    return check(l->left, r->right) && check(r->left, l->right);
}

// 迭代法,借助queue实现
bool isSymmetric(TreeNode* root) {
    if(root == nullptr)
        return true;
    queue<TreeNode*> ql;
    queue<TreeNode*> qr;
    ql.push(root->left);
    qr.push(root->right);
    while(!ql.empty() && !qr.empty()){
        auto l = ql.front();
        auto r = qr.front();
        ql.pop();
        qr.pop();
        if(l == nullptr && r == nullptr)
            continue;
        if(l == nullptr || r == nullptr || l->val != r->val)
            return false;
        ql.push(l->left);
        qr.push(r->right);
        ql.push(l->right);
        qr.push(r->left);
    }
    return true;
}


leetcode.127 单词接龙 medium

给定两个单词(beginWord 和 endWord)和一个字典,找到从 beginWord 到 endWord 的最短转换序列的长度。转换需遵循如下规则:每次转换只能改变一个字母;转换过程中的中间单词必须是字典中的单词。说明:

如果不存在这样的转换序列,返回 0。 所有单词具有相同的长度。 所有单词只由小写字母组成。 字典中不存在重复的单词。 你可以假设 beginWord 和 endWord 是非空的,且二者不相同。

示例 1:

输入:
beginWord = "hit",
endWord = "cog",
wordList = ["hot","dot","dog","lot","log","cog"]
输出: 5
解释: 一个最短转换序列是 "hit" -> "hot" -> "dot" -> "dog" -> "cog",
返回它的长度 5。
// 对beginword里的每一个字母,将它替换为26个字母(26个搜索方向),若替换后存在于wordList中,则可以作为下一层
// 示例1的搜索变换过程为 [hit] -> [hot] -> [dot, lot] -> [dog, log] -> [cog]
int ladderLength(string beginWord, string endWord, vector<string>& wordList) {
    unordered_set<string> dict(wordList.begin(), wordList.end());
    if(dict.count(endWord) == 0)   // 词典中必然存在endWord才能转换
        return 0;
    dict.erase(beginWord);

    queue<string> q;    // 一般使用queue辅助广度优先搜索
    q.push(beginWord);
    int level = 1;      // beginWord算一层

    while(!q.empty()){
        int node_num = q.size();	// 该层的节点数
        while(node_num-- > 0){		// 遍历该层的所有节点
            string node = q.front();
            q.pop();
            if(node == endWord)
                return level;
            for(int i = 0; i < node.size(); ++i){	// 对每个字母
                char node_i = node[i];
                for(char c = 'a'; c < 'z'; ++c){	// 26个搜索方向,即替换为a-z
                    node[i] = c;
                    if(dict.count(node) > 0){
                        q.push(node);
                        dict.erase(node);   // 剔除该节点,防止后面再次搜索到
                    }
                }
                node[i] = node_i;
            }
        }
        ++level;
    }
    return 0;   // 遍历到这说明没搜索到
}

深度优先搜索DFS

leetcode.200 岛屿数量 medium

给定一个由 '1'(陆地)和 '0'(水)组成的的二维网格,计算岛屿的数量。一个岛被水包围,并且它是通过水平方向或垂直方向上相邻的陆地连接而成的。你可以假设网格的四个边均被水包围。

示例 1:

输入:
11110
11010
11000
00000
输出: 1

输入:
11000
11000
00100
00011
输出: 3
// 思路:gird(i,j)=1,岛屿+1,同时删除岛屿连通的陆地->上下左右四个方向
int numIslands(vector<vector<char>>& islands) {
    if(islands.empty() || islands[0].empty())
        return 0;
    int num = 0;
    for(int i = 0; i < islands.size(); ++i)
        for(int j = 0; j < islands[0].size(); ++j)
            if(islands[i][j] == '1')
                ++num, dfs(islands, i, j);
    return num;
}
void dfs(vector<vector<char>> &islands, int i, int j){
    // 超出边界,或者(i,j)是水域,直接返回
    if(i < 0 || i > islands.size() - 1 || j < 0 || j > islands[0].size() - 1 || islands[i][j] == '0')
        return;
    
    // 删除陆地和它上下左右四个方向上的陆地
    islands[i][j] = '0';
    dfs(islands, i - 1, j);
    dfs(islands, i + 1, j);
    dfs(islands, i, j - 1);
    dfs(islands, i, j + 1);
}


leetcode.695 岛屿的最大面积 medium

给定一个包含了一些 0 和 1的非空二维数组 grid , 一个 岛屿 是由四个方向 (水平或垂直) 的 1 (代表土地) 构成的组合。你可以假设二维矩阵的四个边缘都被水包围着。找到给定的二维数组中最大的岛屿面积。(如果没有岛屿,则返回面积为0。)

示例 1:

[[0,0,1,0,0,0,0,1,0,0,0,0,0],
[0,0,0,0,0,0,0,1,1,1,0,0,0],
[0,1,1,0,1,0,0,0,0,0,0,0,0],
[0,1,0,0,1,1,0,0,1,0,1,0,0],
[0,1,0,0,1,1,0,0,1,1,1,0,0],
[0,0,0,0,0,0,0,0,0,0,1,0,0],
[0,0,0,0,0,0,0,1,1,1,0,0,0],
[0,0,0,0,0,0,0,1,1,0,0,0,0]]

对于上面这个给定矩阵应返回 6。注意答案不应该是11,因为岛屿只能包含水平或垂直的四个方向的‘1’。

// 思路:分别从(i,j)出发寻找最大的岛屿
int maxAreaOfIsland(vector<vector<int>>& grid) {
    if(grid.empty() || grid[0].empty())
        return 0;
    int maxArea = 0;

    for(int i = 0; i < grid.size(); ++i)
        for(int j = 0; j < grid[0].size(); ++j)
            maxArea = max(maxArea, dfs(grid, i, j));

    return maxArea;
}
int dfs(vector<vector<int>> &grid, int i, int j){
    // 超出边界,或(i,j)是水域,直接返回0
    if(i < 0 || i > grid.size() - 1 || j < 0 || j > grid[0].size() - 1 || grid[i][j] == 0)
        return 0;    
    // 将该岛屿改成水域,同时向东南西北四个方向搜索存在的陆地
    grid[i][j] = 0;
    return 1 + dfs(grid, i - 1, j) + dfs(grid, i + 1, j) + dfs(grid, i, j - 1) + dfs(grid, i, j + 1);
}


leetcode.547 朋友圈 medium

班上有 N 名学生。其中有些人是朋友,有些则不是。他们的友谊具有是传递性。如果已知 A 是 B 的朋友,B 是 C 的朋友,那么我们可以认为 A 也是 C 的朋友。所谓的朋友圈,是指所有朋友的集合。

给定一个 N * N 的矩阵 M,表示班级中学生之间的朋友关系。如果M[i][j] = 1,表示已知第 i 个和 j 个学生互为朋友关系,否则为不知道。你必须输出所有学生中的已知的朋友圈总数。

示例 1:

输入: 
[[1,1,0],
[1,1,0],
[0,0,1]]
输出: 2 
说明:已知学生0和学生1互为朋友,他们在一个朋友圈。
第2个学生自己在一个朋友圈。所以返回2。

输入: 
[[1,1,0],
[1,1,1],
[0,1,1]]
输出: 1
说明:已知学生0和学生1互为朋友,学生1和学生2互为朋友,所以学生0和学生2也是朋友,所以他们三个在一个朋友圈,返回1。
// 思路:类似leetcode.200岛屿数量
// 此题中,M(i,j)=1,朋友圈+1,同时删除(i,j)的所有朋友关系
// 注意:朋友关系具有传递性,需要删除 i 行和 j 列上的所有关系。不同于leetcode.200中向上下左右四个方向删陆地
int findCircleNum(vector<vector<int>>& M) {
    if(M.empty() || M[0].empty())
        return 0;

    int circles = 0;
    for(int i = 0; i < M.size(); ++i)
        for(int j = 0; j < M[0].size(); ++j)
            if(M[i][j] == 1)
                ++circles, dfs(M, i, j);
    return circles;
}
void dfs(vector<vector<int>> &M, int i, int j){
    // 超出搜索范围或者没有关系,直接返回
    if(i < 0 || i > M.size() - 1 || j < 0 || j > M[0].size() - 1 || M[i][j] == 0)
        return;
    // 删除所有朋友关系
    M[i][j] = 0;        
    for(int row = 0; row < M.size(); ++row)
        dfs(M, row, j);
    for(int col = 0; col < M[0].size(); ++col)
        dfs(M, i, col);
}


leetcode.130 被围绕的区域 medium

给定一个二维的矩阵,包含 'X''O'字母 O)。找到所有被 'X' 围绕的区域,并将这些区域里所有的 'O''X' 填充。

示例:

输入:
X X X X
X O O X
X X O X
X O X X
输出:
X X X X
X X X X
X X X X
X O X X
解释:
被围绕的区间不会存在于边界上,换句话说,任何边界上的 'O' 都不会被填充为 'X'。 任何不在边界上,或不与边界上的 'O' 相连的 'O' 最终都会被填充为 'X'。如果两个元素在水平或垂直方向相邻,则称它们是“相连”的。
// 思路:反向思维,先从边界开始进行dfs,将与边界接壤的区域该为 *
// 然后遍历所有元素,将O该为X,将*改为O
void solve(vector<vector<char>>& board) {
    if(board.empty() || board[0].empty())
        return;
    // 左右边界
    for(int row = 0; row < board.size(); ++row){
        dfs(board, row, 0);
        dfs(board, row, board[0].size() - 1);
    }
    // 上下边界
    for(int col = 0; col < board[0].size(); ++col){
        dfs(board, 0, col);
        dfs(board, board.size() - 1, col);
    }
    for(auto &rows:board)
        for(auto &ele:rows)
            ele = ele == 'O' ? 'X' : ele == '*' ? 'O' : ele;
}
void dfs(vector<vector<char>> &board, int i, int j){
    if(i < 0 || i > board.size() - 1 || j < 0 || j > board[0].size() - 1 || board[i][j] != 'O')
        return;	// 超出边界或者不是'O',直接返回

    board[i][j] = '*';
    dfs(board, i - 1, j);
    dfs(board, i + 1, j);
    dfs(board, i, j - 1);
    dfs(board, i, j + 1);
}

leetcode.417 太平洋大西洋水流问题 medium

给定一个 m x n 的非负整数矩阵来表示一片大陆上各个单元格的高度。“太平洋”处于大陆的左边界和上边界,而“大西洋”处于大陆的右边界和下边界。规定水流只能按照上、下、左、右四个方向流动,且只能从高到低或者在同等高度上流动。请找出那些水流既可以流动到“太平洋”,又能流动到“大西洋”的陆地单元的坐标。

示例:

输入:
太平洋 ~   ~   ~   ~   ~ 
~  1   2   2   3  (5) *
~  3   2   3  (4) (4) *
~  2   4  (5)  3   1  *
~ (6) (7)  1   4   5  *
~ (5)  1   1   2   4  *
				- - - - - 大西洋
返回:
[[0, 4], [1, 3], [1, 4], [2, 2], [3, 0], [3, 1], [4, 0]] (上图中带括号的单元)
// 思路:太平洋的水低到高流,能流到的地图为m1,大西洋的水从低到高流,能流到的地图为m2。则m1和m2的重合部分即为所求
vector<vector<int>> pacificAtlantic(vector<vector<int>>& matrix) {
    if(matrix.empty() || matrix[0].empty())
        return {};
    vector<vector<bool>> pac(matrix.size(), vector<bool>(matrix[0].size(), false));
    vector<vector<bool>> atl(matrix.size(), vector<bool>(matrix[0].size(), false));

    // 从太平洋沿岸出发,即左列和上行
    for(int i = 0; i < matrix.size(); ++i)
        dfs(matrix, i, 0, pac);
    for(int j = 0; j < matrix[0].size(); ++j)
        dfs(matrix, 0, j, pac);
    // 从大西洋沿岸出发,即右列和下行
    for(int i = 0; i < matrix.size(); ++i)
        dfs(matrix, i, matrix[0].size() - 1, atl);
    for(int j = 0; j < matrix[0].size(); ++j)
        dfs(matrix, matrix.size() - 1, j, atl);

    vector<vector<int>> res;
    for(int i = 0; i < matrix.size(); ++i)
        for(int j = 0; j < matrix[0].size(); ++j)
            if(pac[i][j] == true && atl[i][j] == true)
                res.push_back({i,j});
    return res;
}
void dfs(const vector<vector<int>> &matrix, int i, int j, vector<vector<bool>> &map){
    if(map[i][j] == true)
        return;		// 到达曾到达过的区域则直接返回,防止两个点高度相同时陷入死循环
    map[i][j] = true;
    if(i > 0 && matrix[i - 1][j] >= matrix[i][j])
        dfs(matrix, i - 1, j, map);
    if(i < matrix.size() - 1 && matrix[i + 1][j] >= matrix[i][j])
        dfs(matrix, i + 1, j, map);
    if(j > 0 && matrix[i][j - 1] >= matrix[i][j])
        dfs(matrix, i, j - 1, map);
    if(j < matrix[0].size() - 1 && matrix[i][j + 1] >= matrix[i][j])
        dfs(matrix, i, j + 1, map);
}


leetcode.529 扫雷游戏 medium

让我们一起来玩扫雷游戏!给定一个代表游戏板的二维字符矩阵。 'M' 代表一个未挖出的地雷,'E' 代表一个未挖出的空方块,'B' 代表没有相邻(上,下,左,右,和所有4个对角线)地雷的已挖出的空白方块,数字('1' 到 '8')表示有多少地雷与这块已挖出的方块相邻,'X' 则表示一个已挖出的地雷。

现在给出在所有未挖出的方块中('M'或者'E')的下一个点击位置(行和列索引),根据以下规则,返回相应位置被点击后对应的面板:

  1. 如果一个地雷('M')被挖出,游戏就结束了- 把它改为 'X'。
  2. 如果一个没有相邻地雷的空方块('E')被挖出,修改它为('B'),并且所有和其相邻的方块都应该被递归地揭露。
  3. 如果一个至少与一个地雷相邻的空方块('E')被挖出,修改它为数字('1'到'8'),表示相邻地雷的数量。
  4. 如果在此次点击中,若无更多方块可被揭露,则返回面板。

示例 :

输入: 									输入:
[['E', 'E', 'E', 'E', 'E'],					[['B', '1', 'E', '1', 'B'],
['E', 'E', 'M', 'E', 'E'],					['B', '1', 'M', '1', 'B'],
['E', 'E', 'E', 'E', 'E'],					['B', '1', '1', '1', 'B'],
['E', 'E', 'E', 'E', 'E']]					['B', 'B', 'B', 'B', 'B']]
Click : [3,0]								Click : [1,2]
输出: 									输出
[['B', '1', 'E', '1', 'B'],					[['B', '1', 'E', '1', 'B'],
['B', '1', 'M', '1', 'B'],					['B', '1', 'X', '1', 'B'],
['B', '1', '1', '1', 'B'],					['B', '1', '1', '1', 'B'],
['B', 'B', 'B', 'B', 'B']]					['B', 'B', 'B', 'B', 'B']]
vector<vector<int>> dirs = { {-1,0},{1,0},{0,-1},{0,1},{-1,-1},{-1,1},{1,-1},{1,1} };
vector<vector<char>> updateBoard(vector<vector<char>>& board, vector<int>& click) {       
    if(board[click[0]][click[1]] == 'M'){
        board[click[0]][click[1]] = 'X';
        return board;
    }
    bfs(board, click[0], click[1]);
    return board;
}
void bfs(vector<vector<char>> &board, int row, int col){
    // 搜索board[row][col]八个方向的炸弹数量
    int bombs = 0;
    for(auto &dir:dirs){
        int new_row = row + dir[0], new_col = col + dir[1];
        if(new_row < 0 || new_col < 0 
           || new_row >= board.size() || new_col >= board[0].size())
            continue;
        bombs += board[new_row][new_col] == 'M' ? 1 : 0;
    }
    board[row][col] = bombs == 0 ? 'B' : char(48 + bombs);

    // 附近没有雷,才对八个方向进行搜索
    if(board[row][col] == 'B')
        for(auto &dir:dirs){
            int new_row = row + dir[0], new_col = col + dir[1];
            if(new_row < 0 || new_col < 0 
               || new_row >= board.size() || new_col >= board[0].size())
                continue;
            if(board[new_row][new_col] == 'E')  // 该方向是空格才搜索
                bfs(board, new_row, new_col);
        }
}


leetcode.257 二叉树的所有路径 easy

给定一个二叉树,返回所有从根节点到叶子节点的路径。说明: 叶子节点是指没有子节点的节点。

示例:

输入:
   1
 /   \
2     3
 \
  5
输出: ["1->2->5", "1->3"]

解释: 所有根节点到叶子节点的路径为: 1->2->5, 1->3
vector<string> binaryTreePaths(TreeNode* root) {
    if(root == nullptr)
        return {};

    vector<string> res;
    dfs(root, "", res);
    return res;
}
void dfs(TreeNode* root, string prefix, vector<string> &res){
    if(root->left == nullptr && root->right == nullptr){	// 到达叶节点
        res.push_back(prefix + to_string(root->val));
        return;
    }
    if(root->left != nullptr)
        dfs(root->left, prefix + to_string(root->val) + "->", res);
    if(root->right != nullptr)
        dfs(root->right, prefix + to_string(root->val) + "->", res);
}

leetcode.129 求根到叶子节点数字之和 medium

给定一个二叉树,它的每个结点都存放一个 0-9 的数字,每条从根到叶子节点的路径都代表一个数字。例如,从根到叶子节点路径 1->2->3 代表数字 123。计算从根到叶子节点生成的所有数字之和。说明: 叶子节点是指没有子节点的节点。

示例 :

输入: [1,2,3]
  1
 / \
2   3
输出: 25
解释:
从根到叶子节点路径 1->2 代表数字 12.从根到叶子节点路径 1->3 代表数字 13.
因此,数字总和 = 12 + 13 = 25.

输入: [4,9,0,5,1]
    4
   / \
  9   0
 / \
5   1
输出: 1026
解释:
从根到叶子节点路径 4->9->5 代表数字 495.从根到叶子节点路径 4->9->1 代表数字 491.从根到叶子节点路径 4->0 代表数字 40.
因此,数字总和 = 495 + 491 + 40 = 1026.
int sumNumbers(TreeNode* root) {
    if(root == nullptr)
        return 0;
    long long res = 0;
    if(dfs(root, 0, res))	
        return INT_MAX;
    return res;
}
// 返回true表示已超过了INT_MAX,停止dfs
bool dfs(TreeNode *root, long long prefix, long long &res){
    if(root->left == nullptr && root->right == nullptr){
        res += prefix * 10 + root->val;
        return res > INT_MAX ? true : false;
    }
    prefix = prefix * 10 + root->val;
    if(prefix > INT_MAX)
        return true;
    if(root->left != nullptr && dfs(root->left, prefix, res))
        return true;
    if(root->right != nullptr && dfs(root->right, prefix, res))
        return true;
    return false;
}

leetcode.199 二叉树的右视图 medium

给定一棵二叉树,想象自己站在它的右侧,按照从顶部到底部的顺序,返回从右侧所能看到的节点值。

示例:

输入: [1,2,3,null,5,null,4]
输出: [1, 3, 4]
解释:
   1            <---
 /   \
2     3         <---
 \     \
  5     4       <---

// 思路:DFS-先序遍历(右侧开始),某层最先达到的节点即为该层最右侧的节点
vector<int> rightSideView(TreeNode* root) {
    if(root == nullptr)
        return {};

    vector<int> res;
    preorder(root, 0, res);        
    return res;
}
void preorder(TreeNode *root, int level, vector<int> &nums){
    if(root == nullptr)
        return;
    if(nums.size() == level)
        nums.push_back(root->val);
    preorder(root->right, level + 1, nums);
    preorder(root->left, level + 1, nums);
}
// 此题也可使用BFS做,即层次遍历,每层最后一个节点放入vector中即可
// 注:树的前、中、后序遍历均是深度优先搜索


leetcode.124 二叉树中的最大路径和 hard

给定一个非空二叉树,返回其最大路径和。本题中,路径被定义为一条从树中任意节点出发,达到任意节点的序列。该路径至少包含一个节点,且不一定经过根节点。

示例:

输入: [1,2,3]
  1
 / \
2   3
输出: 6

输入: [-10,9,20,null,null,15,7]
   -10
   / \
  9  20
 /  \
15   7
输出: 42
int maxPathSum(TreeNode* root) {
    int maxSum = INT_MIN;
    dfs(root, maxSum);
    return maxSum;
}
int dfs(TreeNode *root, int &maxSum){
    if(root == nullptr)
        return 0;
    // 求左、右子树的最大单边路径和,若为负数则直接置0,刨除负数节点
    int leftSum = max(dfs(root->left, maxSum), 0);
    int rightSum = max(dfs(root->right, maxSum), 0);
    
    // 更新最大路径和, 必然为(左+根+右, maxSum)中的较大值
    maxSum = max(maxSum, leftSum + rightSum + root->val);

    // 返回左、右子树路径和较大值 + 本节点的值,作为以root为根往下的最大单边路径和
    return max(leftSum, rightSum) + root->val;
}


剑指offer 机器人的运动范围

地上有一个m行和n列的方格。一个机器人从坐标0,0的格子开始移动,每一次只能向左,右,上,下四个方向移动一格,但是不能进入行坐标和列坐标的数位之和大于k的格子。 例如,当k为18时,机器人能够进入方格(35,37),因为3+5+3+7 = 18。但是,它不能进入方格(35,38),因为3+5+3+8 = 19。请问该机器人能够达到多少个格子?

int movingCount(int threshold, int rows, int cols){
    vector<vector<bool>> grid(rows, vector<bool>(cols, false));
    dfs(grid, 0, 0, threshold);	// 只需从(0,0)出发开始dfs
    
    int sum = 0;
    for(const auto &row:grid)
        sum += accumulate(row.begin(), row.end(), 0);
    return threshold < 0 ? 0 : sum;
}
void dfs(vector<vector<bool>> &grid, int i, int j, const int &thresh){
    if(i < 0 || i > grid.size() - 1 || j < 0 || j > grid[0].size() - 1 
       || grid[i][j] == true || decbitsum(i, j) > thresh)
        return;
    grid[i][j] = true;
    dfs(grid, i - 1, j, thresh);
    dfs(grid, i + 1, j, thresh);
    dfs(grid, i, j - 1, thresh);
    dfs(grid, i, j + 1, thresh);
}
int decbitsum(int i, int j){
    int sum = 0;
    while(i != 0){
        sum += i % 10;
        i /= 10;
    }
    while(j != 0){
        sum += j % 10;
        j /= 10;
    }
    return sum;
}

回溯法backtracking

回溯法也是DFS,不同之处在于:DFS用于遍历节点,回溯法用于在遍历的过程中得到遍历结果

leetcode.17 电话号码的字母组合 medium

给定一个仅包含数字 2-9 的字符串,返回所有它能表示的字母组合。给出数字到字母的映射如下(与电话按键相同)。注意 1 不对应任何字母

示例:

输入:"23"
输出:["ad", "ae", "af", "bd", "be", "bf", "cd", "ce", "cf"].
vector<string> letterCombinations(string digits) {
    if(digits.empty())
        return {};
    string s;   // 用于保存在dfs过程中的遍历记录
    vector<string> res;
    backtracking(digits, 0, s, res);
    return res;
}

vector<string> phones = {"0", "1", "abc", "def", "ghi", "jkl", "mno", "pqrs", "tuv", "wxyz"};
void backtracking(const string &digits, int idx, string &prefix, vector<string> &res){
    if(idx == digits.size()){
        res.push_back(prefix);
        return;
    }
    for(const auto &c:phones[digits[idx] - '0']){
        prefix.push_back(c);
        backtracking(digits, idx + 1, prefix, res);
        prefix.pop_back();      // 进行下次遍历前,将此次的影响消除
    }
}

leetcode.79 单词搜索 medium

给定一个二维网格和一个单词,找出该单词是否存在于网格中。

单词必须按照字母顺序,通过相邻的单元格内的字母构成,其中“相邻”单元格是那些水平相邻或垂直相邻的单元格。同一个单元格内的字母不允许被重复使用。

示例:

board =
[
['A','B','C','E'],
['S','F','C','S'],
['A','D','E','E']
]
给定 word = "ABCCED", 返回 true.
给定 word = "SEE", 返回 true.
给定 word = "ABCB", 返回 false.
bool exist(vector<vector<char>>& board, string word) {
    if(word.empty())
        return true;
    if(board.empty() || board[0].empty())
        return false;

    for(int i = 0; i < board.size(); ++i)
        for(int j = 0; j < board[0].size(); ++j)
            if(backtracking(board, word, 0, i, j))
                return true;
    return false;
}

bool backtracking(vector<vector<char>> &board, const string &word, int idx, int i, int j){
    // 超出搜索边界,或者(i,j)处的字母与word[idx]不等,直接返回
    if(i < 0 || i > board.size() - 1 || j < 0 || j > board[0].size() - 1 
       || idx > word.size() - 1 || board[i][j] != word[idx])
        return false;
    // 搜索到了,返回true
    if(idx == word.size() - 1 && board[i][j] == word[idx])
        return true;

    char c_backup = board[i][j];
    board[i][j] = '*';  // 已访问标记,防止再次访问

    // 向上下左右四个方向搜索
    if(backtracking(board, word, idx + 1, i + 1, j) 
       || backtracking(board, word, idx + 1, i - 1, j) 
       || backtracking(board, word, idx + 1, i, j + 1) 
       || backtracking(board, word, idx + 1, i, j - 1))
        return true;

    board[i][j] = c_backup; // 未搜索到,需要恢复(i,j)处的字母
    return false;
}

leetcode.93 复原IP地址 medium

给定一个只包含数字的字符串,复原它并返回所有可能的 IP 地址格式。

示例:

输入: "25525511135"
输出: ["255.255.11.135", "255.255.111.35"]
vector<string> restoreIpAddresses(string s) {
    if(s.size() <= 3)
        return {};
    vector<string> res;
    backtracking(res, s, "", 0);
    return res;
}
void backtracking(vector<string> &res, string s, string prefix, int cnt){
    if(cnt == 3){   // 已经经历了3次分割,只需进行最后一次分割
        if(isValid(s))
            res.push_back(prefix + s);
        return;
    }
    for(int i = 1; i <= 3; ++i)	// 分割出1-3位判断有效性
        if(i <= s.size() && isValid(s.substr(0, i)))
            backtracking(res, s.substr(i), prefix + s.substr(0, i) + ".", cnt + 1);
}
bool isValid(string s){
    if(s.empty() || s.size() > 3)
        return false;
    if(s[0] == '0')	// 首位为0,则必然只有一位
        return s.size() == 1;	
    else	       // 只需数值<=255即可
        return stoi(s) <= 255;
}

leetcode.46 全排列 medium剑指offer 字符串的排列

定一个没有重复数字的序列,返回其所有可能的全排列。

示例:

输入: [1,2,3]
输出:
[
[1,2,3],
[1,3,2],
[2,1,3],
[2,3,1],
[3,1,2],
[3,2,1]
]

vector<vector<int>> permute(vector<int>& nums) {
    vector<vector<int>> res;
    permute(res, nums, 0);
    return res;
}
void permute(vector<vector<int>> &res, vector<int> &nums, int idx){
    if(idx == nums.size() - 1){
        res.push_back(nums);
        return;
    }
    for(int i = idx; i < nums.size(); ++i){
        swap(nums[i], nums[idx]);
        permute(res, nums, idx + 1);
        swap(nums[i], nums[idx]);
    }
}

leetcode.47 全排列II medium

给定一个可包含重复数字的序列,返回所有不重复的全排列。

示例:

输入: [1,1,2]
输出:
[
[1,1,2],
[1,2,1],
[2,1,1]
]
// 只需将leetcode.46中的vector<vector<int>> res  ===>>>  set<vector<int>> res;
vector<vector<int>> permuteUnique(vector<int>& nums) {
    set<vector<int>> res;		// 改动
    permute(res, nums, 0);
    return vector<vector<int>> (res.begin(), res.end());	// 改动
}
void permute(set<vector<int>> &res, vector<int> &nums, int idx){
    if(idx == nums.size() - 1){
        res.insert(nums);	   // 改动
        return;
    }
    for(int i = idx; i < nums.size(); ++i){
        swap(nums[i], nums[idx]);
        permute(res, nums, idx + 1);
        swap(nums[i], nums[idx]);
    }
}

leetcode.77 组合 medium

给定两个整数 n 和 k,返回 1 ... n 中所有可能的 k 个数的组合。

示例:

输入: n = 4, k = 2
输出:[
[2,4],
[3,4],
[2,3],
[1,2],
[1,3],
[1,4],
]
vector<vector<int>> combine(int n, int k) {
    vector<vector<int>> res;
    vector<int> nums;
    backtracking(n, k, res, nums, 0);
    return res;
}

void backtracking(const int &n, const int &k, 
                  vector<vector<int>> &res, vector<int> &nums, int last_int){
    if(nums.size() == k){
        res.push_back(nums);
        return;
    }
    // 从上一次放入nums的数的后一位开始放
    for(int i = last_int + 1; i <= n; ++i){
        nums.push_back(i);
        backtracking(n, k, res, nums, i);
        nums.pop_back();
    }
}

leetcode.39 组合之和 medium

给定一个无重复元素的数组 candidates 和一个目标数 target ,找出 candidates 中所有可以使数字和为 target 的组合。candidates 中的数字可以无限制重复被选取。

说明:所有数字(包括 target)都是正整数。解集不能包含重复的组合。 示例 :

输入: candidates = [2,3,6,7], target = 7,
所求解集为:
[
[7],
[2,2,3]
]

输入: candidates = [2,3,5], target = 8,
所求解集为:
[
[2,2,2,2],
[2,3,3],
[3,5]
]
vector<vector<int>> combinationSum(vector<int>& candidates, int target) {
    sort(candidates.begin(), candidates.end());
    set<vector<int>> res;
    vector<int> nums;
    backtracking(candidates, target, res, nums, 0);
    return vector<vector<int>> (res.begin(), res.end());
}

void backtracking(const vector<int> &cdd, int t, set<vector<int>> &res, vector<int> &nums, int idx){
    if(t <= 0){ // 注意这里的条件
        if(t == 0)
            res.insert(nums);
        return;
    }
    for(int i = idx; i < cdd.size(); ++i){
        nums.push_back(cdd[i]);
        backtracking(cdd, t - cdd[i], res, nums, i);
        nums.pop_back();
    }
}

leetcode.40 组合之和II medium

给定一个数组 candidates 和一个目标数 target ,找出 candidates 中所有可以使数字和为 target 的组合。candidates 中的每个数字在每个组合中只能使用一次。

说明:所有数字(包括目标数)都是正整数。解集不能包含重复的组合。

示例 1:

输入: candidates = [10,1,2,7,6,1,5], target = 8,
所求解集为:
[
[1, 7],
[1, 2, 5],
[2, 6],
[1, 1, 6]
]

输入: candidates = [2,5,2,1,2], target = 5,
所求解集为:
[
[1,2,2],
[5]
]
// 在leetcode.39的基础上改动一个小地方
vector<vector<int>> combinationSum2(vector<int>& candidates, int target) {
    sort(candidates.begin(), candidates.end());
    set<vector<int>> res;
    vector<int> nums;
    backtracking(candidates, target, res, nums, 0);
    return vector<vector<int>> (res.begin(), res.end());
}
void backtracking(const vector<int> &cdd, int t, set<vector<int>> &res, vector<int> &nums, int idx){
    if(t <= 0){
        if(t == 0)
            res.insert(nums);
        return;
    }

    for(int i = idx; i < cdd.size(); ++i){
        nums.push_back(cdd[i]);
        backtracking(cdd, t - cdd[i], res, nums, i + 1);    // 改动:下一层遍历从i+1开始
        nums.pop_back();
    }
}

leetcode.78 子集 medium

给定一组不含重复元素的整数数组 nums,返回该数组所有可能的子集(幂集)。说明:解集不能包含重复的子集。

示例:

输入: nums = [1,2,3]
输出:
[ [3], [1], [2], [1,2,3], [1,3], [2,3], [1,2], [] ]
vector<vector<int>> subsets(vector<int>& nums) {
    set<vector<int>> res;
    vector<int> subset;
    backtracking(nums, res, subset, 0);
    return vector<vector<int>> (res.begin(), res.end());
}

void backtracking(const vector<int> &nums, set<vector<int>> &res, vector<int> &subset, int idx){
    res.insert(subset);                         // 不放入nums[i]的情况
    for(int i = idx; i < nums.size(); ++i){
        subset.push_back(nums[i]);
        backtracking(nums, res, subset, i + 1); // 放入nums[i]的情况
        subset.pop_back();
    }
}

leetcode.90 子集II medium

给定一个可能包含重复元素的整数数组 nums,返回该数组所有可能的子集(幂集)。说明:解集不能包含重复的子集。

示例:

输入: [1,2,2]
输出:
[
[2],
[1],
[1,2,2],
[2,2],
[1,2],
[]
]
// 与leetcode.78基本一模一样,唯一改动的是需要进行排序
// 因为 [1,2,2]和[2,1,2]是同一子集,却是不同的vector,在res.insert(subset)时会被重复插入
vector<vector<int>> subsetsWithDup(vector<int>& nums) {
    sort(nums.begin(), nums.end());		// 唯一改动的地方
    set<vector<int>> res;
    vector<int> subset;
    backtracking(nums, res, subset, 0);
    return vector<vector<int>> (res.begin(), res.end());
}

void backtracking(const vector<int> &nums, set<vector<int>> &res, vector<int> &subset, int idx){
    res.insert(subset);
    for(int i = idx; i < nums.size(); ++i){
        subset.push_back(nums[i]);
        backtracking(nums, res, subset, i + 1);
        subset.pop_back();
    }
}

leetcode.491 递增子序列 medium

给定一个整型数组, 你的任务是找到所有该数组的递增子序列,递增子序列的长度至少是2。

示例:

输入: [4, 6, 7, 7]
输出: [[4, 6], [4, 7], [4, 6, 7], [4, 6, 7, 7], [6, 7], [6, 7, 7], [7,7], [4,7,7]]

说明: 给定数组的长度不会超过15。数组中的整数范围是 [-100,100]。给定数组中可能包含重复数字,相等的数字应该被视为递增的一种情况。

// 思路:与leetcode.90子集II类似,只需要改动两个地方:1.保证子集尺寸>=2才放入res, 2. 保证子集是递增子序列
vector<vector<int>> findSubsequences(vector<int>& nums) {
    set<vector<int>> res;
    vector<int> subset;
    backtracking(nums, res, subset, 0);
    return vector<vector<int>> (res.begin(), res.end());
}

void backtracking(const vector<int> &nums, set<vector<int>> &res, vector<int> &subset, int idx){
    if(subset.size() >= 2)		// 改动1. 保证子集尺寸>=2才放入res
        res.insert(subset);
    for(int i = idx; i < nums.size(); ++i)
        if(subset.empty() || nums[i] >= subset.back()){		// 改动2. 保证子集是递增子序列
            subset.push_back(nums[i]);
            backtracking(nums, res, subset, i + 1);
            subset.pop_back();
        }
}

leetcode.131 分割回文串 medium

给定一个字符串 s,将 s 分割成一些子串,使每个子串都是回文串。返回 s 所有可能的分割方案。

示例:

输入: "aab"
输出:
[
["aa","b"],
["a","a","b"]
]
vector<vector<string>> partition(string s) {
    vector<string> strs;
    vector<vector<string>> res;
    backtracking(res, strs, s, 0);
    return res;
}

void backtracking(vector<vector<string>> &res, vector<string> &strs, const string &s, int idx){
    if(idx == s.size()){
        res.push_back(strs);
        return;
    }
    for(int i = idx; i < s.size(); ++i)
        if(isPalindrome(s, idx, i)){
            strs.push_back(s.substr(idx, i - idx + 1));
            backtracking(res, strs, s, i + 1);
            strs.pop_back();
        }
}
bool isPalindrome(const string &s, int left, int right){
    while(left <= right)
        if(s[left++] != s[right--])
            return false;
    return true;
}

leetcode.37 数独 hard

编写一个程序,通过已填充的空格来解决数独问题。一个数独的解法需遵循如下规则:

数字 1-9 在每一行只能出现一次。 数字 1-9 在每一列只能出现一次。 数字 1-9 在每一个以粗实线分隔的 3x3 宫内只能出现一次。 空白格用 '.' 表示。

Note:

  • 给定的数独序列只包含数字 1-9 和字符 '.'
  • 你可以假设给定的数独只有唯一解。
  • 给定数独永远是 9x9 形式的。

一个典型的数独(答案为红色):

void solveSudoku(vector<vector<char>>& board) {
    // rowc[i][c]表示第i行存在多少个字符c, colc[j][c]表示第j行字符c数量
    // subc[k][c]表示第k个3x3子区域存在多少个字符c
    vector<map<char,int>> rowc(9, map<char,int>());
    vector<map<char,int>> colc(9, map<char,int>());
    vector<map<char,int>> subc(9, map<char,int>());

    // 统计board中每行每列各个字符的数量
    for(int i = 0; i < 9; ++i)
        for(int j = 0; j < 9; ++j)
            if(board[i][j] != '.'){
                int k = i / 3 * 3 + j / 3;  // (i,j)与第k个3x3子区域的关系
                char c = board[i][j];
                ++rowc[i][c];
                ++colc[j][c];
                ++subc[k][c];
            }
    // 直接从(0,0)开始求解
    backtracking(board, 0, 0, rowc, colc, subc);
}

// 从左到右、从上到下求解
bool backtracking(vector<vector<char>> &board, int i, int j,
                  vector<map<char,int>> &rowc,
                  vector<map<char,int>> &colc,
                  vector<map<char,int>> &subc){
    // 结束条件:已经求解到最后一行的下一行,说明前面的9x9已经求解完毕
    if(i == 9 && j == 0)
        return true;
    // (i,j)已求解,对下一格求解
    if(board[i][j] != '.')
        if( (j != 8 && backtracking(board, i, j + 1, rowc, colc, subc))
           ||(j == 8 && backtracking(board, i + 1, 0, rowc, colc, subc)))
            return true;
    // (i,j)未求解,对本格求解
    if(board[i][j] == '.')
        for(char c = '1'; c <= '9'; ++c)
            if(rowc[i][c] < 1 && colc[j][c] < 1 && subc[i/3*3+j/3][c] < 1){
                ++rowc[i][c];
                ++colc[j][c];
                ++subc[i/3*3+j/3][c];
                board[i][j] = c;
                if( (j != 8 && backtracking(board, i, j + 1, rowc, colc, subc))
                   ||(j == 8 && backtracking(board, i + 1, 0, rowc, colc, subc)))
                    return true;
                --rowc[i][c];
                --colc[j][c];
                --subc[i/3*3+j/3][c];
                board[i][j] = '.';
            }
    return false;
}

leetcode.698 划分为k个相等的子集

给定一个整数数组 nums 和一个正整数 k,找出是否有可能把这个数组分成 k 个非空子集,其总和都相等。

示例 :

输入: nums = [4, 3, 2, 3, 5, 2, 1], k = 4
输出: True
说明: 有可能将其分成 4 个子集(5),(1,4),(2,3),(2,3)等于总和。

注意:

1 <= k <= len(nums) <= 16 0 < nums[i] < 10000

bool canPartitionKSubsets(vector<int>& nums, int k) {
    int sum = 0, maxnum = INT_MIN;
    for(const auto &num:nums)
        sum += num, maxnum = max(maxnum, num);
    if(nums.size() < k || sum % k != 0 || maxnum > sum / k)
        return false;
    vector<bool> used(nums.size(), false);
    return backtracking(nums, used, k, sum / k, 0, 0);
}

bool backtracking(vector<int> &nums, vector<bool> &used, int k, int tarSum, int curSum, int idx){
    if(k == 0)           // 已经搜索到了k个子集
        return true;
    if(curSum == tarSum) // 搜索下一个子集
        return backtracking(nums, used, k - 1, tarSum, 0, 0);

    for(int i = idx; i < nums.size(); ++i)  // 这里如果每次都从0开始搜索则会超时, 从idx开始
        if(used[i] == false && curSum + nums[i] <= tarSum){
            used[i] = true;
            if(backtracking(nums, used, k, tarSum, curSum + nums[i], i + 1))
                return true;
            used[i] = false;
        }
    return false;
}

分治策略

leetcode.241 为运算表达式设计优先级 medium

给定一个含有数字和运算符的字符串,为表达式添加括号,改变其运算优先级以求出不同的结果。你需要给出所有可能的组合的结果。有效的运算符号包含 +, - 以及 * 。

示例 1:

输入: "2-1-1"
输出: [0, 2]
解释: 
((2-1)-1) = 0 
(2-(1-1)) = 2

输入: "2*3-4*5"
输出: [-34, -14, -10, -10, 10]
解释: 
(2*(3-(4*5))) = -34 
((2*3)-(4*5)) = -14 
((2*(3-4))*5) = -10 
(2*((3-4)*5)) = -10 
(((2*3)-4)*5) = 10
// 分而治之的思想
vector<int> diffWaysToCompute(string s) {
    vector<int> res;

    for(int i = 0; i < s.size(); ++i)
        if(s[i] == '+' || s[i] == '-' || s[i] == '*')	// 以s[i]为节点,分为左、右子串进行处理
            for(auto intl:diffWaysToCompute(s.substr(0, i)))
                for(auto intr:diffWaysToCompute(s.substr(i + 1))){
                    if(s[i] == '+')
                        res.push_back(intl + intr);
                    else if(s[i] == '-')
                        res.push_back(intl - intr);
                    else if(s[i] == '*')
                        res.push_back(intl * intr);
                }

    // res为空,说明s中不存在 +/-/* 运算符,直接将其转换为整数即可
    if(res.empty())
        res.push_back(stoi(s));

    return res;
}

leetcode. 282 给表达式添加运算符 hard

给定一个仅包含数字 0-9 的字符串和一个目标值,在数字之间添加二元运算符(不是一元)+、- 或 * ,返回所有能够得到目标值的表达式。

示例 :

输入: num = "123", target = 6
输出: ["1+2+3", "1*2*3"] 

输入: num = "232", target = 8
输出: ["2*3+2", "2+3*2"]

输入: num = "105", target = 5
输出: ["1*0+5","10-5"]

输入: num = "00", target = 0
输出: ["0+0", "0-0", "0*0"]

输入: num = "3456237490", target = 9191
输出: []
vector<string> addOperators(string num, int target) {
    vector<string> res;
    div_conq(num, target, 0, 0, "", res);
    return res;
}
void div_conq(string num, int target, long last, long cur, string out, vector<string>& res) {
    if(num.size() == 0 && cur == target){
        res.push_back(out); return;
    }

    // 分成左右两半进行操作
    for(int i = 1; i <= num.size(); ++i){
        string left = num.substr(0, i);
        string right = num.substr(i);
        if(left.size() > 1 && left[0] == '0') 
            return;

        // +-* 三种运算符分别进行操作
        if(out.size() > 0){
            div_conq(right, target, stoll(left), cur + stoll(left), out + "+" + left, res);
            div_conq(right, target, -stoll(left), cur - stoll(left), out + "-" + left, res);
            div_conq(right, target, last * stoll(left), cur - last + last * stoll(left), out + "*" + left, res);
            // 对于乘法2-3*5,上次操作中已经得到 cur = 2-3 = -1, last = -3
            // 因此此次操作应为 cur - last + last * 5 = -1 - (-3) + (-3) * 5 = 2 - 3 * 5
        }else 
            div_conq(right, target, stoll(left), stoll(left), left, res);
    }
}

动态规划

动态规划的思想是:将原问题拆分为多个子问题进行求解,并且保存了每个子问题的解

分割整数

leetcode. 343 整数拆分 medium

给定一个正整数 n,将其拆分为至少两个正整数的和,并使这些整数的乘积最大化。 返回你可以获得的最大乘积。

示例 :

输入: 2
输出: 1
解释: 2 = 1 + 1, 1 × 1 = 1。

输入: 10
输出: 36
解释: 10 = 3 + 3 + 4, 3 × 3 × 4 = 36。

说明: 你可以假设 n 不小于 2 且不大于 58。

int integerBreak(int n) {
    vector<int> dp(n + 1, 0);   // dp[i] = integerBreak(i)
    dp[1] = 1;                  // dp[1] = integerBreak(1) = 1
    // 将 i 拆分为 j, i - j两个数之和
    for(int i = 2; i <= n; ++i)
        for(int j = 1; j < i; ++j){
            dp[i] = max(dp[i], j * (i - j));
            dp[i] = max(dp[i], dp[j] * (i - j));
            dp[i] = max(dp[i], dp[j] * dp[i - j]);
        }
    return dp.back();
}

// 另解:拆分出尽量多的3即可
// 5 -> 3 * 2 = 6   8 -> 3 * 3 * 2 = 18		11 -> 3 * 3 * 3 * 2 = 54
// 6 -> 3 * 3 = 9   9 -> 3 * 3 * 3 = 27		12 -> 3 * 3 * 3 * 3 = 81
// 7 -> 3 * 4 = 12  10-> 3 * 3 * 4 = 36		13 -> 3 * 3 * 3 * 4 = 108
int integerBreak(int n){
    if(n <= 3)
        return n - 1;
    int res = 1;
    while(n > 4){
        res *= 3;
        n -= 3;
    }
    return res * n;
}

leetcode. 279 完全平方数 medium

给定正整数 n,找到若干个完全平方数(比如 1, 4, 9, 16, ...)使得它们的和等于 n。你需要让组成和的完全平方数的个数最少。

示例 1:

输入: n = 12
输出: 3 
解释: 12 = 4 + 4 + 4.

输入: n = 13
输出: 2
解释: 13 = 4 + 9.
int numSquares(int n) {
    vector<int> dp(n + 1, INT_MAX);
    dp[0] = 0;
    for(int i = 0; i <= n; ++i)
        for(int j = 0; i + j * j <= n; ++j)
            dp[i + j * j] = min(dp[i + j * j], dp[i] + 1);
    // 注意j的遍历方式,dp[i]可以保证了不为INT_MAX,进而避免INT_MAX+1
    return dp.back();
}

// 四平方和定理:任意数字必然能表示为四个数的平方和(可能有零)
// 1. 预处理n/4直到无法除尽
// 2. n % 8 == 7, 则n必然是4个数的平方和
// 3. 确认n是否能表示为两个数平方和(可能有零),以此确认是3个数还是2个数的平方和
int numSquares(int n){
    while(n % 4 == 0)
        n /= 4;
    if(n % 8 == 7)
        return 4;
    for(int a = 0; a * a <= n; ++a){
        int b = sqrt(n - a * a);
        if(b * b + a * a == n)
            return (a > 0) + (b > 0);
    }
    return 3;
}


leetcode. 91 解码方法 medium

一条包含字母 A-Z 的消息通过以下方式进行了编码:

'A' -> 1 'B' -> 2 ... 'Z' -> 26 给定一个只包含数字的非空字符串,请计算解码方法的总数。

示例 1:

输入: "12"
输出: 2
解释: 它可以解码为 "AB"(1 2)或者 "L"(12)。

输入: "226"
输出: 3
解释: 它可以解码为 "BZ" (2 26), "VF" (22 6), 或者 "BBF" (2 2 6) 
int numDecodings(string s) {
    vector<int>  dp(s.size() + 1, 0);   // dp[i]表示前i个字母可解码的数量
    dp[0] = 1;
    for(int i = 1; i < dp.size(); ++i){
        // "1|23|45" dp[1] = 1, "23"可解码 => dp[3] += dp[1] = 1
        if(i >= 2 && s[i - 2] != '0' && stoi(s.substr(i - 2, 2)) <= 26)
            dp[i] += dp[i - 2];            
        // "12|3|45" dp[2] = 3, "2"可解码 => dp[3] += dp[2] => 1 + 3 = 4 
        if(s[i - 1] != '0')
            dp[i] += dp[i - 1];
    }
    return dp.back();
}

矩阵路径

leetcode.62 不同路径 medium

一个机器人位于一个 m x n 网格的左上角 (起始点在下图中标记为“Start” )。机器人每次只能向下或者向右移动一步。机器人试图达到网格的右下角(在下图中标记为“Finish”)。问总共有多少条不同的路径?

说明:m 和 n 的值均不超过 100。

示例 :

输入: m = 3, n = 2
输出: 3
解释:
从左上角开始,总共有 3 条路径可以到达右下角。
向右 -> 向右 -> 向下
向右 -> 向下 -> 向右
向下 -> 向右 -> 向右

输入: m = 7, n = 3
输出: 28
// 第一行和第一列每个位置都只能有一种方法
// 除此之外每个位置都可以有两个方向(左侧or上侧)到达
int uniquePaths(int m, int n) {
    vector<vector<int>> dp(m, vector<int> (n, 1));
    for(int row = 1; row < m; ++row)
        for(int col = 1; col < n; ++col)
            dp[row][col] = dp[row - 1][col] + dp[row][col - 1];	// 左侧或者上侧来到[row][col]
    return dp.back().back();
}

// 优化空间,前m-1行是不必要的空间消耗
int uniquePaths(int m, int n){
    vector<int> dp(n, 1);
    for(int row = 1; row < m; ++row)
        for(int col = 1; col < n; ++col)
            dp[col] = dp[col - 1] + dp[col];
    return dp.back();
}

leetcode.63 不同路径 II medium

一个机器人位于一个 m x n 网格的左上角 (起始点在下图中标记为“Start” )。机器人每次只能向下或者向右移动一步。机器人试图达到网格的右下角(在下图中标记为“Finish”)。现在考虑网格中有障碍物。那么从左上角到右下角将会有多少条不同的路径?网格中的障碍物和空位置分别用 1 和 0 来表示。

说明:m 和 n 的值均不超过 100。

示例 :

输入:
[
[0,0,0],
[0,1,0],
[0,0,0]
]
输出: 2
解释:
3x3 网格的正中间有一个障碍物。
从左上角到右下角一共有 2 条不同的路径:
向右 -> 向右 -> 向下 -> 向下
向下 -> 向下 -> 向右 -> 向右
// 在leetcode.62的基础上加个判断
int uniquePathsWithObstacles(vector<vector<int>>& obs) {
    vector<long long> dp(obs[0].size(), 0);
    if(obs[0][0] == 1 || obs.back().back() == 1)    // 起点终点堵死就没必要继续了
        return 0;

    // 第一行
    dp[0] = 1;
    for(int col = 1; col < obs[0].size(); ++col)
        dp[col] = obs[0][col] == 1 ? 0 : dp[col - 1];

    // 其它行
    for(int row = 1; row < obs.size(); ++row){
        if(obs[row][0] == 1)
            dp[0] = 0;
        for(int col = 1; col < obs[0].size(); ++col)
            dp[col] = obs[row][col] == 1 ? 0 : dp[col] + dp[col - 1];
    }

    return dp.back() > INT_MAX ? INT_MAX : dp.back();
}

leeedocte.64 最小路径和 medium

给定一个包含非负整数的 m x n 网格,请找出一条从左上角到右下角的路径,使得路径上的数字总和为最小。

说明:每次只能向下或者向右移动一步。

示例:

输入:
[
[1,3,1],
[1,5,1],
[4,2,1]
]
输出: 7
解释: 因为路径 1→3→1→1→1 的总和最小。
// 与leetcode.62类似,除了第一行和第一列只有一个方向可达外,其余点均有两个方向可达
int minPathSum(vector<vector<int>>& grid) {
    if(grid.empty() || grid[0].empty())
        return 0;

    // 第一行只能左侧来,因此第一行的每一列都是左列数字加上本身的数字
    for(int col = 1; col < grid[0].size(); ++col)
        grid[0][col] += grid[0][col - 1];
    // 其他行
    for(int row = 1; row < grid.size(); ++row){
        grid[row][0] += grid[row - 1][0];               // 每行的第一列,必然从上一行来
        for(int col = 1; col < grid[0].size(); ++col)   // 其他列,两个方向来
            grid[row][col] += min(grid[row - 1][col], grid[row][col - 1]);
    }
    return grid.back().back();
}

斐波那契数列

leetcode.70 爬楼梯 easy

假设你正在爬楼梯。需要 n 阶你才能到达楼顶。每次你可以爬 1 或 2 个台阶。你有多少种不同的方法可以爬到楼顶呢?注意:给定 n 是一个正整数。

示例 :

输入: 2
输出: 2
解释: 有两种方法可以爬到楼顶。
1 阶 + 1 阶
2 阶

输入: 3
输出: 3
解释: 有三种方法可以爬到楼顶。
1 阶 + 1 阶 + 1 阶
1 阶 + 2 阶
2 阶 + 1 阶
// dp(n) - n阶楼梯的爬法数量
// dp(n) = dp(n - 1) + dp(n - 2) - 可以从n-2跨两步,或从n-1跨一步到达n
int climbStairs(int n) {
    vector<int> dp(n + 1, 0);   // dp[i] - i阶楼梯的爬法数量
    dp[0] = 1;
    dp[1] = 1;
    for(int i = 2; i < dp.size(); ++i)
        dp[i] = dp[i - 1] + dp[i - 2];
    return dp.back();
}

// 优化空间:通过llast,last替代dp[i - 2], dp[i - 1],写成函数迭代的形式
int climbStairs(int n){
    int llast = 0, last = 1;
    int cnt = 0;
    while(cnt++ < n){
      int cur = llast + last;
      llast = last;		// 注意,这两句话
      last = cur;			// 不能交换顺序
    }
    return last;
}

剑指 offer 变态跳台阶

一只青蛙一次可以跳上1级台阶,也可以跳上2级……它也可以跳上n级。求该青蛙跳上一个n级的台阶总共有多少种跳法。

// dp[n] = dp[n - 1] + dp[n - 2] + ... + dp[1]
// dp[n - 1] = dp[n - 2] + ... + dp[1]
// 状态转移方程: dp[n] = 2 * dp[n - 1] = 2^n*dp[0] = 2 ^ n
int jumpFloorII(int n) {
    if(n <= 0)
      	return 1;
    int res = 1, cnt = 1;
    while(cnt++ < n)
      	res *= 2;
    return res;
}

leetcode. 746 使用最小花费爬楼梯 easy

数组的每个索引做为一个阶梯,第 i个阶梯对应着一个非负数的体力花费值 costi

每当你爬上一个阶梯你都要花费对应的体力花费值,然后你可以选择继续爬一个阶梯或者爬两个阶梯。

您需要找到达到楼层顶部的最低花费。在开始时,你可以选择从索引为 0 或 1 的元素作为初始阶梯。

示例 :

输入: cost = [10, 15, 20]
输出: 15
解释: 最低花费是从cost[1]开始,然后走两步即可到阶梯顶,一共花费15。

输入: cost = [1, 100, 1, 1, 1, 100, 1, 1, 100, 1]
输出: 6
解释: 最低花费方式是从cost[0]开始,逐个经过那些1,跳过cost[3],一共花费6。

注意:cost 的长度将会在 [2, 1000]。每一个 cost[i] 将会是一个Integer类型,范围为 [0, 999]。

// 与leetcode.70爬楼梯类似
// 两种方法爬到i层,1. 从i-1层爬上来,花费dp[i-1]+cost[i-1]   2. 从i-2层爬上来,花费dp[i-2]+cost[i-2]
int minCostClimbingStairs(vector<int>& cost) {
    vector<int> dp(cost.size() + 1, 0);     // dp[i]爬到前i层时的最小花费
    for(int i = 2; i < dp.size(); ++i)
        dp[i] = min(dp[i - 1] + cost[i - 1], dp[i - 2] + cost[i - 2]);
    return dp.back();
}

leetcode. 198 打家劫舍 easy

你是一个专业的小偷,计划偷窃沿街的房屋。每间房内都藏有一定的现金,影响你偷窃的唯一制约因素就是相邻的房屋装有相互连通的防盗系统,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警。

给定一个代表每个房屋存放金额的非负整数数组,计算你在不触动警报装置的情况下,能够偷窃到的最高金额。

示例 :

输入: [1,2,3,1]
输出: 4
解释: 偷窃 1 号房屋 (金额 = 1) ,然后偷窃 3 号房屋 (金额 = 3)。
偷窃到的最高金额 = 1 + 3 = 4 。

输入: [2,7,9,3,1]
输出: 12
解释: 偷窃 1 号房屋 (金额 = 2), 偷窃 3 号房屋 (金额 = 9),接着偷窃 5 号房屋 (金额 = 1)。
偷窃到的最高金额 = 2 + 9 + 1 = 12 。
// dp[i] - 前i家能抢到的最大值,两种情况:抢第i家,或者不抢第i家
// dp[i] = max(dp[i - 1], dp[i - 2] + nums[i])
int rob(vector<int>& nums) {
    if(nums.size() <= 1)
        return nums.empty() ? 0 : nums[0];
    vector<int> dp(nums.size(), 0);
    dp[0] = nums[0];
    dp[1] = max(nums[0], nums[1]);
    for(int i = 2; i < nums.size(); ++i)
        dp[i] = max(dp[i - 1], dp[i - 2] + nums[i]);
    return dp.back();
}

// 优化空间,通过llast,last替代dp[i - 2], dp[i - 1]
int rob(vector<int> &nums){
    if(nums.size() <= 1)
        return nums.empty() ? 0 : nums[0];
    int llast = nums[0], last = max(nums[0], nums[1]);
    for(int i = 2; i < nums.size(); ++i){
        int cur = max(last, llast + nums[i]);
        llast = last;
        last = cur;
    }
    return last;
}

leetcode.213 打家劫舍II medium

你是一个专业的小偷,计划偷窃沿街的房屋,每间房内都藏有一定的现金。这个地方所有的房屋都围成一圈,这意味着第一个房屋和最后一个房屋是紧挨着的。同时,相邻的房屋装有相互连通的防盗系统,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警。

给定一个代表每个房屋存放金额的非负整数数组,计算你在不触动警报装置的情况下,能够偷窃到的最高金额。

示例 :

输入: [2,3,2]
输出: 3
解释: 你不能先偷窃 1 号房屋(金额 = 2),然后偷窃 3 号房屋(金额 = 2), 因为他们是相邻的。

输入: [1,2,3,1]
输出: 4
解释: 你可以先偷窃 1 号房屋(金额 = 1),然后偷窃 3 号房屋(金额 = 3)。
偷窃到的最高金额 = 1 + 3 = 4 。
// 从第0家开始抢,最后一家不能抢  => nums[0...n-1]
// 从第1家开始抢,最后一家可以抢  => nums[1...n]
int rob(vector<int>& nums) {
    if(nums.size() <= 1)
        return nums.empty() ? 0 : nums[0];
    return max(rob(nums, 1, nums.size() - 1), rob(nums, 0, nums.size() - 2));
}
int rob(const vector<int> &nums, int left, int right){
    if(right == left)
        return nums[left];
    int llast = nums[left], last = max(nums[left], nums[left + 1]);
    for(int i = left + 2; i <= right; ++i){
        int cur = max(last, llast + nums[i]);
        llast = last;
        last = cur;
    }
    return last;
}

leetcode.600 不含连续1的非负整数 hard

给定一个正整数 n,找出小于或等于 n 的非负整数中,其二进制表示不包含 连续的1 的个数。

示例 :

输入: 5
输出: 5
解释: 下面是带有相应二进制表示的非负整数<= 5:
0 : 0
1 : 1
2 : 10
3 : 11
4 : 100
5 : 101
其中,只有整数3违反规则(有两个连续的1),其他5个满足规则。

说明: 1 <= n <= 109

// 思路:动态规划中的斐波那契数列,假设二进制位数不大于k的无连续1的非负整数位f(k)
// k = 5时,二进制数为00000-11111,可分为00000-01111 和 10000-10111(11开头的连续两个1,不算在内)
// 则00000-01111表示f(4), 10000-10111就是f(3), 因此有f(5) = f(4) + f(3)
// 先求得1位,2位,3位,...,32位的二进制数不存在连续1的个数
// 再对十进制不超过num的数进行计数即可
int findIntegers(int num) {
    vector<int> fib(32, 0);
    fib[0] = 1;     // 1位的只有 0
    fib[1] = 2;     // 2位的则有 01, 10
    for(int i = 2; i <= 31; ++i)
        fib[i] = fib[i - 1] + fib[i - 2];
    int bit_num = 31;
    bool last_bit = false;
    int res = 0;
    while(bit_num >= 0){					// 如:对100110, 考察<=100000, 100的数中分别有多少个不存在连续1的数
        if(num & (1 << bit_num)){
            res += fib[bit_num];
            if(last_bit == true)  // 连续两个1, 返回。如100110中的110,因为两个1,比他小的都不满足
                return res;
            last_bit = true;
        }else
            last_bit = false;
        --bit_num;
    }
    return res + 1; // 遍历到了这里,说明小于num中没有连续的11, num也应该算在内
}

子序列问题

注:子序列问题区别于后面的子区间问题:子序列是从原序列中提取满足某条件的数,这些数不一定相邻;而子区间问题则必须满足相邻(形成一个不间断的区间)。

leetcode. 300 最长上升子序列 medium

给定一个无序的整数数组,找到其中最长上升子序列的长度。

示例:

输入: [10,9,2,5,3,7,101,18]
输出: 4 
解释: 最长的上升子序列是 [2,3,7,101],它的长度是 4。

说明: 可能会有多种最长上升子序列的组合,你只需要输出对应的长度即可。你算法的时间复杂度应该为 O(n2) 进阶: 你能将算法的时间复杂度降低到 O(n log n) 吗?

// dp[i] = max(dp[k] + 1)  s.t. k < i && nums[k] < nums[i]
// dp[i] 表示以nums[i]结尾的最长递增子序列,nums[i]必然包含在子序列中
int lengthOfLIS(vector<int>& nums) {
    if(nums.size() <= 1)
        return nums.empty() ? 0 : 1;
    vector<int> dp(nums.size(), 1);	// 每个数本身算入其中一个
    for(int i = 1; i < dp.size(); ++i)
        for(int k = 0; k < i; ++k)
            if(nums[k] < nums[i])
                dp[i] = max(dp[i], dp[k] + 1);
    return *max_element(dp.begin(), dp.end());       // 这里需要注意,最大长度不一定出现在dp.back()
}

// 没明白原理的二分法,建立一个vector<int> ends = {nums[0]};
// 遍历nums, 如果num<ends.front(), 直接替换;如果num>ends.back(),直接压入(非替换)
// 如果num < ends.back(),则在ends中查找不小于num的最小数,并将其替换
// 最后ends的长度即为所求(注意此时ends只是长度吻合,其中的数并不是目标上升子序列)
int lengthOfLIS(vector<int> &nums){
    if(nums.empty())
      	return 0;
    vector<int> ends = { nums[0] };
    for(int i = 1; i < nums.size(); ++i)
        if(nums[i] < ends[0])
            ends[0] = nums[i];
        else if(nums[i] > ends.back())
            ends.push_back(nums[i]);
        else{
            int left = 0, right = ends.size();
            while(left < right){
                int mid = (right - left) / 2 + left;
                if(ends[mid] < nums[i])
                    left = mid + 1;
                else
                    right = mid;
            }
            ends[right] = nums[i];
        }
    return ends.size();
}

leetcode.376 摆动序列 medium

如果连续数字之间的差严格地在正数和负数之间交替,则数字序列称为摆动序列。第一个差(如果存在的话)可能是正数或负数。少于两个元素的序列也是摆动序列。

例如, [1,7,4,9,2,5] 是一个摆动序列,因为差值 (6,-3,5,-7,3) 是正负交替出现的。相反, [1,4,7,2,5] 和 [1,7,4,5,5] 不是摆动序列,第一个序列是因为它的前两个差值都是正数,第二个序列是因为它的最后一个差值为零。

给定一个整数序列,返回作为摆动序列的最长子序列的长度。 通过从原始序列中删除一些(也可以不删除)元素来获得子序列,剩下的元素保持其原始顺序。

示例 :

输入: [1,7,4,9,2,5]
输出: 6 
解释: 整个序列均为摆动序列。

输入: [1,17,5,10,13,15,10,5,16,8]
输出: 7
解释: 这个序列包含几个长度为 7 摆动序列,其中一个可为[1,17,10,13,10,16,8]。

输入: [1,2,3,4,5,6,7,8,9]
输出: 2

进阶: 你能否用 O(n) 时间复杂度完成此题?

int wiggleMaxLength(vector<int>& nums) {
    if(nums.size() <= 1)
        return nums.size();

    vector<int> p(nums.size(), 1);  // p[i] 表示 nums[0...i] 中首先往上摆的最长子序列长度
    vector<int> q(nums.size(), 1);  // q[i] 表示 nums[0...i] 中首先往下摆的最长子序列长度
    for(int i = 1; i < nums.size(); ++i)
        for(int j = 0; j < i; ++j)
            if(nums[i] > nums[j])           // nums[i] 相比于 nums[j] 向上摆
                p[i] = max(p[i], q[j] + 1); // q[j]是往下摆,因此下一次就往上摆到达到达nums[j]
    else if(nums[i] < nums[j])      // nums[i] 相比于 nums[j] 向下摆
        q[i] = max(q[i], p[j] + 1);
    return max(q.back(), p.back());
}

// O(n),使用贪心思想,用p,q来记录上述两个状态,一遍遍历即可(不知道为什么可以一遍遍历)
int wiggleMaxLength(vector<int> &nums){
    if(nums.size() <= 1)
        return nums.size();

    int p = 1, q = 1;
    for(int i = 1; i < nums.size(); ++i)
        if(nums[i] > nums[i - 1])       // 往上摆
            p = q + 1;
    else if(nums[i] < nums[i - 1])  // 往下摆
        q = p + 1;
    return max(p, q);
}

最长公共子序列

对于两个子序列 S1 和 S2,找出它们最长的公共子序列。子序列可以是不连续的

// dp[i][j] - S1中前i项和S2中前j项的最长公共长度。根据S1[i]与S2[j]是否相等有两种情况
// S1[i] == S2[j], 则dp[i][j] = dp[i - 1][j - 1] + 1
// S1[i] != S2[j], dp[i][j] = max(dp[i - 1][j], dp[i][j - 1])
int maxCommonLength(const string &s1, const string &s2){
    if(s1.empty() || s2.empty())
        return 0;
  	
    vector<vector<int> > dp(s1.size() + 1, vector<int>(s2.size() + 1, 0));
    for(int i = 1; i <= s1.size(); ++i)
      	for(int j = 1; j <= s2.size(); ++j)
            if(s1[i - 1] == s2[j - 1])
                dp[i][j] = dp[i - 1][j - 1] + 1;
            else
                dp[i][j] = max(dp[i - 1][j], dp[i][j - 1]);

    return dp.back().back();
}

// 优化空间
int minDistance(const string &s1, const string &s2){
    if(s1.empty() || word2.s2())
        return 0;

    vector<int> dp(s2.size() + 1, 0);
    vector<int> last_dp(s2.size() + 1, 0);
    for(int i = 1; i <= s1.size(); ++i){
        for(int j = 1; j <= s2.size(); ++j)
            if(s1[i - 1] == s2[j - 1])
                dp[j] = last_dp[j - 1] + 1;	// dp[i - 1][j - 1]可能已被更新掉,因此用last_dp保存状态
            else
                dp[j] = max(dp[j - 1], dp[j]);
        last_dp = dp;
    }

    return dp.back();
}

leetcode.583 两个字符串的删除操作 medium

给定两个单词 word1 和 word2,找到使得 word1 和 word2 相同所需的最小步数,每步可以删除任意一个字符串中的一个字符。

示例 :

输入: "sea", "eat"
输出: 2
解释: 第一步将"sea"变为"ea",第二步将"eat"变为"ea"

说明: 给定单词的长度不超过500。给定单词中的字符只含有小写字母。

// 思路:找到两个字符串的最长公共子串的大小maxlen, 则需要删除字符的数量为:
// word1.size() + word2.size() - 2 * maxLen
int minDistance(string word1, string word2) {
    if(word1.empty() || word2.empty())
        return word1.empty() ? word2.size() : word1.size();

    vector<vector<int>> dp(word1.size() + 1, vector<int>(word2.size() + 1, 0));
    for(int i = 1; i <= word1.size(); ++i)
        for(int j = 1; j <= word2.size(); ++j)
            if(word1[i - 1] == word2[j - 1])
                dp[i][j] = dp[i - 1][j - 1] + 1;
            else
                dp[i][j] = max(dp[i - 1][j], dp[i][j - 1]);

    return word1.size() + word2.size() - 2 * dp.back().back();
}

// 优化空间同上一题

leetcode. 712 两个字符串的最小ASCII删除和 medium

给定两个字符串s1, s2,找到使两个字符串相等所需删除字符的ASCII值的最小和。

示例 :

输入: s1 = "sea", s2 = "eat"
输出: 231
解释: 在 "sea" 中删除 "s" 并将 "s" 的值(115)加入总和。在 "eat" 中删除 "t" 并将 116 加入总和。结束时,两个字符串相等,115 + 116 = 231 就是符合条件的最小和。

输入: s1 = "delete", s2 = "leet"
输出: 403
解释: 在 "delete" 中删除 "dee" 字符串变成 "let",将 100[d]+101[e]+101[e] 加入总和。在 "leet" 中删除 "e" 将 101[e] 加入总和。结束时,两个字符串都等于 "let",结果即为 100+101+101+101 = 403 。如果改为将两个字符串转换为 "lee" 或 "eet",我们会得到 433 或 417 的结果,比答案更大。

注意: 0 < s1.length, s2.length <= 1000。所有字符串中的字符ASCII值在[97, 122]之间。

// dp[i][j] - s1前i项和s2前j项的最小ASCII删除和,分两种情况
// s1[i] == s2[j], 不用删除,则dp[i][j] = dp[i - 1][j - 1]
// s1[i] != s2[j], 需要删除,可以删除s1[i],也可以删除s2[j]
//      看删除后哪个更小 => dp[i][j] = min(s1[i] + dp[i - 1][j], s2[j] + dp[i][j - 1])
int minimumDeleteSum(string s1, string s2) {
    vector<vector<int>> dp(s1.size() + 1, vector<int>(s2.size() + 1, 0));

    // 第一行,dp[0][j], 表示s2前j个字母变为空字符的ASCII数。第一列同理
    for(int j = 1; j <= s2.size(); ++j)
        dp[0][j] = dp[0][j - 1] + int(s2[j - 1]);
    for(int i = 1; i <= s1.size(); ++i)
        dp[i][0] = dp[i - 1][0] + int(s1[i - 1]);

    for(int i = 1; i <= s1.size(); ++i)
        for(int j = 1; j <= s2.size(); ++j)
            if(s1[i - 1] == s2[j - 1])
                dp[i][j] = dp[i - 1][j - 1];
            else
                dp[i][j] = min(dp[i - 1][j] + s1[i - 1], dp[i][j - 1] + s2[j - 1]);
    
    return dp.back().back();
}

// 优化空间
int minimumDeleteSum(string s1, string s2){
    vector<int> dp(s2.size() + 1, 0);
    vector<int> last_dp(s2.size() + 1, 0);

    for(int j = 1; j <= s2.size(); ++j)
        dp[j] = dp[j - 1] + int(s2[j - 1]);
    last_dp = dp;

    for(int i = 1; i <= s1.size(); ++i){
        dp[0] = dp[0] + int(s1[i - 1]);
        for(int j = 1; j <= s2.size(); ++j)
            if(s1[i - 1] == s2[j - 1])
                dp[j] = last_dp[j - 1]; // 只有这里需要用上一次的状态(即二维dp时该处的左上角)
        else
            dp[j] = min(dp[j] + int(s1[i - 1]), dp[j - 1] + int(s2[j - 1]));
        last_dp = dp;
    }

    return dp.back();
}

leetcode.72 编辑距离

给定两个单词 word1 和 word2,计算出将 word1 转换成 word2 所使用的最少操作数 。

你可以对一个单词进行如下三种操作:插入一个字符,删除一个字符,替换一个字符 示例 :

输入: word1 = "horse", word2 = "ros"
输出: 3
解释: 
horse -> rorse (将 'h' 替换为 'r')
rorse -> rose (删除 'r')
rose -> ros (删除 'e')

输入: word1 = "intention", word2 = "execution"
输出: 5
解释: 
intention -> inention (删除 't')
inention -> enention (将 'i' 替换为 'e')
enention -> exention (将 'n' 替换为 'x')
exention -> exection (将 'n' 替换为 'c')
exection -> execution (插入 'u')
//   ∅ a b c d		当 s1[i] != s2[j]时,分三种情况可以从s1->s2
// ∅ 0 1 2 3 4			操作a. s1[i]处插入字符s[j]使之相等,则比较s1[i + 1], s2[j]即可
// b 1 1 1 2 3			操作b. s1[i]处的字符删除,直接比较s1[i],s1[j + 1]即可
// b 2 2 1 2 3			操作c. s1[i]处的字符变为s[j]处的字符,则比较s1[i + 1], s2[j + 1]即可
// c 3 3 2 1 2		☆这三种操作分别对应的 dp[i][j] 的左侧、上侧、左上侧 + 1,看哪种操作更少
// 
// 对比[最长公共子序列]问题,该问题和本问题类似。区别在于:
// [最长公共子序列]:
// 		允许操作a,b(删除),不允许c(修改), 对应的是 dp[i][j] 的左侧、上侧 + 1, 而没有左上侧 + 1
// [编辑距离]:
//		允许操作a,b,c,因此是左侧、上侧、左上侧 + 1, 并看哪种更少
int minDistance(string s1, string s2) {
    vector<vector<int>> dp(s1.size() + 1, vector<int>(s2.size() + 1, 0));

    for(int j = 1; j <= s2.size(); ++j)
        dp[0][j] = j;
    for(int i = 1; i <= s1.size(); ++i)
        dp[i][0] = i;

    for(int i = 1; i <= s1.size(); ++i)
        for(int j = 1; j <= s2.size(); ++j)
            if(s1[i - 1] == s2[j - 1])
                dp[i][j] = dp[i - 1][j - 1];
    else
        dp[i][j] = min(dp[i - 1][j - 1], min(dp[i - 1][j], dp[i][j - 1])) + 1;

    return dp.back().back();
}

// 优化空间,同前面几个题

子区间问题

leetcode.718 最长重复子数组 medium

注:区别于上一个题的最长公共子序列(子序列不需要连续),本题最长重复子数组需要自数组连续

给两个整数数组 A 和 B ,返回两个数组中公共的、长度最长的子数组的长度。

示例 :

输入:
A: [1,2,3,2,1]
B: [3,2,1,4,7]

输出: 3
解释: 长度最长的公共子数组是 [3, 2, 1]。

说明: 1 <= len(A), len(B) <= 1000,0 <= A[i], B[i] < 100

// dp[i][j] - A中以第i项结尾子数组,和B中以第j项结尾子数组的最长公共长度,根据A[i],B[j]是否相等分两种情况
// A[i] == B[j], 则dp[i][j] = dp[i - 1][j - 1] + 1
// A[i] != B[j], dp[i][j] = 0
int findLength(vector<int>& A, vector<int>& B) {
    vector<vector<int>> dp(A.size() + 1, vector<int>(B.size() + 1, 0));
    int maxLen = 0;
    for(int i = 1; i <= A.size(); ++i)
        for(int j = 1; j <= B.size(); ++j)
            if(A[i - 1] == B[j - 1]){
                dp[i][j] = dp[i - 1][j - 1] + 1;
                maxLen = max(maxLen, dp[i][j]);
            }else
                dp[i][j] = 0;
    return maxLen;
}

// 优化空间
int findLength(vector<int> &A, vector<int> &B){
    vector<int> dp(B.size() + 1, 0);
    int maxLen = 0;
    for(int i = 1; i <= A.size(); ++i)
        for(int j = B.size(); j >= 1; --j)  // 注意,这里必须从后往前遍历
            if(A[i - 1] == B[j - 1]){       //
                dp[j] = dp[j - 1] + 1;      // dp[j]需要使用前一个,前一个却可能已经更新过了
                maxLen = max(maxLen, dp[j]);
            }else
                dp[j] = 0;
    return maxLen;
}

leetcode. 53 最大子序和 easy

给定一个整数数组 nums ,找到一个具有最大和的连续子数组(子数组最少包含一个元素),返回其最大和。

示例:

输入: [-2,1,-3,4,-1,2,1,-5,4],
输出: 6
解释: 连续子数组 [4,-1,2,1] 的和最大,为 6。
// 思路:使用curSum,maxSum记录当前区间、最大区间和
int maxSubArray(vector<int>& nums) {
    if(nums.empty())
        return 0;
    int maxSum = INT_MIN;
    int curSum = 0;
    for(const auto &num:nums){			// curSum + num < num 时,说明curSum对区间和为负增益效果
        curSum = max(num, curSum + num); // 因此直接让num为当前的区间(区间内只含一个数)
        maxSum = max(curSum, maxSum);
    }
    return maxSum;
}

leetcode. 918 环形子数组的最大和 medium

给定一个由整数数组 A 表示的环形数组 C,求 C 的非空子数组的最大可能和。在此处,环形数组意味着数组的末端将会与开头相连呈环状。(形式上,当0 <= i < A.length 时 C[i] = A[i],而当 i >= 0 时 C[i+A.length] = C[i])。此外,子数组最多只能包含固定缓冲区 A 中的每个元素一次。(形式上,对于子数组 C[i], C[i+1], ..., C[j],不存在 i <= k1, k2 <= j 其中 k1 % A.length = k2 % A.length)

示例 1:

输入:[1,-2,3,-2]
输出:3
解释:从子数组 [3] 得到最大和 3
示例 2:

输入:[5,-3,5]
输出:10
解释:从子数组 [5,5] 得到最大和 5 + 5 = 10
示例 3:

输入:[3,-1,2,-1]
输出:4
解释:从子数组 [2,-1,3] 得到最大和 2 + (-1) + 3 = 4
// 思路:与leetcode.53最大子序和类似,区别在于53求的是i~j, 本题可以为i~j, 也可以为0~i + j~n
// 对于 i~j 的情况:直接沿用leetcode.53中的解法
// 对于 0~i + j~n 的情况:可以求最小子序和, sum-最小子序和即为环形子数组的最大和
// 判断两种情况谁的结果更大即可
int maxSubarraySumCircular(vector<int>& A) {
    if(A.empty())
        return 0;

    // i~j 的情况
    int maxSum = INT_MIN;
    int curSum = 0;
    bool all_minus = true;
    for(const auto &num:A){
        curSum = max(num, curSum + num);
        maxSum = max(curSum, maxSum);
        if(num > 0)
            all_minus = false;
    }
    if(all_minus == true)	// 对于所有数均为负数时,minSum == sum
        return maxSum;		// 此时sum - minSum必然为0, 因此需要返回A中最大值, 即maxSum

    // 0~i + j~n 的情况
    int sum = 0;            // 保存A的所有元素和
    int minSum = INT_MAX;
    curSum = 0;
    for(const auto &num:A){
        curSum = min(num, curSum + num);
        minSum = min(curSum, minSum);
        sum += num;
    }

    return max(maxSum, sum - minSum);	//
}

leetcode. 413 等差数列划分 medium

如果一个数列至少有三个元素,并且任意两个相邻元素之差相同,则称该数列为等差数列。例如:

以下数列为等差数列: 1, 3, 5, 7, 9 7, 7, 7, 7 3, -1, -5, -9

以下数列不是等差数列:1, 1, 2, 5, 7

数组 A 包含 N 个数,且索引从0开始。数组 A 的一个子数组划分为数组 (P, Q),P 与 Q 是整数且满足 0<=P<Q<N 。如果满足以下条件,则称子数组(P, Q)为等差数组:元素 A[P], A[p + 1], ..., A[Q - 1], A[Q] 是等差的。并且 P + 1 < Q 。函数要返回数组 A 中所有为等差数组的子数组个数。

示例:

A = [1, 2, 3, 4]
返回: 3
解释:A 中有三个子等差数组: [1, 2, 3], [2, 3, 4] 以及自身 [1, 2, 3, 4]。
// 思路:dp[i] - A[i]结尾的子区间的数量(注意,A[i])
// 因此,A[i] - A[i - 1] == A[i - 1] - A[i - 2], 则dp[i] = dp[i - 1] + 1
// 0 1 2 3 4 5 9 10
// dp[2] = 1, 等差数列为 0 1 2
// dp[3] = 2, 等差数列为 1 2 3, 0 1 2 3
// dp[4] = 3, 登差数列为 2 3 4, 1 2 3 4, 0 1 2 3 4
// dp[5] = 4, 等差数列为 3 4 5, 2 3 4 5, 1 2 3 4 5, 0 1 2 3 4 5
// dp[6] = dp[7] = 0
int numberOfArithmeticSlices(vector<int>& A) {
    if(A.size() <= 2)
        return 0;

    vector<int> dp(A.size(), 0);
    for(int i = 2; i < A.size(); ++i)
        if(A[i] - A[i - 1] == A[i - 1] - A[i - 2])
            dp[i] = dp[i - 1] + 1;

    return accumulate(dp.begin(), dp.end(), 0);
}

// 优化空间
int numberOfArithmeticSlices(vector<int> &A){
    if(A.size() <= 2)
        return 0;

    int dp = 0, sum = 0;
    for(int i = 2; i < A.size(); ++i)
        if(A[i] - A[i - 1] == A[i - 1] - A[i - 2])
            dp += 1, sum += dp;
    else
        dp = 0;

    return sum;
}

leetcode. 303 区域和检索 - 数组不可变 easy

给定一个整数数组 nums,求出数组从索引 i 到 j (i ≤ j) 范围内元素的总和,包含 i, j 两点。

示例:

给定 nums = [-2, 0, 3, -5, 2, -1],求和函数为 sumRange()

sumRange(0, 2) -> 1
sumRange(2, 5) -> -1
sumRange(0, 5) -> -3

说明: 你可以假设数组不可变。会多次调用 sumRange 方法。

// 思路:sums保存累加和,则区间[i...j]的和为sums[j] - sums[i - 1] (包含i在内)
class NumArray {
private:
    vector<int> sums;
public:
    NumArray(vector<int>& nums) {
        int sum = 0;
        for(auto num:nums){
            sum += num;
            sums.push_back(sum);
        }
    }
    
    int sumRange(int i, int j) {
        return i == 0 ? sums[j] : sums[j] - sums[i - 1];
    }
};

leetcode. 560 和为K的子数组 medium

给定一个整数数组和一个整数 k,你需要找到该数组中和为 k 的连续的子数组的个数。

示例 1 :

输入:nums = [1,1,1], k = 2
输出: 2 , [1,1] 与 [1,1] 为两种不同的情况。

说明 : 数组的长度为 [1, 20,000]。数组中元素的范围是 [-1000, 1000] ,且整数 k 的范围是 [-1e7, 1e7]。

// 思路:受leetcode.303启发, 建立累加数组sums, 其中sums[i] = nums[0] + ... + nums[i]
// 则区间(i, j]的和为sums[j] - sums[i]. 为使(i, j]区间和为 target, 问题转化为:
// 在区间和sums[0] ··· sums[j - 1]中,是否存在k使得sums[j]-sums[k]为target,存在多少个?
// 为提高效率,使用哈希表ordered_map保存sum出现的次数
int subarraySum(vector<int>& nums, int target) {
    if(nums.empty())
        return 0;

    unordered_map<int, int> map;
    map[0] = 1;                             // 这里map[0]=1是为了保证区间[0, i]也能计算
    int cur_sum = 0, cnt = 0;               // ↓    举个栗子,比如对 nums = 2,3,10  target = 5
    for(int i = 0; i < nums.size(); ++i){   // ↓    如果不加入map[0] = 2, 则遍历到3时:
        cur_sum += nums[i];                 // ↓    sum = 5, map = { 2:1 }, cur_sum - target = 0
        if(map.count(cur_sum - target) > 0) // ↓    此时map中没有0这一项, 没法统计 2 + 3 这一项
            cnt += map[cur_sum - target];   // 这里
        ++map[cur_sum];
    }

    return cnt;
}

leetcode.930 和相同的二元子数组 medium

在由若干 0 和 1 组成的数组 A 中,有多少个和为 S 的非空子数组。

示例:

输入:A = [1,0,1,0,1], S = 2
输出:4

解释:如下面黑体所示,有 4 个满足题目要求的子数组

[1,0,1,0,1] [1,0,1,0,1] [1,0,1,0,1] [1,0,1,0,1]

提示:A.length <= 30000,0 <= S <= A.length,A[i] 为 0 或 1

// 思路:解法同leetcode.560完全相同
int numSubarraysWithSum(vector<int>& nums, int target) {
    if(nums.empty())
        return 0;

    unordered_map<int, int> map;
    map[0] = 1;                             
    int cur_sum = 0, cnt = 0;
    for(int i = 0; i < nums.size(); ++i){
        cur_sum += nums[i];
        if(map.count(cur_sum - target) > 0)
            cnt += map[cur_sum - target];
        ++map[cur_sum];
    }

    return cnt;
}

leetcode. 523 连续的子数组和 medium

给定一个包含非负数的数组和一个目标整数 k,编写一个函数来判断该数组是否含有连续的子数组,其大小至少为 2,总和为 k 的倍数,即总和为 n*k,其中 n 也是一个整数。

示例 1:

输入: [23,2,4,6,7], k = 6
输出: True
解释: [2,4] 是一个大小为 2 的子数组,并且和为 6。

输入: [23,2,6,4,7], k = 6
输出: True
解释: [23,2,6,4,7]是大小为 5 的子数组,并且和为 42。

说明: 数组的长度不会超过10,000。你可以认为所有数字总和在 32 位有符号整数范围内。

// 思路:和leetcode.560思路相同,使用累加和,需要改动一些地方
// 改动1. 对累加和 % target. 在这样的情况下假设区间 [i, j] 的和为target的倍数
//        则有:sums[i] + n * target = sums[j]  => sums[j] % sums[j] = sums[i] % target
//        即当出现两个累加和相同时,则可确定存在区间和为target的倍数
// 改动2. 为了保证区间长度 > 1, 可用unordered_map保存每个累加和第一次出现的位置
bool checkSubarraySum(vector<int>& nums, int target) {
    if(nums.size() <= 1)
        return false;

    unordered_map<int, int> map;
    map[0] = -1;
    int cur_sum = 0;
    for(int i = 0; i < nums.size(); ++i){
        cur_sum += nums[i];         // 累加和
        if(target != 0)
            cur_sum %= target;      // 改动1, 让区间和为0 ... target - 1
        if(map.count(cur_sum) > 0){ // 存在该累加和
            if(i - map[cur_sum] > 1)// 且区间长度 > 1
                return true;
        }else
            map[cur_sum] = i;       // 改动2. 保存第一个出现该累加和的位置
    }
    return false;
}

leetcode.152 乘积最大子序列 medium

给定一个整数数组 nums ,找出一个序列中乘积最大的连续子序列(该序列至少包含一个数)。

示例 :

输入: [2,3,-2,4]
输出: 6
解释: 子数组 [2,3] 有最大乘积 6。
示例 2:

输入: [-2,0,-1]
输出: 0
解释: 结果不能为 2, 因为 [-2,-1] 不是子数组。
// 思路:使用两个个数组dp1, dp2, 分别表示以nums[i]结尾的最大乘积、最小乘积,再通过maxprod记录最大乘积
// 则: dp1[i] 必然从 dp1[i - 1]*nums[i], dp2[i - 1]*nums[i], nums[i] 中选择最大的一个
//      dp2[i] 必然从 ...选择最小的一个
//      maxprod = max(maxprod, dp1[i]);
int maxProduct(vector<int>& nums) {
    int maxprod = nums[0];
    vector<int> dp1(nums.size(), 0), dp2(nums.size(), 0);
    dp1[0] = nums[0];
    dp2[0] = nums[0];
    for(int i = 1; i < nums.size(); ++i){
        dp1[i] = max(nums[i], max(dp1[i - 1] * nums[i], dp2[i - 1] * nums[i]));
        dp2[i] = min(nums[i], min(dp1[i - 1] * nums[i], dp2[i - 1] * nums[i]));
        maxprod = max(maxprod, dp1[i]);
    }
    return maxprod;
}

// 优化空间
int maxProduct(vector<int> &nums){
    int maxprod = nums[0];
    int dp1 = nums[0], last_dp1 = dp1;		// 使用last_dp1保存上一次的状态
    int dp2 = nums[0], last_dp2 = dp2;
    for(int i = 1; i < nums.size(); ++i){
        dp1 = max(nums[i], max(last_dp1 * nums[i], last_dp2 * nums[i]));
        dp2 = min(nums[i], min(last_dp1 * nums[i], last_dp2 * nums[i]));
        maxprod = max(maxprod, dp1);
        last_dp1 = dp1;
        last_dp2 = dp2;
    }
    return maxprod;
}

leetcode.713 乘积小于K的子数组 medium

给定一个正整数数组 nums。

找出该数组内乘积小于 k 的连续的子数组的个数。

示例 :

输入: nums = [10,5,2,6], k = 100
输出: 8
解释: 8个乘积小于100的子数组分别为: [10], [5], [2], [6], [10,5], [5,2], [2,6], [5,2,6]。
需要注意的是 [10,5,2] 并不是乘积小于100的子数组。

说明: 0 < nums.length <= 50000, 0 < nums[i] < 1000, 0 <= k < 10^6

// 思路:由于nums中全都是正数,因此可以使用双指针,而非动态规划,注意区别
int numSubarrayProductLessThanK(vector<int>& nums, int target) {
    if(target <= 0 || nums.empty())
        return 0;

    int left = 0, right = 0;
    int prod = 1;
    int cnt = 0;
    while(right < nums.size()){
        prod *= nums[right];
        while(left <= right && prod >= target)
            prod /= nums[left++];
        cnt += prod < target ? right - left + 1 : 0;
        ++right;
    }
    return cnt;
}

0-1背包问题

0-1背包问题是非常经典的动态规划问题,是非常容易考到的点。问题描述:

有一个容量为N的背包,要用这个背包装下物品的价值最大,每件物品有体积w、价值v两种属性,应该怎么装

// 思路:动态规划,dp[i][j] - 前i件物品体积不超过j的最大价值
// 假设第i件物品的体积为vol, 价值为val. 则改件物品有两种选项:
//      1. 放入背包, 则dp[i][j] = dp[i - 1][j - vol] + val
//      2. 不放入背包, 则dp[i][j] = dp[i - 1][j]
// 两者取较大的 ===> dp[i][j] = max(dp[i - 1][j - vol] + val, dp[i - 1][j])

// V为背包容量, volumes和values分别为每件物品的体积和价值
int knapsack(const int V, const vector<int> &volumes, const vector<int> &values){
    if(V <= 0 || volumes.size() <= 0)	// 确保背包有容积、存在需要放入背包的物体
        return 0;
    int num = volumes.size();			// 物品数量为num
    vector<vector<int>> dp(num, vector<int>(V + 1, 0));
    
    // 第0件物品: 在背包容积vol >= volumes[0]时,可以将第0件物品放入背包
    for(int vol = 0; vol <= V; ++vol)
        dp[0][vol] = (vol >= volumes[0]) ? values[0] : 0;
    
    // 第1, 2, 3, ..., num-1件物品,使用上述通项公式
    for(int i = 1; i < num; ++i)
        for(int vol = 0; vol <= V; ++vol)
            if(vol >= volumes[i])   // 背包容积vol >= 物品体积时, 尝试放入该物品
                dp[i][vol] = max(dp[i - 1][vol], dp[i - 1][vol - volumes[i]] + values[i]);
            else                    // 背包容积vol < 物品体积时, 必然不放入该物品
                dp[i][vol] = dp[i - 1][vol];
    
    return dp.back().back();
}

// 空间优化, 因为对dp是一行一行进行操作, 所以dp可以直接使用一维数组
int knapsack1(const int V, const vector<int> &volumes, const vector<int> &values){
    if(V <= 0 || volumes.size() <= 0)
        return 0;
    int num = volumes.size();
    vector<int> dp(V + 1, 0);
    
    // 第0件物品
    for(int vol = 0; vol <= V; ++vol)
        dp[vol] = (vol >= volumes[0]) ? values[0] : 0;
    
    // 第1,2,...,num-1件物品
    for(int i = 1; i < num; ++i)
        for(int vol = V; vol >= volumes[i]; --vol)// 必须从大到小遍历, 防止小到大遍历时使用已更改数据
            dp[vol] = max(dp[vol], dp[vol - volumes[i]] + values[i]);
    
    return dp.back();
}

0-1背包问题的变种:

  • 完全0-1背包问题:每种物品可以无限添加

    令每种物品的体积和价值变为1/2/3/4/5...倍,作为一个新物品,每种物品只能加入一次

  • 多维0-1背包问题:每种物品有体积、以及其它属性

    使用多维dp即可

  • 其它变种...


leetcode. 416 分割等和子集 medium

给定一个只包含正整数的非空数组。是否可以将这个数组分割成两个子集,使得两个子集的元素和相等。

注意: 每个数组中的元素不会超过 100, 数组的大小不会超过 200 示例 :

输入: [1, 5, 11, 5]
输出: true
解释: 数组可以分割成 [1, 5, 5] 和 [11].

示例 :
输入: [1, 2, 3, 5]
输出: false
解释: 数组不能分割成两个元素和相等的子集.
// 思路:0-1背包, 背包容量为sum/2, 物品只存在重量属性, 挑选物品恰好装满背包
// dp[i][j] - 下标0~i的数,是否存在某个子集使其总重量为j, 递推关系为
// dp[i][j] = dp[i - 1][j] || dp[i - 1][j - nums[i]]
bool canPartition1(vector<int> &nums){
    int sum = accumulate(nums.begin(), nums.end(), 0);
    if(nums.size() <= 1 || sum % 2 == 1)
        return false;

    int V = sum / 2;
    vector<vector<bool>> dp(nums.size(), vector<bool>(V + 1, 0));

    dp[0][0] = true;        // 不放入第0件物品
    dp[0][nums[0]] = true;  // 要放入第0件物品

    for(int i = 1; i < nums.size(); ++i)
        for(int vol = 0; vol <= V; ++vol)
            if(vol >= nums[i])  // 背包容积 >= 物品体积, 尝试放入物体
                dp[i][vol] = dp[i - 1][vol] || dp[i - 1][vol - nums[i]];
            else                // 背包容积 < 物品体积,必然不放入物体
                dp[i][vol] = dp[i - 1][vol];

    return dp.back().back();
}

// 优化空间, 同时根据可以提前结束节省时间
bool canPartition(vector<int>& nums) {
    int sum = accumulate(nums.begin(), nums.end(), 0);
    if(nums.size() <= 1 || sum % 2 == 1)
        return false;

    int V = sum / 2;
    vector<bool> dp(V + 1, false);
    dp[0] = true;  // 不放入任何物品

    for(const auto &num:nums)
        for(int vol = V; vol >= num; --vol){ // 必须从大到小遍历, 防止小到大使用了已更改数据
            dp[vol] = dp[vol] || dp[vol - num];
            if(dp.back())                   // 已找到, 提前结束
                return true;
        }
    return dp.back();
}

leetcode.698 划分为k个相等的子集

给定一个整数数组 nums 和一个正整数 k,找出是否有可能把这个数组分成 k 个非空子集,其总和都相等。

示例 :

输入: nums = [4, 3, 2, 3, 5, 2, 1], k = 4
输出: True
说明: 有可能将其分成 4 个子集(5),(1,4),(2,3),(2,3)等于总和。

注意:

1 <= k <= len(nums) <= 16 0 < nums[i] < 10000

// 思路1: 可以使用0-1背包的变种, 此题中为多个背包, 物品1种属性,解法比较复杂
// 思路2: 直接使用backtracking方法
bool canPartitionKSubsets(vector<int>& nums, int k) {
    int sum = 0, maxnum = INT_MIN;
    for(const auto &num:nums)
        sum += num, maxnum = max(maxnum, num);
    if(nums.size() < k || sum % k != 0 || maxnum > sum / k)
        return false;

    vector<bool> used(nums.size(), false);
    return backtracking(nums, used, k, sum / k, 0, 0);
}

bool backtracking(vector<int> &nums, vector<bool> &used, int k, int tarSum, int curSum, int idx){
    if(k == 0)           // 已经搜索到了k个子集
        return true;
    if(curSum == tarSum) // 搜索下一个子集
        return backtracking(nums, used, k - 1, tarSum, 0, 0);

    for(int i = idx; i < nums.size(); ++i)  // 这里如果每次都从0开始搜索则会超时, 从idx开始
        if(used[i] == false && curSum + nums[i] <= tarSum){
            used[i] = true;
            if(backtracking(nums, used, k, tarSum, curSum + nums[i], i + 1))
                return true;
            used[i] = false;
        }
    return false;
}


leetcode.494 目标和 medium

给定一个非负整数数组,a1, a2, ..., an, 和一个目标数,S。现在你有两个符号 + 和 -。对于数组中的任意一个整数,你都可以从 + 或 -中选择一个符号添加在前面。返回可以使最终数组和为目标数 S 的所有添加符号的方法数。

示例 :

输入: nums: [1, 1, 1, 1, 1], S: 3
输出: 5
解释:
-1+1+1+1+1 = 3
+1-1+1+1+1 = 3
+1+1-1+1+1 = 3
+1+1+1-1+1 = 3
+1+1+1+1-1 = 3
一共有5种方法让最终目标和为3。

注意:

数组的长度不会超过20,并且数组中的值全为正数。 初始的数组的和不会超过1000。 保证返回的最终结果为32位整数。

// 思路:假设其中符号为+的数的和为P, 符号为-的数的和为N, P + N = nums的和
// P - N = S  => 两边都加上sum =>  (P + N) + (P - N) = S + sum   =>  P = (S + sum) / 2
// 转化:是否存在nums的子集, 使其和为(S + sum) / 2, 这样的子集有多少个?
// dp[j] - 子集和为j的子集个数, dp[j] = dp[j] + dp[j - nums[i]]
int findTargetSumWays(vector<int>& nums, int S) {
    int sum = accumulate(nums.begin(), nums.end(), 0);
    if(sum < S || (sum + S) % 2 != 0)
        return 0;

    int N = (sum + S) / 2;
    vector<int> dp(N + 1, 0);
    dp[0] = 1;      					  // 空集的和为0, 一种情况
    for(const auto &num:nums)
        for(int vol = N; vol >= num; --vol) // 必须从大到小遍历, 防止小到大使用了已更改数据
            dp[vol] = dp[vol] + dp[vol - num];

    return dp.back();
}

leetcode. 282 给表达式添加运算符 hard

给定一个仅包含数字 0-9 的字符串和一个目标值,在数字之间添加二元运算符(不是一元)+、- 或 * ,返回所有能够得到目标值的表达式。

示例 :

输入: num = "123", target = 6
输出: ["1+2+3", "1*2*3"] 

输入: num = "232", target = 8
输出: ["2*3+2", "2+3*2"]

输入: num = "105", target = 5
输出: ["1*0+5","10-5"]

输入: num = "00", target = 0
输出: ["0+0", "0-0", "0*0"]

输入: num = "3456237490", target = 9191
输出: []

提示:此题虽然和leetcode.494目标和类似,但应该使用分治思想求解,具体请查看分治策略


leetcode.139 单词拆分 medium

给定一个非空字符串 s 和一个包含非空单词列表的字典 wordDict,判定 s 是否可以被空格拆分为一个或多个在字典中出现的单词。

说明:拆分时可以重复使用字典中的单词。你可以假设字典中没有重复的单词。 示例 :

输入: s = "leetcode", wordDict = ["leet", "code"]
输出: true
解释: 返回 true 因为 "leetcode" 可以被拆分成 "leet code"。

输入: s = "applepenapple", wordDict = ["apple", "pen"]
输出: true
解释: 返回 true 因为 "applepenapple" 可以被拆分成 "apple pen apple"。注意你可以重复使用字典中的单词。

输入: s = "catsandog", wordDict = ["cats", "dog", "sand", "and", "cat"]
输出: false
// 思路:与leetcode.416类似, 视为0-1背包, 每种物品只有一种属性
// 转化为:背包的大小为字符串长度, 是否存在一个物品的子集恰好装满背包, 且每种物品都是无限的
// dp[i] - 前i个字符是否能被拆分为wordDict中的单词, 递推关系为dp[i] = dp[i] || dp[i - word.size()]
bool wordBreak(string s, vector<string>& wordDict) {
    vector<bool> dp(s.size() + 1, false);
    dp[0] = true;
    for(int i = 1; i <= s.size(); ++i)
        for(const auto &word:wordDict)
            if(word.size() <= i && s.substr(i - word.size(), word.size()) == word)
                dp[i] = dp[i] || dp[i - word.size()];
    return dp.back();
}

leetcode.322 零钱兑换 medium

给定不同面额的硬币 coins 和一个总金额 amount。编写一个函数来计算可以凑成总金额所需的最少的硬币个数。如果没有任何一种硬币组合能组成总金额,返回 -1。

示例 :

输入: coins = [1, 2, 5], amount = 11
输出: 3 
解释: 11 = 5 + 5 + 1

输入: coins = [2], amount = 3
输出: -1
说明:你可以认为每种硬币的数量是无限的。
// 和leetcode.139单词拆分类似,是一个0-1背包问题,每种物品无限多
int coinChange(vector<int>& coins, int amount) {
    vector<int> dp(amount + 1, INT_MAX);
    dp[0] = 0;
    for(int i = 1; i < dp.size(); ++i)
        for(const auto &coin:coins)
            if(coin <= i && dp[i - coin] != INT_MAX)
                dp[i] = min(dp[i], dp[i - coin] + 1);
    return dp.back() == INT_MAX ? -1 : dp.back();
}

leetcode.377 组合总数IV medium

给定一个由正整数组成且不存在重复数字的数组,找出和为给定目标正整数的组合的个数。

示例:

nums = [1, 2, 3]
target = 4
所有可能的组合为:
(1, 1, 1, 1)
(1, 1, 2)
(1, 2, 1)
(1, 3)
(2, 1, 1)
(2, 2)
(3, 1)
请注意,顺序不同的序列被视作不同的组合。因此输出为 7。

进阶: 如果给定的数组中含有负数会怎么样? 问题会产生什么变化? 我们需要在题目中添加什么限制来允许负数的出现?

// 思路:又双叒叕是每种物品无限多的0-1背包问题
int combinationSum4(vector<int>& nums, int target) {
    vector<unsigned long long> dp(target + 1, 0);
    dp[0] = 1;
    for(int i = 1; i < dp.size(); ++i)
        for(const auto &num:nums)
            if(num <= i)
                dp[i] += dp[i - num];
    return dp.back();
}

// 存在负数时:TODO

leetcode.140 单词拆分II hard

给定一个非空字符串 s 和一个包含非空单词列表的字典 wordDict,在字符串中增加空格来构建一个句子,使得句子中所有的单词都在词典中。返回所有这些可能的句子。

说明:分隔时可以重复使用字典中的单词。你可以假设字典中没有重复的单词。 示例 :

输入: s = "catsanddog", wordDict = ["cat", "cats", "and", "sand", "dog"]
输出:
[
"cats and dog",
"cat sand dog"
]

输入: s = "pineapplepenapple", wordDict = ["apple", "pen", "applepen", "pine", "pineapple"]
输出:
[
"pine apple pen apple",
"pineapple pen apple",
"pine applepen apple"
]
解释: 注意你可以重复使用字典中的单词。

输入: s = "catsandog", wordDict = ["cats", "dog", "sand", "and", "cat"]
输出:
[]
// 思路1:直接使用回溯backtracking方法,但是会超时,写在这里做下参考
vector<string> wordBreak(string s, vector<string>& wordDict) {
    vector<string> res;
    string sbroken;
    backtracking(s, res, sbroken, wordDict, 0);
    return res;
}
void backtracking(const string &s, vector<string> &res, string sbroken, 
                 vector<string> &wordDict, int idx){
    if(idx == s.size()){
        res.push_back(sbroken);
        return;
    }
    for(const auto &word:wordDict)
        if(word.size() + idx <= s.size() && s.substr(idx, word.size()) == word){
            string tmp = sbroken;
            sbroken += sbroken.empty() ? word : (" " + word);
            backtracking(s, res, sbroken, wordDict, idx + word.size());
            sbroken = tmp;
        }
}

// 思路2:使用动态规划,unordered_map<string, vector<string>> 记录s的拆分结果
vector<string> wordBreak(string s, vector<string>& wordDict) {
    unordered_map<string, vector<string>> map;
    map[""] = {""};     // 这里初始化空串的拆分方式,肥肠重要!!
    dynamicProg(s, wordDict, map);        
    return map[s];
}
void dynamicProg(string s, vector<string> &wordDict, 
                          unordered_map<string, vector<string>> &map){
    if(map.count(s) > 0 || s.empty())    // s已被拆分过或为空,直接返回,不用拆分
        return;
    vector<string> res;
    for(const auto &word:wordDict)
        if(s.substr(0, word.size()) == word){           				// s可拆分为word + 后半部分
            dynamicProg(s.substr(word.size()), wordDict, map);  // 继续拆分后半

            auto brokens = map[s.substr(word.size())];  				// 将word和后半部分的拆分组合起来
            for(auto broken:brokens)            
                res.push_back(word + (broken.empty() ? "" : " ") + broken);
        }
  
    map[s] = res;   // s的拆分结果放入map备用
}

leetcode.472 连接词 hard

给定一个不含重复单词的列表,编写一个程序,返回给定单词列表中所有的连接词。

连接词的定义为:一个字符串完全是由至少两个给定数组中的单词组成的。

示例:

输入: ["cat","cats","catsdogcats","dog","dogcatsdog","hippopotamuses","rat","ratcatdogcat"]
输出: ["catsdogcats","dogcatsdog","ratcatdogcat"]
解释: "catsdogcats"由"cats", "dog" 和 "cats"组成; 
"dogcatsdog"由"dog", "cats"和"dog"组成; 
"ratcatdogcat"由"rat", "cat", "dog"和"cat"组成。

说明:

给定数组的元素总数不超过 10000。 给定数组中元素的长度总和不超过 600000。 所有输入字符串只包含小写字母。 不需要考虑答案输出的顺序。

// 思路1:与leetcode.139单词拆分基本一样,只需要对words中的每个词调用单词拆分函数即可
// 这一思路很容易超时
vector<string> findAllConcatenatedWordsInADict(vector<string>& words) {
    if(words.size() <= 1)
        return {};

    vector<string> res;
    unordered_set<string> wordsets(words.begin(), words.end());
    for(const auto &s:words){
        wordsets.erase(s);  // 不能拆分为自己
        vector<bool> dp(s.size() + 1, false);
        dp[0] = true;
        for(int i = 1; i <= s.size(); ++i){
            for(int j = 0; j < i; ++j)
                if(dp[j] == true && wordsets.count(s.substr(j, i - j)) > 0)
                    dp[i] = true;

            if(dp.back() == true){  // 找到后提前截止
                res.push_back(s);
                wordsets.insert(s);
                break;
            }
        }    
        wordsets.insert(s);         // 没找到也要将该词复原
    }
    return res;
}

// 思路二:使用前缀树
// TODO: 写完数据结构中的树再来解这个题

leetcode.474 一和零 medium

在计算机界中,我们总是追求用有限的资源获取最大的收益。现在,假设你分别支配着 m 个 0 和 n 个 1。另外,还有一个仅包含 0 和 1 字符串的数组。你的任务是使用给定的 m 个 0 和 n 个 1 ,找到能拼出存在于数组中的字符串的最大数量。每个 0 和 1 至多被使用一次。

注意: 给定 0 和 1 的数量都不会超过 100。给定字符串数组的长度不会超过 600。 示例 :

输入: Array = {"10", "0001", "111001", "1", "0"}, m = 5, n = 3
输出: 4
解释: 总共 4 个字符串可以通过 5 个 0 和 3 个 1 拼出,即 "10","0001","1","0" 。

输入: Array = {"10", "0", "1"}, m = 1, n = 1
输出: 2
解释: 你可以拼出 "10",但之后就没有剩余数字了。更好的选择是拼出 "0" 和 "1" 。
// 思路:多维背包为题,每个物品都有两类属性,0、1的数量,两类属性的背包容量为m、n,最多能放多少个物品
int findMaxForm(vector<string>& strs, int m, int n) {
    vector<vector<int>> dp(m + 1, vector<int>(n + 1, 0));
    for(const auto &s:strs){
        int zeros = 0, ones = 0;
        for(const auto &c:s)
            c == '0' ? ++zeros : ++ones;	// 统计属性
        for(int i = m; i >= zeros; --i)		// 必须大到小遍历,防止小到大使用已更改数据
            for(int j = n; j >= ones; --j)
                dp[i][j] = max(dp[i][j], dp[i - zeros][j - ones] + 1);
    }
    return dp.back().back();
}

其它问题

leetcode.123 买卖股票的最佳时间III hard

给定一个数组,它的第 i 个元素是一支给定的股票在第 i 天的价格。设计一个算法来计算你所能获取的最大利润。你最多可以完成 两笔 交易。注意: 你不能同时参与多笔交易(你必须在再次购买前出售掉之前的股票)。

示例 :

输入: [3,3,5,0,0,3,1,4]
输出: 6
解释: 在第 4 天(股票价格 = 0)的时候买入,在第 6 天(股票价格 = 3)的时候卖出,这笔交易所能获得利润 = 3-0 = 3 。随后,在第 7 天(股票价格 = 1)的时候买入,在第 8 天 (股票价格 = 4)的时候卖出,这笔交易所能获得利润 = 4-1 = 3 。

输入: [1,2,3,4,5]
输出: 4
解释: 在第 1 天(股票价格 = 1)的时候买入,在第 5 天 (股票价格 = 5)的时候卖出, 这笔交易所能获得利润 = 5-1 = 4 。   
注意你不能在第 1 天和第 2 天接连购买股票,之后再将它们卖出。因为这样属于同时参与了多笔交易,你必须在再次购买前出售掉之前的股票。

输入: [7,6,4,3,1] 
输出: 0 
解释: 在这个情况下, 没有交易完成, 所以最大利润为 0。
// 思路:动态规划
// local[i][j] - 到达第i天最多可进行j次交易并且最后一次交易在最后一天卖出的最大利润, 此为局部最大值
// global[i][j] - 到达第i天最多可进行j次交易的最大利润,此为全局最大值
// 局部最大:前一天少交易一次的全局最优加上>0的差值,和前一天的局部最后加差值(不在前一天卖,而在这一天卖),取大值
// 全局最大:比较当天的局部最优,对比前一天的全局最优,两者取较大值
// 	local[i][j] = max(global[i - 1][j - 1] + max(diff, 0), local[i - 1][j] + diff)
// 	global[i][j] = max(local[i][j], global[i - 1][j])
int maxProfit1(vector<int>& prices) {
    if(prices.empty())
        return 0;
    vector<vector<int>> global(prices.size(), vector<int>(2 + 1, 0));
    vector<vector<int>> local(prices.size(), vector<int>(2 + 1, 0));
    for(int i = 1; i < prices.size(); ++i){
        int diff = prices[i] - prices[i - 1];
        for(int j = 1; j <= 2; ++j){
            local[i][j] = max(global[i - 1][j - 1] + max(0, diff), local[i - 1][j] + diff);
            global[i][j] = max(local[i][j], global[i - 1][j]);
        }
    }
    return global.back().back();
}

// 优化空间
int maxProfit(vector<int> &prices){
    if(prices.empty())
        return 0;
    vector<int> global(2 + 1, 0);
    vector<int> local(2 + 1, 0);
    for(int i = 1; i < prices.size(); ++i){
        int diff = prices[i] - prices[i - 1];
        for(int j = 2; j >= 1; --j){    // 必须从大到小遍历,防止小到大使用已更新数据
            local[j] = max(global[j - 1] + max(0, diff), local[j] + diff);
            global[j] = max(local[j], global[j]);
        }
    }
    return global.back();
}

leetcode.188 买卖股票的最佳时机IV hard

给定一个数组,它的第 i 个元素是一支给定的股票在第 i 天的价格。设计一个算法来计算你所能获取的最大利润。你最多可以完成 k 笔交易。注意: 你不能同时参与多笔交易(你必须在再次购买前出售掉之前的股票)。

示例 :

输入: [2,4,1], k = 2
输出: 2
解释: 在第 1 天 (股票价格 = 2) 的时候买入,在第 2 天 (股票价格 = 4) 的时候卖出,这笔交易所能获得利润 = 4-2 = 2 。

输入: [3,2,6,5,0,3], k = 2
输出: 7
解释: 在第 2 天 (股票价格 = 2) 的时候买入,在第 3 天 (股票价格 = 6) 的时候卖出, 这笔交易所能获得利润 = 6-2 = 4。随后,在第 5 天 (股票价格 = 0) 的时候买入,在第 6 天 (股票价格 = 3) 的时候卖出, 这笔交易所能获得利润 = 3-0 = 3 。
// 思路:同leetcode.123买卖股票最佳时机III完全类似
int maxProfit(int k, vector<int>& prices) {
    if(prices.size() <= 1)
        return 0;
  
    // 可交易的次数大于股票数量,可以随意进行买卖,此题退化为leetcode.122买卖股票最佳时机II
    if(k >= prices.size()){
        int prof = 0;
        for(int i = 1; i < prices.size(); ++i)
            prof += (prices[i] - prices[i - 1] >= 0) ? prices[i] - prices[i - 1] : 0;
        return prof;
    }

    vector<vector<int>> local(prices.size(), vector<int>(k + 1, 0));
    vector<vector<int>> global(prices.size(), vector<int>(k + 1, 0));
    for(int i = 1; i < prices.size(); ++i){
        int diff = prices[i] - prices[i - 1];
        for(int j = 1; j <= k; ++j){
            local[i][j] = max(global[i - 1][j - 1] + max(0, diff), local[i - 1][j] + diff);
            global[i][j] = max(local[i][j], global[i - 1][j]);
        }
    }
    return global.back().back();
}

// 空间优化,同leetcode.123完全类似

leetcode. 309 最佳买卖股票时机,含冷冻期 medium

给定一个整数数组,其中第 i 个元素代表了第 i 天的股票价格 。设计一个算法计算出最大利润。在满足以下约束条件下,你可以尽可能地完成更多的交易(多次买卖一支股票)。你不能同时参与多笔交易(你必须在再次购买前出售掉之前的股票)。卖出股票后,你无法在第二天买入股票 (即冷冻期为 1 天)。

示例:

输入: [1,2,3,0,2]
输出: 3 
解释: 对应的交易状态为: [买入, 卖出, 冷冻期, 买入, 卖出]
// 动态规划的思想:
// sell[i] - 第i天结束后,手里没有股票,获得的最大收益
// hold[i] - 第i天结束后,手里持有股票,获得的最大收益
// 对sell[i], 当天啥也没干(sell[i - 1]), 当天卖出昨天持有的股票(hold[i - 1], price[i]),取较大值。
// sell[i] = max(sell[i - 1], hold[i - 1] + price[i]) ,	由于当天结束手里没有股票,因此当天不可能买入。
// 对hold[i], 当天啥也没干(hold[i - 1]), 当天买入了股票(sell[i - 2] - price[i]),取较大值。
// hold[i] = max(hold[i - 1], sell[i - 2] - price[i]), 由于当天结束手里持有股票,因此当天不可能卖出
int maxProfit(vector<int>& prices) {
    if(prices.size() <= 1)
        return 0;
    vector<int> sell(prices.size(), INT_MIN);
    vector<int> hold(prices.size(), INT_MIN);
    sell[0] = 0;            // 第0天卖出股票,收益为0
    hold[0] = -prices[0];   // 第0天买入股票,收益为-prices[0]
    sell[1] = max(sell[0], hold[0] + prices[1]);
    hold[1] = max(hold[0], 0 - prices[1]);
    for(int i = 2; i < prices.size(); ++i){
        sell[i] = max(sell[i - 1], hold[i - 1] + prices[i]);
        hold[i] = max(hold[i - 1], sell[i - 2] - prices[i]);
    }
    return sell.back();
}

leetcode.338 比特位计数 medium

给定一个非负整数 num。对于 0 ≤ i ≤ num 范围中的每个数字 i ,计算其二进制数中的 1 的数目并将它们作为数组返回。

示例 :

输入: 2
输出: [0,1,1]

输入: 5
输出: [0,1,1,2,1,2]

进阶:

给出时间复杂度为O(n*sizeof(integer))的解答非常容易。但你可以在线性时间O(n)内用一趟扫描做到吗? 要求算法的空间复杂度为O(n)。 你能进一步完善解法吗?要求在C++或任何其他语言中不使用任何内置函数(如 C++ 中的 __builtin_popcount)来执行此操作。

// 动态规划: dp[i] - 十进制i的二进制表示中1的个数
// n & (n - 1) 用来去除n的最低位上的1,因此该结果恰好比n少一个二进制1 ==>> dp[n] = dp[n & (n - 1)] + 1
// n 					 = 1100
// n - 1 			 = 1011
// n & (n - 1) = 1000
vector<int> countBits(int num) {
    vector<int> dp(num + 1, 0);
    for(int i = 1; i <= num; ++i)
        dp[i] = dp[i & (i - 1)] + 1;
    return dp;
}

leetcode.646 最长数对链 medium

给出 n 个数对。 在每一个数对中,第一个数字总是比第二个数字小。现在,我们定义一种跟随关系,当且仅当 b < c 时,数对(c, d) 才可以跟在 (a, b) 后面。我们用这种形式来构造一个数对链。

给定一个对数集合,找出能够形成的最长数对链的长度。你不需要用到所有的数对,你可以以任何顺序选择其中的一些数对来构造。

示例 :

输入: [[1,2], [2,3], [3,4]]
输出: 2
解释: 最长的数对链是 [1,2] -> [3,4]
// 思路1:动态规划法O(n^2), 类似暴力求解
int findLongestChain1(vector<vector<int>> &pairs){
    sort(pairs.begin(), pairs.end());
    vector<int> dp(pairs.size(), 1);
    for(int i = 0; i < pairs.size(); ++i)
        for(int j = 0; j < i; ++j)
            if(pairs[j][1] < pairs[i][0])
                dp[i] = max(dp[i], dp[j] + 1);
    return *max_element(dp.begin(), dp.end());
}

// 思路2:贪心算法O(nlogn),按照第二个数字小到大排序,最小的压入栈,
// 遍历每个数对,数对第一个数比栈顶数对的第二个数大即将其入栈,遍历完成后栈的大小即为所求
int findLongestChain(vector<vector<int>>& pairs) {
    if(pairs.empty())
        return 0;
    sort(pairs.begin(), pairs.end(), comp);
    stack<vector<int>> st;
    st.push(pairs[0]);
    for(int i = 1; i < pairs.size(); ++i)
        if(st.top()[1] < pairs[i][0])
            st.push(pairs[i]);
    return st.size();
}
bool comp(vector<int> p1, vector<int> p2){
    return p1[1] < p2[1];
}

leetcode.650 只有两个键的键盘 medium

最初在一个记事本上只有一个字符 'A'。你每次可以对这个记事本进行两种操作:

Copy All (复制全部) : 你可以复制这个记事本中的所有字符(部分的复制是不允许的)。 Paste (粘贴) : 你可以粘贴你上一次复制的字符。 给定一个数字 n 。你需要使用最少的操作次数,在记事本中打印出恰好 n 个 'A'。输出能够打印出 n 个 'A' 的最少操作次数。

示例 :

输入: 3
输出: 3
解释:
最初, 我们只有一个字符 'A'。
第 1 步, 我们使用 Copy All 操作。
第 2 步, 我们使用 Paste 操作来获得 'AA'。
第 3 步, 我们使用 Paste 操作来获得 'AAA'。

说明: n 的取值范围是 [1, 1000] 。

// 动态规划,dp[i] - i个A最少可有多少次操作
  int minSteps(int n) {
      vector<int> dp(n + 1, INT_MAX);
      dp[1] = 0;  // 1个字符,无需操作,本身就有
    
      // 复制前i个字符(1次操作), 粘贴若干次
      for(int i = 1; i < dp.size(); ++i)
          for(int paste = 1; i + paste * i < dp.size(); ++paste)
              dp[i + paste * i] = min(dp[i + paste * i], dp[i] + 1 + paste);	
      return dp.back();
  }

数学

素数

leetcode.204 计数质数 easy

统计所有小于非负整数 n 的质数的数量。

示例:

输入: 10
输出: 4
解释: 小于 10 的质数一共有 4 个, 它们是 2, 3, 5, 7 。
// 找到一个质数a,可以将小于n的a的倍数全部刨除
int countPrimes(int n) {
    if(n <= 1)
        return 0;
  
    vector<int> primes(n, true);
    int cnt = 0;
    for(int i = 2; i < primes.size(); ++i){
        if(primes[i] == false)	// 该数不是质数,不用理会
            continue;
        ++cnt;									// 该数是质数,计数+1,并将该质数的倍数全部排除在外
        for(int multi = 2; multi*i < n; ++multi)
            primes[multi * i] = false;
    }
    return cnt;
}

leetcode.263 丑数 easy

编写一个程序判断给定的数是否为丑数。丑数就是只包含质因数 2, 3, 5 的正整数。

示例 :

输入: 6
输出: true
解释: 6 = 2 × 3

输入: 8
输出: true
解释: 8 = 2 × 2 × 2

输入: 14
输出: false 
解释: 14 不是丑数,因为它包含了另外一个质因数 7。

说明:1 是丑数。输入不会超过 32 位有符号整数的范围: [−2^31, 2^31 − 1]。

bool isUgly(int num) {
    if(num <= 0)
        return false;
    while(num % 2 == 0)     // 排除质因子2
        num /= 2;
    while(num % 3 == 0)     // 排除质因子3
        num /= 3;
    while(num % 5 == 0)     // 排除质因子5
        num /= 5;
    return num == 1;
}

leetcode.264 丑数II medium剑指 offer 丑数

编写一个程序,找出第 n 个丑数。丑数就是只包含质因数 2, 3, 5 的正整数。

示例:

输入: n = 10
输出: 12
解释: 1, 2, 3, 4, 5, 6, 8, 9, 10, 12 是前 10 个丑数。

说明: 1 是丑数。n 不超过1690。

// 质因子|  1倍  2倍  3倍  4倍  5倍 
//   2  |   2    4   6    8   10			由质因子个1/2/3/4...倍即可生成丑数序列
//   3  |   3    6   9    12  15
//   5  |   5    10  15   20  25
int nthUglyNumber(int n) {
    vector<int> uglys = {1};					// 初始化,第一个丑数是1
    int p2 = 0, p3 = 0, p5 = 0;
    while(uglys.size() < n){
        // 对uglys={1}, 		 下一个丑数必然是1*2, 1*3, 1*5的较小值,得到uglys={1,2}
      	// 对uglys={1,2}, 	 下一个丑数必然是2*2, 1*3, 1*5的较小值,得到uglys={1,2,3}
      	// 对uglys={1,2,3}, 下一个丑数必然是2*2, 2*3, 1*5的较小值,得到uglys={1,2,3,4}  以此类推
        int nextUgly = min(uglys[p2]*2, min(uglys[p3]*3, uglys[p5]*5));
        if(nextUgly == uglys[p2]*2)			// 如果下一个丑数是uglys[p2]*2,则将下标p2右移一格. 下面同理
            ++p2;   										
        if(nextUgly == uglys[p3]*3)
            ++p3;
        if(nextUgly == uglys[p5]*5)
            ++p5;
        uglys.push_back(nextUgly);			// 下一个丑数放入丑数数组里
    }
    return uglys.back();
}

leetcode. 313 超级丑数 medium

编写一段程序来查找第 n 个超级丑数。超级丑数是指其所有质因数都是长度为 k 的质数列表 primes 中的正整数。

示例:

输入: n = 12, primes = [2,7,13,19]
输出: 32 
解释: 给定长度为 4 的质数列表 primes = [2,7,13,19],前 12 个超级丑数序列为:[1,2,4,7,8,13,14,16,19,26,28,32] 。

说明:

1 是任何给定 primes 的超级丑数。 给定 primes 中的数字以升序排列。 0 < k ≤ 100, 0 < n ≤ 106, 0 < primes[i] < 1000 。 第 n 个超级丑数确保在 32 位有符整数范围内。

// 思路:本题和leetcode.264丑数的解法完全一致
int nthSuperUglyNumber(int n, vector<int>& primes) {
    vector<int> uglys = {1};
    vector<int> idxs(primes.size(), 0);
    while(uglys.size() < n){
        int nextUgly = INT_MAX;
        for(int i = 0; i < idxs.size(); ++i)
            nextUgly = min(nextUgly, uglys[idxs[i]] * primes[i]);
        for(int i = 0; i < idxs.size(); ++i)
            if(nextUgly == uglys[idxs[i]] * primes[i])
                ++idxs[i];
        uglys.push_back(nextUgly);
    }
    return uglys.back();
}

公约数和公倍数

最大公约数求法:辗转相除法,即gcd(a,b) = gcd(b, a % b)

int gcd(int a, int b){		// a、b谁大谁小没有要求
    return (a % b == 0) ? b : gcd(b, a % b);
}
int gcd(int a, int b){		// 和上一个两者等价
    return b == 0 ? a : gcd(b, a % b);
}

最大公倍数求法:两数乘积除以最大公约数

int lcm(int a, int b){
    return a * b / gcd(a, b);
}

leetcode.858 镜面反射

有一个特殊的正方形房间,每面墙上都有一面镜子。除西南角以外,每个角落都放有一个接受器,编号为 0, 1,以及 2。

正方形房间的墙壁长度为 p,一束激光从西南角射出,首先会与东墙相遇,入射点到接收器 0 的距离为 q 。

返回光线最先遇到的接收器的编号(保证光线最终会遇到一个接收器)。

示例:

输入: p = 2, q = 1
输出: 2
解释: 这条光线在第一次被反射回左边的墙时就遇到了接收器 2 。
// 本质上是求最大公约数, 得到 gcd = gcd_func(p, q)
// 水平方向走了 w = p / gcd 格, 竖直方向走了 h = q / gcd 格
// 根据 w 和 q 的奇偶性判断哪个接收器接收
int mirrorReflection(int p, int q) {
    int gcd = gcd_func(p, q);
    int w = p / gcd, h = q / gcd;
    if((w % 2 == 1) && (h % 2 == 0))  // 横向走了奇数格, 纵向走了偶数格
        return 0;
    if((w % 2 == 1) && (h % 2 == 1))  // 横向走了奇数格, 纵向走了奇数格
        return 1;
    if((w % 2 == 0) && (h % 2 == 1))  // 横向走了偶数个, 纵向走了奇数格
        return 2;
    return -1;                        // 不可能横向偶数个, 纵向偶数格, 因为保证有解
}

// 函数:求最大公约数
int gcd_func(int a, int b){
    return b == 0 ? a : gcd_func(b, a % b);
}

进制转换

leetcode. 504 七进制数 easy

给定一个整数,将其转化为7进制,并以字符串形式输出。

示例 :

输入: 100
输出: "202"
示例 2:

输入: -7
输出: "-10"
注意: 输入范围是 [-1e7, 1e7] 。
// 100 = 2 * 7^2 + 0 * 7^1 + 2 * 7^0
string convertToBase7(int num) {
    if(num == 0)
        return "0";
    string res;
    bool sign = num < 0;
    num = num > 0 ? num : -num;
    while(num > 0){
        res = to_string(num % 7) + res;
        num /= 7;
    }
    return sign ? ("-" + res) : res;
}

leetcode.168 Excel表列名称 easy

给定一个正整数,返回它在 Excel 表中相对应的列名称。

例如,

1 -> A
2 -> B
3 -> C
...
26 -> Z
27 -> AA
28 -> AB 
...

示例 :

输入: 1
输出: "A"

输入: 28
输出: "AB"

输入: 701
输出: "ZY"
// 思路:本题是26进制转换的一个变种,要肥肠注意与普通的26进制的区别
// 比如7进制,每一位上可能出现的数是 0,1,2,3,4,5,6
// 再比如普通的26进制,每一位上可能出现的数是 0,1,2,...,24,25
// 但是,本题是26进制转换的一个变种,每一位上可能出现的数数 1,2,3,...,24,25,26,要肥肠注意
string convertToTitle(int n) {
    string res;
    while(n > 0){
        n -= 1;		// 关键点
        res = table[n % 26] + res;
        n /= 26;
    }
    return res;
}
vector<string> table = { "A","B","C","D","E","F","G","H","I","J","K","L","M","N","O","P","Q","R","S","T","U","V","W","X","Y","Z" };

leetcode.171 Excel表列序号 easy

给定一个Excel表格中的列名称,返回其相应的列序号。

例如,

A -> 1
B -> 2
C -> 3
...
Z -> 26
AA -> 27
AB -> 28 
...

示例

输入: "A"
输出: 1

输入: "AB"
输出: 28

输入: "ZY"
输出: 701
int titleToNumber(string s) {
    long long res = 0;
    for(int i = 0; i < s.size(); ++i){
        res *= 26;
        res += s[i] - 'A' + 1;
    }
    return res > INT_MAX ? INT_MAX : res;;
}

leetcode.405 数字转换成十六进制数 easy

给定一个整数,编写一个算法将这个数转换为十六进制数。对于负整数,我们通常使用 补码运算 方法。

注意:

十六进制中所有字母(a-f)都必须是小写。 十六进制字符串中不能包含多余的前导零。如果要转化的数为0,那么以单个字符'0'来表示;对于其他情况,十六进制字符串中的第一个字符将不会是0字符。 给定的数确保在32位有符号整数范围内。 不能使用任何由库提供的将数字直接转换或格式化为十六进制的方法。

示例 :

输入:26
输出:"1a"

输入:-1
输出:"ffffffff"
// 思路:转化为无符号之后再进制转换
string toHex(int num) {
    if(num == 0)
        return "0";
    unsigned int n = num;
    string res;
    while(n != 0){
        res = table[n % 16] + res;
        n /= 16;
    }
    return res;
}
vector<string> table = {"0","1","2","3","4","5","6","7","8","9","a","b","c","d","e","f"};

leetcode.483 最小好进制

对于给定的整数 n, 如果n的k(k>=2)进制数的所有数位全为1,则称 k(k>=2)是 n 的一个好进制。

以字符串的形式给出 n, 以字符串的形式返回 n 的最小好进制。

示例 :

输入:"13"
输出:"3"
解释:13 的 3 进制是 111。

输入:"4681"
输出:"8"
解释:4681 的 8 进制是 11111。

输入:"1000000000000000000"
输出:"999999999999999999"
解释:1000000000000000000 的 999999999999999999 进制是 11。

提示:n的取值范围是 [3, 10^18]。输入总是有效且没有前导 0。

// 思路:数字n是(n-1)进制下的11,(n-1)进制是好进制,因此最小好进制必然为k=[2,n-1], 可用二分法
// k进制,共m位1,则有关系式
// n = 1 + k + k^2 + ... + k^(m - 1) = (1 - k^m) / (1 - k) >= k^(m - 1)  => k <= n^(1-m)
// k >= 2, 则有m <= 1 + logn/logk <= 1 + logn/log2
// 为使位数m更大,可让 m 从 1+logn/log2 -> 2进行遍历,遍历过程中二分搜索进制k
string smallestGoodBase(string n) {
    unsigned long long num = stol(n);
    for(int m = log(num) / log(2) + 1; m >= 2; --m){          // 搜索位数m
        int k_left = 2, k_right = pow(num, 1.0 / (m - 1)) + 1;    // 搜索进制k
        while(k_left <= k_right){
            int k_mid = (k_right - k_left) / 2 + k_left;
            unsigned long long sum = 0;
            for(int i = 0; i < m; ++i)
                sum = sum * k_mid + 1;
            if(sum == num)
                return to_string(k_mid);
            else if(sum < num)
                k_left = k_mid + 1;
            else
                k_right = k_mid - 1;
        }
    }
    return to_string(num - 1);  // n-1进制的11
}

leetcode.468 验证IP地址 medium

编写一个函数来验证输入的字符串是否是有效的 IPv4 或 IPv6 地址。

IPv4 地址由十进制数和点来表示,每个地址包含4个十进制数,其范围为 0 - 255, 用(".")分割。比如,172.16.254.1;

同时,IPv4 地址内的数不会以 0 开头。比如,地址 172.16.254.01 是不合法的。

IPv6 地址由8组16进制的数字来表示,每组表示 16 比特。这些组数字通过 (":")分割。比如, 2001:0db8:85a3:0000:0000:8a2e:0370:7334 是一个有效的地址。而且,我们可以加入一些以 0 开头的数字,字母可以使用大写,也可以是小写。所以, 2001:db8:85a3:0:0:8A2E:0370:7334 也是一个有效的 IPv6 address地址 (即,忽略 0 开头,忽略大小写)。

然而,我们不能因为某个组的值为 0,而使用一个空的组,以至于出现 (::) 的情况。 比如, 2001:0db8:85a3::8A2E:0370:7334 是无效的 IPv6 地址。

同时,在 IPv6 地址中,多余的 0 也是不被允许的。比如, 02001:0db8:85a3:0000:0000:8a2e:0370:7334 是无效的。

说明: 你可以认为给定的字符串里没有空格或者其他特殊字符。

示例 :

输入: "172.16.254.1"
输出: "IPv4"
解释: 这是一个有效的 IPv4 地址, 所以返回 "IPv4"。

输入: "2001:0db8:85a3:0:0:8A2E:0370:7334"
输出: "IPv6"
解释: 这是一个有效的 IPv6 地址, 所以返回 "IPv6"。

输入: "256.256.256.256"
输出: "Neither"
解释: 这个地址既不是 IPv4 也不是 IPv6 地址。
// 没什么难度,就是需要考虑的情况过多
string validIPAddress(string IP) {
    vector<int> seg;    // 分割符 . : 的位置
    for(int i = 0; i < IP.size(); ++i)
        if(IP[i] == '.' || IP[i] == ':')
            seg.push_back(i);
    if(seg.size() == 3){        // IPv4, 3个 .
        for(auto c:IP)
            if(!(isdigit(c) || c == '.'))
                return "Neither";
        string ad1 = IP.substr(0, seg[0]);
        string ad2 = IP.substr(seg[0] + 1, seg[1] - seg[0] - 1);
        string ad3 = IP.substr(seg[1] + 1, seg[2] - seg[1] - 1);
        string ad4 = IP.substr(seg[2] + 1);
        if(ad1.empty() || ad1.size() > 3 || stoi(ad1) > 255 || (ad1[0] == '0' && ad1.size() > 1)
           || ad2.empty() || ad2.size() > 3 || stoi(ad2) > 255 || (ad2[0] == '0' && ad2.size() > 1)
           || ad3.empty() || ad3.size() > 3 || stoi(ad3) > 255 || (ad3[0] == '0' && ad3.size() > 1)
           || ad4.empty() || ad4.size() > 4 || stoi(ad4) > 255 || (ad4[0] == '0' && ad4.size() > 1))
            return "Neither";
        else
            return "IPv4";
    }else if(seg.size() == 7){  // IPv6, 7个 :
        string ad1 = IP.substr(0, seg[0]);
        if(ad1.empty() || hex2dec(ad1) > 65535 || ad1.size() > 4)
            return "Neither";
        for(int i = 0; i < seg.size() - 1; ++i){
            string ad = IP.substr(seg[i] + 1, seg[i + 1] - seg[i] - 1);
            if(ad.empty() || hex2dec(ad) > 65535 || ad.size() > 4)
                return "Neither";
        }
        string ad8 = IP.substr(seg[6] + 1);
        if(ad8.empty() || hex2dec(ad8) > 65535 || ad8.size() > 4)
            return "Neither";
        return "IPv6";
    }else           // 分割符号数量不对
        return "Neither";
    return "";
}

int hex2dec(string s){
    int res = 0;
    for(int i = 0; i < s.size(); ++i)
        if(isdigit(s[i]))
            res = res * 16 + s[i] - '0';
    else if(isalpha(s[i])){
        if(tolower(s[i]) > 'f')
            return 65536;
        res = res * 16 + (tolower(s[i]) - 'a' + 10);
    }else
        return 65536;
    return res;
}

阶乘

leetcode.172 阶乘后的零 easy

给定一个整数 n,返回 n! 结果尾数中零的数量。

示例 :

输入: 3
输出: 0
解释: 3! = 6, 尾数中没有零。
示例 2:

输入: 5
输出: 1
解释: 5! = 120, 尾数中有 1 个零.
// 思路:阶乘后的所有零都是由质因子 5 引起的,因此n!由多少5的质因子就有多少个0
// n = 1-9      1个0     (5引起)
// n = 10-14    2个0     (多了一个10)
// n = 15-19    3个0     (多了一个15)
// n = 20-24    4个0     (多了一个20)
// n = 25-29    5+1个0   (多了一个25,且有两个5的质因子)
int trailingZeroes(int n) {
    int res = 0;
    while(n > 0){
        res += n / 5;
        n /= 5;
    }
    return res;
}

leetcode.1006 笨阶乘 medium

通常,正整数 n 的阶乘是所有小于或等于 n 的正整数的乘积。例如,factorial(10) = 10 * 9 * 8 * 7 * 6 * 5 * 4 * 3 * 2 * 1。

相反,我们设计了一个笨阶乘 clumsy:在整数的递减序列中,我们以一个固定顺序的操作符序列来依次替换原有的乘法操作符:乘法(*),除法(/),加法(+)和减法(-)。

例如,clumsy(10) = 10 * 9 / 8 + 7 - 6 * 5 / 4 + 3 - 2 * 1。然而,这些运算仍然使用通常的算术运算顺序:我们在任何加、减步骤之前执行所有的乘法和除法步骤,并且按从左到右处理乘法和除法步骤。

另外,我们使用的除法是地板除法(floor division),所以 10 * 9 / 8 等于 11。这保证结果是一个整数。

实现上面定义的笨函数:给定一个整数 N,它返回 N 的笨阶乘。

示例 :

输入:4
输出:7
解释:7 = 4 * 3 / 2 + 1

输入:10
输出:12
解释:12 = 10 * 9 / 8 + 7 - 6 * 5 / 4 + 3 - 2 * 1

提示:1 <= N <= 10000,-2^31 <= answer <= 2^31 - 1

// 思路:前3个数一组,然后每4个数一组,最后0~3个数一组
int clumsy(int N) {
    int res = 0;
    if(N == 1)
        return N;
    if(N == 2)
        return N * (N - 1);
    if(N >= 3)
        res = N * (N - 1) / (N - 2);
    N -= 3;
    while(N >= 4){
        res += N - (N - 1) * (N - 2) / (N - 3);
        N -= 4;
    }
    if(N == 3)
        res = res + 3 - 2 * 1;
    if(N == 2)
        res = res + 2 - 1;
    if(N == 1)
        res = res + 1;
    return res;
}

leetcode.793 阶乘函数后K个零 hard

f(x) 是 x! 末尾是0的数量。(回想一下 x! = 1 * 2 * 3 * ... * x,且0! = 1)

例如, f(3) = 0 ,因为3! = 6的末尾没有0;而 f(11) = 2 ,因为11!= 39916800末端有2个0。给定 K,找出多少个非负整数x ,有 f(x) = K 的性质。

示例 :

输入:K = 0
输出:5
解释: 0!, 1!, 2!, 3!, and 4! 均符合 K = 0 的条件。

输入:K = 5
输出:0
解释:没有匹配到这样的 x!,符合K = 5 的条件。

注意:K是范围在 [0, 10^9] 的整数。

// 与leetcode.172类似, n!的零数量为 K = n/5 + n/5/5 + ... = n/5*(1-1/5^k)/(1-1/5) < n/4 => n > 4K
// 只要有1个n使得n!=K, 那必然一共有5个; 否则一个都没有
int preimageSizeFZF(int K) {
    long long left_n = 0, right_n = ((long long)K) * 5;
    while(left_n <= right_n){
        long long mid_n = (right_n - left_n) / 2 + left_n;
        long long zeros = trailingZeroes(mid_n);
        if(zeros < K)
            left_n = mid_n + 1;
        else if(zeros > K)
            right_n = mid_n - 1;
        else
            return 5;
    }
    return 0;
}

long long trailingZeroes(long long n) {	// n!的零的个数
    long long res = 0;
    while(n > 0){
        res += n / 5;
        n /= 5;
    }
    return res;
}

[编程之美2.2]

求N!的二进制表示中最低位1的位置

// 思路:二进制 000001 每乘以一个质因数2, 则000001中的最低位1往左移1位
// 转化为 N!有多少个质因数2
int lowestOne(int N){
    int res = 0;
    while(N > 0){
        res += n / 2;
        n /= 2;
    }
    return res;
}

字符串加减法

leetcode.67 二进制求和 easy

给定两个二进制字符串,返回他们的和(用二进制表示)。输入为非空字符串且只包含数字 1 和 0。

示例 :

输入: a = "11", b = "1"
输出: "100"

输入: a = "1010", b = "1011"
输出: "10101"
string addBinary(string a, string b) {
    string res;
    int p1 = a.size() - 1, p2 = b.size() - 1;// 从后往前遍历
    int carry = 0;                           // 是否有进位
    while(p1 >= 0 || p2 >= 0 || carry > 0){
        int cur = carry;
        if(p1 >= 0)
            cur += stoi(a.substr(p1--, 1));
        if(p2 >= 0)
            cur += stoi(b.substr(p2--, 1));
        res = to_string(cur % 2) + res;
        carry = cur / 2;
    }
    return res;
}

leetcode.415 字符串相加 easy

给定两个字符串形式的非负整数 num1 和num2 ,计算它们的和。

注意:

num1 和num2 的长度都小于 5100. num1 和num2 都只包含数字 0-9. num1 和num2 都不包含任何前导零。 你不能使用任何內建 BigInteger 库, 也不能直接将输入的字符串转换为整数形式。

// 与leetcode.67完全一致
string addStrings(string num1, string num2) {
    string res;
    int p1 = num1.size() - 1, p2 = num2.size() - 1;
    int carry = 0;
    while(p1 >= 0 || p2 >= 0 || carry > 0){
        int cur = carry;
        if(p1 >= 0)
            cur += stoi(num1.substr(p1--, 1));
        if(p2 >= 0)
            cur += stoi(num2.substr(p2--, 1));
        res = to_string(cur % 10) + res; // 改为 % 10
        carry = cur / 10;                // 该为 / 10
    }
    return res;
}

leetcode.2 两数相加 medium

给出两个 非空 的链表用来表示两个非负的整数。其中,它们各自的位数是按照 逆序 的方式存储的,并且它们的每个节点只能存储 一位 数字。

如果,我们将这两个数相加起来,则会返回一个新的链表来表示它们的和。

您可以假设除了数字 0 之外,这两个数都不会以 0 开头。

示例:

输入:(2 -> 4 -> 3) + (5 -> 6 -> 4)
输出:7 -> 0 -> 8
原因:342 + 465 = 807
// 思路与上面的题一致
ListNode* addTwoNumbers(ListNode* l1, ListNode* l2) {
    ListNode *head = new ListNode(0);   // 哨兵头结点
    ListNode *cur = head, *p1 = l1, *p2 = l2;
    int carry = 0;
    while(p1 != nullptr || p2 != nullptr || carry > 0){
        int sum = carry;
        if(p1 != nullptr){
            sum += p1->val;
            p1 = p1->next;
        }
        if(p2 != nullptr){
            sum += p2->val;
            p2 = p2->next;
        }
        cur->next = new ListNode(sum % 10);
        cur = cur->next;
        carry = sum / 10;
    }
    return head->next;
}

leetcode.445 两数相加II medium

给定两个非空链表来代表两个非负整数。数字最高位位于链表开始位置。它们的每个节点只存储单个数字。将这两数相加会返回一个新的链表。你可以假设除了数字 0 之外,这两个数字都不会以零开头。

进阶: 如果输入链表不能修改该如何处理?换句话说,你不能对列表中的节点进行翻转。

示例:

输入: (7 -> 2 -> 4 -> 3) + (5 -> 6 -> 4)
输出: 7 -> 8 -> 0 -> 7
// 与leetcode.445类似, 但此题是按顺序进行存储的
// 思路1:借助栈
// 思路2:对较短一个链表前面补齐0使长度相等, 然后递归
ListNode* addTwoNumbers(ListNode* l1, ListNode* l2) {
    if(l1 == nullptr || l2 == nullptr)
        return l1 == nullptr ? l2 : l1;
    stack<ListNode*> st1, st2;
    ListNode* p = l1;
    while(p != nullptr){
        st1.push(p);
        p = p->next;
    }
    p = l2;
    while(p != nullptr){
        st2.push(p);
        p = p->next;
    }

    int carry = 0;
    ListNode *cur = nullptr;
    while(!st1.empty() || !st2.empty() || carry > 0){
        int sum = carry;
        if(!st1.empty()){
            sum += st1.top()->val;
            st1.pop();
        }
        if(!st2.empty()){
            sum += st2.top()->val;
            st2.pop();
        }
        ListNode *node = new ListNode(sum % 10);
        node->next = cur;
        cur = node;
        carry = sum / 10;
    }
    return cur;
}

leetcode.43 字符串相乘

给定两个以字符串形式表示的非负整数 num1 和 num2,返回 num1 和 num2 的乘积,它们的乘积也表示为字符串形式。

示例 :

输入: num1 = "2", num2 = "3"
输出: "6"
示例 2:

输入: num1 = "123", num2 = "456"
输出: "56088"

说明:

num1 和 num2 的长度小于110。 num1 和 num2 只包含数字 0-9。 num1 和 num2 均不以零开头,除非是数字 0 本身。 不能使用任何标准库的大数类型(比如 BigInteger)或直接将输入转换为整数来处理。

// 关键点:num1的i位 * num2的j位 -> 不考虑进位时, 对输出的i+j位产生贡献
// idx 4  3  2  1  0       
//           1  2  3
//        X  4  5  6
//      _____________
//           6 12 18
//        5 10 18
//     4  8 15
//
string multiply(string num1, string num2) {
    if(num1 == "0" || num2 == "0")
        return "0";

    vector<int> bitres(num1.size() + num2.size() - 1, 0);
    for(int i = 0; i < num1.size(); ++i)
        for(int j = 0; j < num2.size(); ++j)
            bitres[i + j] += (num1[i] - '0') * (num2[j] - '0');

    string res;
    int carry = 0;
    for(int i = bitres.size() - 1; i >= 0; --i){
        int sum = carry + bitres[i];
        res = to_string(sum % 10) + res;
        carry = sum / 10;
    }
    if(carry)
        res = to_string(carry) + res;

    return res;
}

相遇问题

leetcode.453 最少移动次数使数组元素相等 easy

给定一个长度为 n 的非空整数数组,找到让数组所有元素相等的最小移动次数。每次移动可以使 n - 1 个元素增加 1。

示例:

输入: [1,2,3]
输出: 3
解释: 只需要3次移动(注意每次移动会增加两个元素的值):
[1,2,3]  =>  [2,3,3]  =>  [3,4,3]  =>  [4,4,4]
// 其中n-1个数+1等价于另一个数-1, 最小步数为:每个数-1, 一直减到最小值即可
int minMoves(vector<int>& nums) {
    int min = *min_element(nums.begin(), nums.end());
    int moves = 0;
    for(auto num:nums)
        moves += num - min;
    return moves;
}

leetcode.462 最少移动次数使数组元素相等II medium

给定一个非空整数数组,找到使所有数组元素相等所需的最小移动数,其中每次移动可将选定的一个元素加1或减1。 您可以假设数组的长度最多为10000。

例如:

输入: [1,2,3]
输出: 2
说明:只有两个动作是必要的(记得每一步仅可使其中一个元素加1或减1): 
[1,2,3]  =>  [2,2,3]  =>  [2,2,2]
// 将所有数移动到中位数
// 思路1:先排序再计算O(nlogn)
// 思路2: 快排找到中位数再计算O(n)
int minMoves2(vector<int>& nums) {
    sort(nums.begin(), nums.end());
    int moves = 0;
    int left = 0, right = nums.size() - 1;
    while(left <= right)	// ↓ 等价于 moves += (nums[right--] - mid) + (mid - nums[left++])
        moves += nums[right--] - nums[left++];	
    return moves;        
}

多数投票问题

leetcode.169 求众数 easy剑指 offer 数组中出现次数超过一半的数字

给定一个大小为 n 的数组,找到其中的众数。众数是指在数组中出现次数大于 ⌊ n/2 ⌋ 的元素。

你可以假设数组是非空的,并且给定的数组总是存在众数。

示例 :

输入: [3,2,3]
输出: 3

输入: [2,2,1,1,1,2,2]
输出: 2
// 思路1:排序后的中间数 O(nlogn)
// 思路2:统计出现次数unordered_map
// 思路3:优化思路2的空间消耗 =>  Boyer-Moore多数投票算法
int majorityElement(vector<int>& nums) {
    int midsz = nums.size() / 2;
    unordered_map<int,int> umap;
    for(const auto &num:nums)
        if(++umap[num] > midsz)
            return num;
    return 0;
}

int majorityElement(vector<int>& nums){
    int cnt = 0, me = 0;
    for(const auto &num:nums)
        if(cnt == 0){
            me = num;
            ++cnt;
        }else
            num != me ? --cnt : ++ cnt;
    return me;
}

leetcode.229 求众数II medium

给定一个大小为 n 的数组,找出其中所有出现超过 ⌊ n/3 ⌋ 次的元素。

说明: 要求算法的时间复杂度为 O(n),空间复杂度为 O(1)。

示例 :

输入: [3,2,3]
输出: [3]

输入: [1,1,1,3,3,2,2,2]
输出: [1,2]
// 思路:与leetcode.169类似,是Boyer-moore多数投票算法的变种
vector<int> majorityElement(vector<int>& nums) {
    if(nums.empty())
        return {};

    int n = nums[0], cntn = 0;  // 第一候选人及其计数次数
    int m = nums[0], cntm = 0;  // 第二候选人及其计数次数
    for(const auto &num:nums)
        if(num == n)
            ++cntn;
        else if(num == m)
            ++cntm;
        else if(cntn == 0)
            cntn = 1, n = num;
        else if(cntm == 0)
            cntm = 1, m = num;
        else
            --cntn, --cntm;

    // 对候选人进行统计
    cntn = 0, cntm = 0;
    for(const auto &num:nums){
        if(num == n)
            ++cntn;
        if(num == m)
            ++cntm;
    }

    // 出现次数 > n/3即为所求
    vector<int> res;
    if(cntn > nums.size() / 3)
        res.push_back(n);
    if(cntm > nums.size() / 3)
        res.push_back(m);
    if(n == m)
        res.pop_back();

    return res;
}

其它数学问题

leetcode.367 有效的完全平方数 medium

给定一个正整数 num,编写一个函数,如果 num 是一个完全平方数,则返回 True,否则返回 False。

说明:不要使用任何内置的库函数,如 sqrt。

示例 :

输入:16
输出:True

输入:14
输出:False
// 思路1:二分法
// 思路2:检查1,2,3,4,5....的平方是否为num
bool isPerfectSquare1(int num) {
    if(num < 0)
        return false;
    int left = 1, right = num;
    while(left <= right){
        int mid = (right - left) / 2 + left;
        if(num / mid == mid && num % mid == 0)
            return true;
        else if( num / mid > mid)
            left = mid + 1;
        else
            right = mid - 1;
    }
    return false;
}

bool isPerfectSquare(int num){
    int i = 1;
    while(num / i >= i)
        if(num / i == i && num % i == 0)
            return true;
        else
            ++i;
    
    return false;
}

leetcode.191 位1的个数 easy剑指 offer 二进制中1的个数

返回其二进制表达式中数字位数为 ‘1’ 的个数

// n = n & (n - 1) 可以消除最低位的1
// 检查一共需要消除多少次
int hammingWeight(uint32_t n) {
    int cnt = 0;
    while(n != 0){
        ++cnt;
        n = n & (n - 1);
    }
    return cnt;
}

leetcode.231 2的幂 easy

给定一个整数,编写一个函数来判断它是否是 2 的幂次方。

示例 :

输入: 1
输出: true
解释: 20 = 1

输入: 16
输出: true
解释: 24 = 16

输入: 218
输出: false
// 只需判断n的二进制表示中是否只有一位1
// 利用 n & (n - 1) 消除 n 的最低位1, 看结果是否为零即可
bool isPowerOfTwo(int n) {
    return n > 0 && (n & (n - 1)) == 0;
}

leetcode.326 3的幂 easy

给定一个整数,写一个函数来判断它是否是 3 的幂次方。

示例 :

输入: 27
输出: true
示例 2:

输入: 0
输出: false

输入: 9
输出: true

输入: 45
输出: false
// 小于INT_MAX的3的幂的最大数为3^19=1162261467,它只有质因子3
// 只要 3^19 % n == 0 即可
bool isPowerOfThree(int n) {
  return n > 0 && 1162261467 % n == 0;
}

leetcode. 342 4的幂

给定一个整数 (32 位有符号整数),请编写一个函数来判断它是否是 4 的幂次方。

示例 :

输入: 16
输出: true

输入: 5
输出: false
// 确保 num 只有一位二进制 1, 且该位1在偶数位上
bool isPowerOfFour(int n) {
    return n > 0 && (n & (n - 1)) == 0 && (n & 0x55555555); // 0x5 -> 0101
}

leetcode. 628 三个数的最大乘积

给定一个整型数组,在数组中找出由三个数组成的最大乘积,并输出这个乘积。

示例 :

输入: [1,2,3]
输出: 6
示例 2:

输入: [1,2,3,4]
输出: 24

注意:

给定的整型数组长度范围是[3,104],数组中所有的元素范围是[-1000, 1000]。 输入的数组中任意三个数的乘积不会超出32位有符号整数的范围。

// 思路:最大的三个数m1, m2, m3,最小的两个数n1, n2 (可能为负)
// 最大值必然是 m1*m2*m3, m1*n1*n2中的一个
int maximumProduct(vector<int>& nums) {
    int max1 = INT_MIN, max2 = INT_MIN, max3 = INT_MIN;
    int min1 = INT_MAX, min2 = INT_MAX;
    for(const auto &num:nums){
        if(num > max1){
            max3 = max2;
            max2 = max1;
            max1 = num;
        }else if(num > max2){
            max3 = max2;
            max2 = num;
        }else if(num > max3){
            max3 = num;
        }

        if(num < min1){
            min2 = min1;
            min1 = num;
        }else if(num < min2)
            min2 = num;
    }
    return max(max1 * max2 * max3, max1 * min1 * min2);
}

leetcode.238 除自身以外数组的乘积 medium剑指 offer 构建乘积数组

给定长度为 n 的整数数组 nums,其中 n > 1,返回输出数组 output ,其中 output[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积。

示例:

输入: [1,2,3,4]
输出: [24,12,8,6]
说明: 请不要使用除法,且在 O(n) 时间复杂度内完成此题。

进阶: 你可以在常数空间复杂度内完成这个题目吗?( 出于对空间复杂度分析的目的,输出数组不被视为额外空间。)

// idx  0   1   2   3
//      1   2   3   4
//  b0  \   2   6   24
//  b1  1   \   3   12
//  b2  1   2   \   8
//  b3  1   2   6   \
// 第一遍0-n遍历:计算下三角,第二遍n-0遍历:计算上三角
vector<int> productExceptSelf(vector<int>& nums) {
    vector<int> B(nums.size(), 1);

    int tmp = 1;
    for(int i = 0; i < nums.size() - 1; ++i){
        tmp *= nums[i];
        B[i + 1] *= tmp;
    }

    tmp = 1;
    for(int i = nums.size() - 1; i > 0; --i){
        tmp *= nums[i];
        B[i - 1] *= tmp;
    }

    return B;
}

leetcode.152 乘积最大子序列 medium

动态规划中的子区间问题,查看动态规划-子区间问题

About

C++按tag刷leetcode,主要是算法思想的分类解答。原仓库@CyC2018@h2pl,并增加了若干题目

Topics

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages