歡迎你們前往騰訊雲+社區,獲取更多騰訊海量技術實踐乾貨哦~javascript
本文由落影發表java
LeetCode上的題目是大公司面試常見的算法題,今天的目標是拿下5道算法題: 題目1是基於鏈表的大數加法,既考察基本數據結構的瞭解,又考察在處理加法過程當中的邊界處理; 題目2是求數組出現頻率前k大的數字,考察思惟能力,代碼很短; 題目3是給出從兩個數組中選擇數字,組成一個最大的數字,考察的是貪心的思想; 前三個都偏向於考察想法,實現的代碼都比較簡單; 題目四、5是數據結構實現題,也是大部分人比較頭疼的題目,由於須要較多的數據結構和STL實現,而且還有時間和空間的限制。面試
題目連接 題目大意:算法
給倆個鏈表,節點由0~9的數字組成,分別表示兩個數字; 求出兩個數字的和,以鏈表的形式返回。數組
例如
Input: (7 -> 2 -> 4 -> 3) + (5 -> 6 -> 4)
7243 + 564 =7807
Output: 7 -> 8 -> 0 -> 7
複製代碼
題目解析: 題目的意思很明顯,就是把兩個數字加起來,須要考慮進位的狀況。 由於是單向的鏈表,遍歷後很難回溯,因此先把數字存到vec中。 而且爲了處理方便,vec的最低位存在vec的起始部分。 因而從0開始遍歷兩個vec便可,注意考慮最後進位的狀況。數據結構
複雜度解析: 時間複雜度是O(N) 空間複雜度是O(N)dom
struct ListNode {
int val;
ListNode *next;
ListNode(int x) : val(x), next(NULL) {}
};
class Solution {
public:
ListNode* addTwoNumbers(ListNode* l1, ListNode* l2) {
ListNode *ret = NULL;
vector<int> vec1, vec2;
sum(l1, vec1);
sum(l2, vec2);
int n = vec1.size(), m = vec2.size(), flag = 0;
for (int i = 0; i < n || i < m; ++i) {
int x = 0, y = 0;
if (i < n) {
x = vec1[i];
}
if (i < m) {
y = vec2[i];
}
int s = x + y + flag;
if (s > 9) {
s -= 10;
flag = 1;
}
else {
flag = 0;
}
ListNode *tmp = new ListNode(s);
tmp->next = ret;
ret = tmp;
}
if (flag) {
ListNode *tmp = new ListNode(1);
tmp->next = ret;
ret = tmp;
}
return ret;
}
void sum(ListNode* list, vector<int> &vec) {
if (list->next) {
sum(list->next, vec);
}
vec.push_back(list->val);
}
};
複製代碼
題目連接 題目大意:優化
給出一個數組和一個數字k,返回按數字出現頻率的前k個的數字; 1 <= k <= n, n是數組大小;ui
example,
Given [1,1,1,2,2,3] and k = 2, return [1,2].
複製代碼
題目解析:this
題目分爲兩個步驟: 一、統計每一個數字的出現次數; 二、從中選擇k個次數最多的數字;
一個簡單的作法: 用哈希表統計每一個數字的出現次數; 把每一個數字的出現次數和數字組成一個pair,放入優先隊列;
這樣從優先隊列中取出k個便可。
複雜度解析: 時間複雜度是O(NlogN),主要在最後的優先隊列。
其餘解法: 有一個O(NlogK)的優化; 首先把隊列變成最小有限隊列, 每次pair放入優先對時,若是當前的size大於k,那麼彈出top; 這樣每次的操做從O(logN)變成O(logK)。
class Solution {
public:
vector<int> topKFrequent(vector<int>& nums, int k) {
unordered_map<int, int> numsHash;
for (int i = 0; i < nums.size(); ++i) {
++numsHash[nums[i]];
}
priority_queue<pair<int, int>> q;
for (int i = 0; i < nums.size(); ++i) {
if(numsHash[nums[i]]) {
q.push(make_pair(numsHash[nums[i]], nums[i]));
numsHash[nums[i]] = 0;
}
}
vector<int> ret;
for (int i = 0; i < k; ++i) {
ret.push_back(q.top().second);
q.pop();
}
return ret;
}
}leetcode;
複製代碼
題目連接 題目大意: 給出兩個數組,數組只包括0~9十個數字,長度分別爲n、m; 從兩個數組中選出k個數,組成一個長度爲k的數字,要求: 一、從數組n、m選擇出來的數字相對位置不變; 二、最後的數字最大; 輸出最後的數字。
Example 1:
nums1 = [3, 4, 6, 5]
nums2 = [9, 1, 2, 5, 8, 3]
k = 5
return [9, 8, 6, 5, 3]
Example 2:
nums1 = [6, 7]
nums2 = [6, 0, 4]
k = 5
return [6, 7, 6, 0, 4]
複製代碼
題目解析:
要求最後數字最大,那麼儘量把數字大的排在前面; 在都合法的前提下,99* 確定比 98*要大; 那麼能夠按照這樣的貪心策略: 先枚舉t,t表示從數組nums1中選出t個數字,那麼數組nums2中應該選出k-t個數字; 兩個數組的全部數字組成最大的數字,由於兩個數組間的數字是能夠任意順序,那麼只需每次選擇較大的放在前面便可。
問題簡化成,O(N)每次從數組中選出t個最大的數字; 這個能夠用貪心解決: 假設數組當前枚舉到第i個,且nums[i]=x; 從左到右遍歷已經選擇的數,當遇到一個數字t,t<x時,判斷插入x後,後續是否存在合法解;若是存在則替換,不然直到最後,插入尾部;
class Solution {
public:
vector<int> maxNumber(vector<int>& nums1, vector<int>& nums2, int k) {
int n = (int)nums1.size(), m = (int)nums2.size();
vector<int> ret(k, 0);
for (int i = max(0, k - m); i <= k && i <= n; ++i) {
vector<int> tmp1 = maxArray(nums1, i);
vector<int> tmp2 = maxArray(nums2, k - i);
vector<int> tmp = merge(tmp1, tmp2, k);
if (greater(tmp, 0, ret, 0)) {
ret = tmp;
}
}
return ret;
}
vector<int> maxArray(vector<int> &nums, int k) {
int n = (int)nums.size();
vector<int> ret(k, 0);
for (int i = 0, j = 0; i < n; ++i) {
while (n - i + j > k && j > 0 && ret[j - 1] < nums[i]) {
--j;
}
if (j < k) {
ret[j++] = nums[i];
}
}
return ret;
}
vector<int> merge(vector<int>& nums1, vector<int>& nums2, int k) {
vector<int> ret(k, 0);
for (int i = 0, j = 0, r = 0; r < k; ++r) {
ret[r] = greater(nums1, i, nums2, j) ? nums1[i++] : nums2[j++];
}
return ret;
}
bool greater(vector<int> &nums1, int i, vector<int> &nums2, int j) {
while (i < nums1.size() && j < nums2.size() && nums1[i] == nums2[j]) {
++i;
++j;
}
return j == nums2.size() || (i < nums1.size() && nums1[i] > nums2[j]);
}
};
複製代碼
題目連接 題目大意: 實現一個數據結構,包括如下三個方法: 一、insert(val): 插入一個數字; 二、remove(val): 移除一個數字; 三、getRandom: O(1)隨機返回一個數字;
Example
插入數字1;
collection.insert(1);
插入數字1:
collection.insert(1);
插入數字2
collection.insert(2);
隨機返回數字,要求 2/3可能返回1, 1/3可能返回2;
collection.getRandom();
複製代碼
題目解析:
插入和移除數字不麻煩,考慮如何在O(1)時間返回一個數字。 容易知道,放在數組裏面能夠,而後隨機返回一個位置能夠實現。 增長能夠在數組最末端增長; 刪除數組中間某個數字時,能夠把最末端的數字放到刪除的位置上;
如今的問題是,如何快速找到數組中該刪除的某個位置; 考慮用hash來實現。 數組就是vector<pair<int, int> >; first存val,second存出現次數; 再用一個哈希map,unordered_map<int, vector> 裏面存對應數字出現的位置;
class RandomizedCollection {
public:
/** Initialize your data structure here. */
RandomizedCollection() {
}
/** Inserts a value to the collection. Returns true if the collection did not already contain the specified element. */
bool insert(int val) {
bool ret = hashMap.find(val) == hashMap.end();
hashMap[val].push_back(randVec.size());
randVec.push_back(make_pair(val, hashMap[val].size() - 1));
return ret;
}
/** Removes a value from the collection. Returns true if the collection contained the specified element. */
bool remove(int val) {
bool ret = hashMap.find(val) != hashMap.end();
if (ret) {
auto last = randVec.back();
hashMap[last.first][last.second] = hashMap[val].back();
randVec[hashMap[val].back()] = last;
hashMap[val].pop_back();
if (hashMap[val].empty()) {
hashMap.erase(val);
}
randVec.pop_back();
}
return ret;
}
/** Get a random element from the collection. */
int getRandom() {
return randVec[rand() % randVec.size()].first;
}
private:
unordered_map<int, vector<int>> hashMap;
vector<pair<int, int>> randVec;
}leetcode;
複製代碼
題目連接 題目大意:
實現一個數據結構,要求: 一、Inc(Key) - Inserts a new key with value 1. Or increments an existing key by 1. Key is guaranteed to be a non-empty string. 二、Dec(Key) - If Key's value is 1, remove it from the data structure. Otherwise decrements an existing key by 1. If the key does not exist, this function does nothing. Key is guaranteed to be a non-empty string. 三、GetMaxKey() - Returns one of the keys with maximal value. If no element exists, return an empty string "". 四、GetMinKey() - Returns one of the keys with minimal value. If no element exists, return an empty string "".
要求全部的數據結構的時間複雜度是O(1);
題目解析:
在不考慮複雜度的前提下,樸素作法是遍歷,O(N); 簡單的優化,用map來維護優先隊列,操做一、2先獲取key值,更新完從新插入;操做三、4直接拿隊列top;每一個操做的複雜度是O(LogN);
題目要求是O(1),那麼必然不能使用樹類型的結構,應該利用題目特性,配合hash以及貪心來實現。
假設有一個key-hash表,來存key的對應值。 操做一、先看keyHash裏面是否有key,有則+1,無則插入; 操做二、先看keyHash裏面是否有key,有則-1,無則Nothing;
爲了維護最值,引入鏈表list,裏面全部的元素是從小到大;每一個元素是一個桶,桶裏放着值相同的key; 操做三、直接獲取list頭元素的值; 操做四、直接獲取list尾元素的值;
同時,操做一、2在操做的過程當中,須要把當前key值從list對應的桶裏移除,放到上一個或者下一個桶裏,或者丟棄。 爲了實現O(1)獲取key所在位置,能夠用iter-hash來維護key所對應元素的迭代器。
struct Bucket {
int value;
unordered_set<string> keys;
};
class AllOne {
public:
list<Bucket> buckets;
unordered_map<string, list<Bucket>::iterator> bucketOfKey;
/** Initialize your data structure here. */
AllOne() {
}
/** Inserts a new key <Key> with value 1. Or increments an existing key by 1. */
void inc(string key) {
if (bucketOfKey.find(key) == bucketOfKey.end()) {
bucketOfKey[key] = buckets.insert(buckets.begin(), {0, {key}});
}
auto next = bucketOfKey[key], bucket = next++;
if (next == buckets.end() || next->value > bucket->value + 1) {
next = buckets.insert(next, {bucket->value+1, {}});
}
next->keys.insert(key);
bucketOfKey[key] = next;
bucket->keys.erase(key);
if (bucket->keys.empty()) {
buckets.erase(bucket);
}
}
/** Decrements an existing key by 1. If Key's value is 1, remove it from the data structure. */
void dec(string key) {
if (bucketOfKey.find(key) == bucketOfKey.end()) {
return ;
}
auto pre = bucketOfKey[key], bucket = pre;
if (pre != buckets.begin()) {
--pre;
}
bucketOfKey.erase(key);
if (bucket->value > 1) {
if (bucket == buckets.begin() || pre->value < bucket->value - 1) {
pre = buckets.insert(bucket, {bucket->value - 1, {}});
}
pre->keys.insert(key);
bucketOfKey[key] = pre;
}
bucket->keys.erase(key);
if (bucket->keys.empty()) {
buckets.erase(bucket);
}
}
/** Returns one of the keys with maximal value. */
string getMaxKey() {
return buckets.empty() ? "" : *(buckets.rbegin()->keys.begin());
}
/** Returns one of the keys with Minimal value. */
string getMinKey() {
return buckets.empty() ? "" : *(buckets.begin()->keys.begin());
}
}leetcode;
複製代碼
這5個題目若是都能獨立完成,那麼水平已經能夠足以應付國內各大企業的算法面。 算法重在勤思多練,埋怨公司出算法題是沒用的,多花時間準備纔是正道。
此文已由做者受權騰訊雲+社區發佈,更多原文請點擊
搜索關注公衆號「雲加社區」,第一時間獲取技術乾貨,關注後回覆1024 送你一份技術課程大禮包!