给定一个整数数组和一个目标值,找出数组中和为目标值的两个数。
你可以假设每个输入只对应一种答案,且同样的元素不能被重复利用。
给定 nums = [2, 7, 11, 15], target = 9
因为 nums[0] + nums[1] = 2 + 7 = 9
所以返回 [0, 1]
1.暴力法[C++(默认)]
前言vector初步理解:
pushback()操作函数 :算法中里面的一个函数名,如c++中的vector头文件里面就有这个push_back函数,在vector类中作用为在vector尾部加入一个数据。string中也有这个函数,作用是字符串之后插入一个字符。简单地说,vector是一个能够存放任意类型的动态数组,能够增加和压缩数据。
本文是指标准模板库(stl)中容器的pushback()操作函数,那么是指在容器尾端插入一项数据,比如
#include
#include"vector"//vscode下#include报错
#include"algorithm"
using namespace std;
void main01(){
vector a(3);//int型vector,包含3个元素
a.push_back(10);
//打印
for(vector::iterator it = a.begin();it!=a.end();it++)
{
cout<<*it<<" ";
}
cout<
结果:
0 0 0 10
暴力法代码O(n^2):
class Solution {
public:
vector twoSum(vector& nums, int target) {
vector res;
for(int i=0;i
2.哈希map复杂度O(n)
思路:耗费O(n)空间构造哈希表,遍历数组每个元素nums[i]
,哈希表对应存储
,存储nums[i]
期望的“另一半”,一旦哈希表中包含nums[i]
,代表“另一半”早已存储在哈希表中,直接返回即可;
复杂度分析:时间复杂度O(n),空间复杂度O(n)
说明:
C++中map提供的是一种键值对容器,里面的数据都是成对出现的,如下图:每一对中的第一个值称之为关键字(key),每个关键字只能在map中出现一次;第二个称之为该关键字的对应值。如 map[1120217]="Nikhilesh"map基本介绍
Snipaste_2018-06-06_23-31-03.png
C++代码:
class Solution {
public:
vector twoSum(vector& nums, int target) {
unordered_map m;
vector res;
for(int i=0;i
反例: //m[target-nums[i]]=i;导致冲突target是自身2倍,到底是先放还是后放入hash表中,逻辑!逻辑!逻辑!一定要搞清楚,不要乱,
Snipaste_2018-06-07_00-17-43.png
总结:逻辑上是先在存放 之前理想值的 hash表 里找 是否是 当前的值,首先最开始的第一个元素不需要查找,因为它肯定没有之前的理想值。
3.排序查找法
思路:首先将数组排序O(nlogn),然后通过双指针 left
和 right
分别从数组两端同时遍历,但排序会打乱原来数组index的顺序。我们可以建立一个class/struct/pair来存储val/index,并overload operator < 来以val值排序。保存数组排序前的元素位置(空间复杂度(O(n)),
复杂度分析:时间复杂度O(nlog(n)),空间复杂度(O(n))或者O(1)
代码C++:
class Solution {
class elem{
public:
int val;
int index;
elem(int v,int i):val(v),index(i){}
bool operator<(const elem &e)const{
return val twoSum(vector& nums, int target) {
//vector res;替换这样运行不了
vector res(2,1);//两个值为1的vector
vector arr;
for(int i=0;i
给定一个包含 n 个整数的数组 nums,判断 nums 中是否存在三个元素 a,b,c ,使得 a + b + c = 0 ?找出所有满足条件且不重复的三元组。
例如, 给定数组 nums = [-1, 0, 1, 2, -1, -4],
满足要求的三元组集合为:
[
[-1, 0, 1],
[-1, -1, 2]
]
题目链接
思想参考
C++创建动态二维数组
vector
注意:答案中不可以包含重复的三元组。
分析:和上题排序二分查找思想一样
解法一:c++ code:AC 95%
#include
#include
#include
#include
#include
#include
#include
using namespace std;
bool compare(int i, int j)
{
return (i < j);
}
class Solution {
public:
vector> threeSum(vector& nums) {
int len = nums.size();
//vector>res(len,vector(3));
vector>res;
/*vector > ivec;
ivec.resize(len);
for (int i = 0; i0||nums.back()<0) return{};
for (int t = 0; t < nums.size(); t++)
{
if (nums[t]>0)break;
if (t>0 && nums[t] == nums[t-1])continue;//前面的去重复。++t不行。还有可能三个连续的
int target = 0-nums[t];
int i = t + 1, j = len - 1;
while (i < j)
{
////不能放在这里,反例000
//while (i < j&&nums[i] == nums[i + 1]) ++i;
//while (i < j&&nums[j] == nums[j - 1]) --j;
if (target == (nums[i] + nums[j]))
{
res.push_back({ nums[t], nums[i], nums[j] });
//必须收集完之后进行后面的去重
while (i < j&&nums[i] == nums[i + 1]) ++i;
while (i < j&&nums[j] == nums[j - 1]) --j;
++i; --j;
}
else if (target < (nums[i] + nums[j]))
{
j--;
}
else
{
i++;
}
}
}
return res;
}
};
void trimLeftTrailingSpaces(string &input) {
input.erase(input.begin(), find_if(input.begin(), input.end(), [](int ch) {
return !isspace(ch);
}));
}
void trimRightTrailingSpaces(string &input) {
input.erase(find_if(input.rbegin(), input.rend(), [](int ch) {
return !isspace(ch);
}).base(), input.end());
}
vector stringToIntegerVector(string input) {
vector output;
trimLeftTrailingSpaces(input);
trimRightTrailingSpaces(input);
input = input.substr(1, input.length() - 2);
stringstream ss;
ss.str(input);
string item;
char delim = ',';
while (getline(ss, item, delim)) {
output.push_back(stoi(item));
}
return output;
}
int main() {
string line;
while (getline(cin, line)) {
vector height = stringToIntegerVector(line);
vector> threeSum = Solution().threeSum(height);
vectortemp_vect;
for (vector>::iterator ite = threeSum.begin(); ite != threeSum.end(); ite++)
{
temp_vect = *ite;
for (vector::iterator jte = temp_vect.begin(); jte != temp_vect.end(); jte++)
{
cout << *jte<<" ";
}cout << endl;
}
}
return 0;
}
解法二:c++ code:AC 12.5%
上述考虑了去重比较麻烦,可以用set解决:
bool compare(int i, int j)
{
return (i < j);
}
class Solution {
public:
vector> threeSum(vector& nums) {
int len = nums.size();
//vector>res(len,vector(3));
set>res;
/*vector > ivec;
ivec.resize(len);
for (int i = 0; i0||nums.back()<0) return{};
for (int t = 0; t < nums.size(); t++)
{
if (nums[t]>0)break;
int target = 0-nums[t];
int i = t + 1, j = len - 1;
while (i < j)
{
if (target == (nums[i] + nums[j]))
{
res.insert({ nums[t], nums[i], nums[j] });
++i; --j;
}
else if (target < (nums[i] + nums[j]))
{
j--;
}
else
{
i++;
}
}
}
return vector>(res.begin(),res.end());
}
};
通过分析:我们可以想到一种时间复杂度为的解法:假设数组中有len个元素,首先我们将数组中的元素按照从小到大的顺序进行排序。其次,看最终取出的三个数中的第一个数,若数组长度为n,那么有n种取法。假设取的第一个数是A[i],那么第二三两个数从A[i+1]~A[len]中取出。找到“第一个数为A[i]固定,后两个数在A[i]后面元素中取。并且三数之和离target最近的情况。”这时,我们用两个指针j,k分别指向A[i+1]和A[len],如果此时三数之和A[i]+A[j]+A[k]
核心代码:c++ code AC 93%
class Solution {
public:
int threeSumClosest(vector& nums, int target) {
int len = nums.size();
sort(nums.begin(),nums.end());
if (nums.empty() || nums.size()<3) return{};
int res = nums[0] + nums[1] + nums[2];
int min = abs(res - target);
for (int i = 0; i < nums.size(); i++)
{
int j = i + 1, k =len - 1;
while (j < k)
{
int temp = abs(nums[i] + nums[j] + nums[k] - target);
if (temp
完整测试代码:
#include
#include
#include
#include
#include
#include
#include
#include
using namespace std;
class Solution {
public:
int threeSumClosest(vector& nums, int target) {
int len = nums.size();
sort(nums.begin(),nums.end());
if (nums.empty() || nums.size()<3) return{};
int res = nums[0] + nums[1] + nums[2];
int min = abs(res - target);
for (int i = 0; i < nums.size(); i++)
{
int j = i + 1, k =len - 1;
while (j < k)
{
int temp = abs(nums[i] + nums[j] + nums[k] - target);
if (temp stringToIntegerVector(string input) {
vector output;
trimLeftTrailingSpaces(input);
trimRightTrailingSpaces(input);
input = input.substr(1, input.length() - 2);
stringstream ss;
ss.str(input);
string item;
char delim = ',';
while (getline(ss, item, delim)) {
output.push_back(stoi(item));
}
return output;
}
int stringToInterger(string s)
{
return stoi(s);
}
int main() {
string line;
while (getline(cin, line)) {
vector height = stringToIntegerVector(line);
getline(cin, line);
int target = stringToInterger(line);
int res = Solution().threeSumClosest(height, target);
cout << res << endl;
}
return 0;
}
思路和上面三数之和一样,只是加了一个for循环,时间复杂度
class Solution {
public:
vector> fourSum(vector& nums, int target) {
if (nums.empty())return{};
vector>res;
int len = nums.size();
sort(nums.begin(), nums.end());
for (int i = 0; i < len - 3; i++)
{
if (i > 0 && nums[i] == nums[i - 1])continue;
for (int j = i + 1; j < len - 2; j++)
{
if (j > i + 1 && nums[j] == nums[j - 1])continue;
int low = j + 1; int high = len - 1;
while (low < high)
{
int sum = nums[i] + nums[j] + nums[low] + nums[high];
if (sum == target)
{
res.push_back({ nums[i],nums[j] ,nums[low] ,nums[high] });//同时放进多个元素
while (low < high&&nums[low] == nums[low + 1])low++;
while (low < high&&nums[high] == nums[high - 1])high--;
low++; high--;
}
else if (sum < target) low++;
else high--;
}
}
}
return res;
}
};
参考1
参考2
Leetcode16. 最接近的三数之和.参考