题目:给定一个整数数组 nums 和一个目标值 target,请你在该数组中找出和为目标值的那 两个 整数,并返回他们的数组下标。
解析:采用暴力法
有负数 不好hash 且要记录出现和下标
int* twoSum(int* nums, int numsSize, int target, int* returnSize){
int* ans = (int*)malloc(sizeof(int) * 2);
*returnSize = 0;
for (int i = 0; i < numsSize; i++)
{
for (int j = i + 1; j < numsSize; j++)
{
if (nums[j] == target - nums[i])
{
ans[0] = i;
ans[1] = j;
*returnSize = 2;
return ans;
}
}
}
return ans;
}
题目:给定一个排序数组,你需要在原地删除重复出现的元素,使得每个元素只出现一次,返回移除后数组的新长度。
解析:删除元素->重写数组
int removeDuplicates(int* nums, int numsSize){
if (numsSize <= 1)
return numsSize;
int i = 0;
for (int j = 1; j <numsSize; j++)
{
if (nums[j] != nums[i])
{
i++;
nums[i] = nums[j];
}
}
return i+1;
}
题目:给定一个数组 nums 和一个值 val,你需要原地移除所有数值等于 val 的元素,返回移除后数组的新长度。
解析:删除元素->重写数组
int removeElement(int* nums, int numsSize, int val){
if (numsSize == 0 )
return 0;
int i = 0;
for (int j = 0; j < numsSize; j++)
{
if (nums[j] != val)
{
nums[i] = nums[j];
i++;
}
}
return i;
}
题目:给定一个排序数组和一个目标值,在数组中找到目标值,并返回其索引。如果目标值不存在于数组中,返回它将会被按顺序插入的位置。假设数组中无重复元素。
解析:
int searchInsert(int* nums, int numsSize, int target){
int left = 0;
int right = numsSize - 1;
int mid;
while (left <= right)
{
mid = (left + right) / 2;
if (nums[mid] == target)
return mid;
else if (nums[mid] > target)
right = mid - 1;
else
left = mid + 1;
}
return left;
}
题目:给定一个整数数组 nums ,找到一个具有最大和的连续子数组(子数组最少包含一个元素),返回其最大和。
解析:
int maxSubArray(int* nums, int numsSize){
int sum;
int ans;
sum = nums[0];
ans = nums[0];
for (int i = 1; i < numsSize; i++)
{
//sum = Max(sum, 0) + nums[i]; 下面的if..else..等同意思
if (sum > 0)
sum += nums[i];
else
sum = nums[i];
if (sum > ans)
ans = sum;
}
return ans;
}
题目:给定一个由整数组成的非空数组所表示的非负整数,在该数的基础上加一。最高位数字存放在数组的首位, 数组中每个元素只存储单个数字。你可以假设除了整数 0 之外,这个整数不会以零开头。
解析:
int* plusOne(int* digits, int digitsSize, int* returnSize){
for (int i = digitsSize - 1; i >= 0; i--)
{
digits[i] += 1;
digits[i] %= 10;
if (digits[i] != 0)
{
*returnSize = digitsSize;
return digits;
}
}
int* ans = (int*)calloc(digitsSize + 1, sizeof(int));
*returnSize = digitsSize + 1;
ans[0] = 1;
return ans;
}
扩展:如果不是加 1,可用 carry 记录进位数,初始为加数。
最后用 malloc 申请后,再用 memcpy 赋值到 ans+1,ans 首位置1
题目:给定两个有序整数数组 nums1 和 nums2,将 nums2 合并到 nums1 中,使得 num1 成为一个有序数组。
说明:
初始化 nums1 和 nums2 的元素数量分别为 m 和 n。
你可以假设 nums1 有足够的空间(空间大小大于或等于 m + n)来保存 nums2 中的元素。
解析:
void merge(int* nums1, int nums1Size, int m, int* nums2, int nums2Size, int n){
int pm = m-1;
int pn = n-1;
int p = nums1Size-1; //nums1尾部下标
while(pm >=0 && pn >= 0)
{
nums1[p] = nums1[pm] > nums2[pn] ? nums1[pm--] : nums2[pn--]; //可用if..else..
p--;
}
if (pn >= 0) //nums2赋值完应该是-1
memcpy(nums1,nums2, sizeof(int) * (pn+1));
}
题目:给定一个非负整数 numRows,生成杨辉三角的前 numRows 行。
在杨辉三角中,每个数是它左上方和右上方的数的和。
解析:
*returnSize
传进来为了改变returnSize**returnColumnSizes
传进来为了改变*returnColumnSizes
ans[i][j] = ans[i-1][j-1] + ans[i-1][j];
int** generate(int numRows, int* returnSize, int** returnColumnSizes) {
int i, index;
*returnColumnSizes = malloc(numRows * sizeof(int)); //*returnColumnSizes存放每行的列数
for (i = 0; i < numRows; i++)
(*returnColumnSizes)[i] = i + 1;
int** res = malloc(numRows * sizeof(int*));
for (i = 0; i < numRows; i++) {
res[i] = malloc((*returnColumnSizes)[i] * sizeof(int));
res[i][0] = 1;
res[i][(*returnColumnSizes)[i] - 1] = 1;
if (i > 1)
{
for (index = 1; index < (*returnColumnSizes)[i] - 1; index++)
res[i][index] = res[i - 1][index - 1] + res[i - 1][index];
}
}
*returnSize = numRows;
return res;
}
题目:给定一个非负索引 k,其中 k ≤ 33,返回杨辉三角的第 k 行。
在杨辉三角中,每个数是它左上方和右上方的数的和。
解析:
long long
mov eax,dword ptr [a]
imul eax,dword ptr [b]
cdq
idiv eax,dword ptr [c]
mov dword ptr [d],eax
int* getRow(int rowIndex, int* returnSize){
int len = rowIndex + 1;
int* ans = (int*)malloc(sizeof(int) * len);
ans[0] = 1;
long long pre = 1;
for (int i = 1; i < len; i++)
{
pre = pre * (len - i) / i ;
ans[i] = (int)pre;
}
*returnSize = len;
return ans;
}
注:源自于题解
题目:给定一个数组,它的第 i 个元素是一支给定股票第 i 天的价格。
如果你最多只允许完成一笔交易(即买入和卖出一支股票),设计一个算法来计算你所能获取的最大利润。
注意你不能在买入股票前卖出股票。
解析:
int maxProfit(int* prices, int pricesSize){
if (pricesSize == 0)
return 0;
int min = prices[0];
int ans = 0;
for (int i = 1; i < pricesSize; i++)
{
if (prices[i] < min)
min = prices[i];
else if (prices[i] - min > ans)
ans = prices[i] - min;
}
return ans;
}
题目:给定一个数组,它的第 i 个元素是一支给定股票第 i 天的价格。
设计一个算法来计算你所能获取的最大利润。你可以尽可能地完成更多的交易(多次买卖一支股票)。
解析:因为尽可能完成更多交易。
int maxProfit(int* prices, int pricesSize){
int ans = 0;
for (int i = 1; i < pricesSize; i++)
{
if (prices[i] > prices[i-1])
ans += prices[i] - prices[i-1];
}
return ans;
}
题目:给定一个已按照升序排列的有序数组,找到两个数使得它们相加之和等于目标数。
函数应该返回这两个下标值 index1 和 index2,其中 index1 必须小于 index2。
说明:
返回的下标值(index1 和 index2)不是从零开始的。
假设每个输入只对应唯一的答案,而且你不可以重复使用相同的元素。
解析:加数如果大了,就减小;如果小了,就增大。
right--
,如果小了left++
int* twoSum(int* numbers, int numbersSize, int target, int* returnSize){
int left = 0;
int right = numbersSize - 1;
int* ans = (int*)malloc(sizeof(int) * 2);
int sum;
*returnSize = 0;
while (left < right)
{
sum = numbers[left] + numbers[right];
if (sum == target)
{
ans[0] = left + 1;
ans[1] = right + 1;
*returnSize = 2;
break;
}
else if (sum < target)
left++;
else
right--;
}
return ans;
}
题目:给定一个大小为 n 的数组,找到其中的多数元素。多数元素是指在数组中出现次数大于 ⌊ n/2 ⌋ 的元素。
假设数组是非空的,并且给定的数组总是存在多数元素。
解析:
int majorityElement(int* nums, int numsSize){
int ans;
int cnt = 0;
for (int i = 0; i < numsSize; i++)
{
if (cnt == 0)
ans = nums[i];
cnt += (nums[i] == ans) ? 1 : -1;
}
return ans;
}
题目:给定一个数组,将数组中的元素向右移动 k 个位置,其中 k 是非负数。
解析:
k = k % numsSize;
void Reverse(int* arr, int left, int right)
{
while (left < right)
{
int temp = arr[left];
arr[left] = arr[right];
arr[right] = temp;
left++;
right--;
}
}
void rotate(int* nums, int numsSize, int k){
k = k % numsSize;
Reverse(nums,0,numsSize-k-1);
Reverse(nums,numsSize-k,numsSize-1);
Reverse(nums,0,numsSize-1);
}
题目:给定一个整数数组,判断是否存在重复元素。
如果任何值在数组中出现至少两次,函数返回 true。如果数组中每个元素都不相同,则返回 false。
解析:
void DownAdjust(int* arr, int len, int pos)
{
int temp = arr[pos];
int child = pos * 2 + 1;
while (child < len)
{
if (child + 1 < len && arr[child] < arr[child + 1])
child++;
if (arr[child] > temp)
{
arr[pos] = arr[child];
pos = child;
}
else
break;
child = pos * 2 + 1;
}
arr[pos] = temp;
}
void HeapSort(int* arr, int len)
{
for (int i = len/2 - 1; i >= 0; i--)
DownAdjust(arr,len,i);
for (int i = len-1; i > 0; i--)
{
int temp = arr[0];
arr[0] = arr[i];
arr[i] = temp;
DownAdjust(arr, i, 0);
}
}
bool containsDuplicate(int* nums, int numsSize){
if (numsSize < 2)
return false;
HeapSort(nums,numsSize);
for (int i = 1; i < numsSize; i++)
{
if (nums[i] == nums[i-1])
return true;
}
return false;
}
题目:给定一个整数数组和一个整数 k,判断数组中是否存在两个不同的索引 i 和 j,使得 nums [i] = nums [j],并且 i 和 j 的差的绝对值最大为 k。
解析:
typedef struct node
{
int index;
int data;
}node;
void DownAdjust(node* arr, int len, int pos)
{
node temp = arr[pos];
int child = pos * 2 + 1;
while (child < len)
{
if (child + 1 < len && arr[child].data < arr[child+1].data)
child++;
if (arr[child].data > temp.data)
{
arr[pos] = arr[child];
pos = child;
}
else
break;
child = pos * 2 + 1;
}
arr[pos] = temp;
}
void HeapSort(node* arr, int len)
{
for (int i = len/2 - 1; i >= 0; i--)
DownAdjust(arr, len, i);
for (int i = len - 1; i > 0; i--)
{
node temp = arr[i];
arr[i] = arr[0];
arr[0] = temp;
DownAdjust(arr, i, 0);
}
}
bool containsNearbyDuplicate(int* nums, int numsSize, int k){
if (numsSize < 2)
return false;
node* newNums = (node*)malloc(sizeof(node) * numsSize);
for (int i = 0; i < numsSize; i++)
{
newNums[i].index = i;
newNums[i].data = nums[i];
}
HeapSort(newNums, numsSize);
for (int i = 1; i < numsSize; i++)
{C
if (newNums[i].data == newNums[i-1].data)
{
if (abs(newNums[i].index - newNums[i-1].index) <= k)
return true;
}
}
free(newNums); //需要手动释放
return false;
}
题目:给定一个包含 0, 1, 2, …, n 中 n 个数的序列,找出 0 … n 中没有出现在序列中的那个数。
解析:
int missingNumber(int* nums, int numsSize){
int* hash = (int*)calloc(numsSize + 1, sizeof(int));
for (int i = 0; i < numsSize; i++)
hash[nums[i]] = 1;
for (int i = 0; i <= numsSize; i++)
{
if (hash[i] == 0)
return i;
}
return -1; //为了oj能过
题目:给定一个数组 nums,编写一个函数将所有 0 移动到数组的末尾,同时保持非零元素的相对顺序。
解析:一开始想到把 0 全部后移,这要移动很多次非 0 元素
如果 0 少,第一种更快;0 多,第二种更快
这里采用第二种
void moveZeroes(int* nums, int numsSize){
int pos = 0;
for (int i = 0; i < numsSize; i++)
{
if (nums[i] != 0)
{
int temp = nums[i];
nums[i] = nums[pos];
nums[pos] = temp;
pos++;
}
}
}
注:源自于题解
题目:给定一个非空数组,返回此数组中第三大的数。如果不存在,则返回数组中最大的数。要求算法时间复杂度必须是O(n)。
解析:因为要求为 O ( n ) O(n) O(n),所以不能排序
int thirdMax(int* nums, int numsSize){
long long ans[3];
for (int i = 0; i < 3; i++)
ans[i] = LLONG_MIN;
for (int i = 0; i < numsSize; i++)
{
if (nums[i] == ans[0] || nums[i] == ans[1] || nums[i] == ans[2])
continue;
if (nums[i] > ans[0])
{
ans[2] = ans[1];
ans[1] = ans[0];
ans[0] = nums[i];
}
else if (nums[i] > ans[1])
{
ans[2] = ans[1];
ans[1] = nums[i];
}
else if (nums[i] > ans[2])
ans[2] = nums[i];
}
if (ans[2] != LLONG_MIN)
return (int)ans[2];
else
return (int)ans[0];
}
题目:给定一个范围在 1 ≤ a[i] ≤ n ( n = 数组大小 ) 的 整型数组,数组中的元素一些出现了两次,另一些只出现一次。
找到所有在 [1, n] 范围之间没有出现在数组中的数字。
您能在不使用额外空间且时间复杂度为 O ( n ) O(n) O(n)的情况下完成这个任务吗? 你可以假定返回的数组不算在额外空间内。
解析:先想到 hash,用不同数字记录出现否
int* findDisappearedNumbers(int* nums, int numsSize, int* returnSize){
int* ans = (int*)malloc(sizeof(int) * numsSize);
int len = 0;
for (int i = 0; i < numsSize; i++)
{
int pos = abs(nums[i]) - 1; //因为从 1 开始,减 1 位
if (nums[pos] > 0)
nums[pos] *= -1;
}
for (int i = 0; i < numsSize; i++)
{
if (nums[i] > 0)
{
ans[len] = i + 1;
len++;
}
}
*returnSize = len;
return ans;
}
注:源自于题解
题目:给定一个二进制数组, 计算其中最大连续1的个数。
解析:
int findMaxConsecutiveOnes(int* nums, int numsSize){
int sum = 0;
int max = 0;
for (int i = 0; i < numsSize; i++)
{
if (nums[i] == 1)
sum += 1;
else
sum = 0;
if (sum > max)
max = sum;
}
return max;
}
注:如果数据 1 比较多,可以在等于 0 时再更新最大值,返回前更新一次
题目:给定一个整数数组和一个整数 k, 你需要在数组里找到不同的 k-diff 数对。这里将 k-diff 数对定义为一个整数对 (i, j), 其中 i 和 j 都是数组中的数字,且两数之差的绝对值是 k.
解析:
void DownAdjust(int* arr, int len, int pos)
{
int temp = arr[pos];
int child = pos * 2 + 1;
while (child < len)
{
if (child + 1 < len && arr[child] < arr[child + 1])
child++;
if (arr[child] > temp)
{
arr[pos] = arr[child];
pos = child;
}
else
break;
child = pos * 2 + 1;
}
arr[pos] = temp;
}
void HeapSort(int* arr, int len)
{
for (int i = len/2 - 1; i >= 0; i--)
DownAdjust(arr, len, i);
for (int i = len - 1; i > 0; i--)
{
int temp = arr[i];
arr[i] = arr[0];
arr[0] = temp;
DownAdjust(arr, i, 0);
}
}
int findPairs(int* nums, int numsSize, int k){
if (k < 0 || numsSize < 2)
return 0;
int cnt = 0;
HeapSort(nums, numsSize);
for (int i = 0; i < numsSize - 1; i++)
{
while (i < numsSize - 1 && nums[i] == nums[i+1])
i++;
if (k == 0)
{
if (i > 0 && nums[i] == nums[i-1])
++cnt;
}
else
{
for (int j = i + 1; j < numsSize; j++)
{
if (nums[j] - nums[i] == k)
{
cnt++;
break;
}
}
}
}
return cnt;
}