本文共 18326 字,大约阅读时间需要 61 分钟。
言简意赅,持续更新,利于速览复习。有导航、有代码、有细节、有引申。
已记录题目编号:1, 3, 5, 10, 15, 20, 21, 26, 53, 54, 56, 65, 72, 79, 84, 88, 101, 102, 103, 104, 105, 121, 122, 123, 125, 136, 137, 138, 145, 146, 153, 154, 155, 161, 167, 169, 170, 172, 190, 191, 198, 203, 206, 215, 217, 219, 220, 226, 229, 240, 295, 297, 343,426, 653, 946, 974, 1209
ListNode*head=new ListNode(0);ListNode*p=head;...return head->next;
int maxSubArray(vector & nums) { if(nums.size()==0)return 0; int i=1; int maxsum=nums[0]; int sum=maxsum; while(i<=nums.size()-1){ if(sum<0){ sum=nums[i]; } else sum+=nums[i]; if(sum>=maxsum) { maxsum=sum; } i++; } return maxsum;}
vector spiralOrder(vector>& matrix) { vector ret; int m=matrix.size(); if(!m)return ret; int n=matrix[0].size(); int b=0,t=m-1,l=0,r=n-1; while(1){ for(int j=l;j<=r;j++)ret.push_back(matrix[b][j]); if(++b>t)break; for(int i=b;i<=t;i++)ret.push_back(matrix[i][r]); if(--r =l;j--)ret.push_back(matrix[t][j]); if(--t =b;i--)ret.push_back(matrix[i][l]); if(++l>r)break; } return ret;}
static bool cmp1(vector &a, vector &b){ return a[0] < b[0] || (a[0] == b[0] && a[1] < b[1]);}
int pointer;bool isNumber(string s) { if(s=="")return -1; scanSpace(s); bool numeric=scanInteger(s); if(s[pointer]=='.'){ ++pointer; numeric=scanUnsignedInteger(s)||numeric; //用||因为整数、小数部分有一即可 } if(s[pointer]=='e'||s[pointer]=='E'){ ++pointer; numeric=numeric&&scanInteger(s); } scanSpace(s); return numeric&&s[pointer]=='\0';}
KaTeX parse error: Undefined control sequence: \notag at position 108: …i]\\\end{cases}\̲n̲o̲t̲a̲g̲ ̲
int largestRectangleArea(vector &heights){ stack st; int top = 0; int maxarea = 0; heights.insert(heights.begin(), 0); heights.push_back(0); //左右插0,利于边界条件 for (int i = 0; i < heights.size(); i++) { top = i; while (!st.empty() && heights[i] < heights[st.top()]) { top = st.top(); st.pop(); maxarea = max(maxarea, heights[top] * (i-1 - st.top())); }//求出栈处往左右延伸的最大矩形面积 st.push(i); } return maxarea;}
stack st;//此处一般需要给数组最后添加结束标志符for (遍历这个数组){ while (栈不为空 && 栈顶元素小于当前元素) { 栈顶元素出栈; 更新结果; } 当前数据入栈;}
bool isSymmetric(TreeNode* root) { if(!root)return 1; else return isSymmetric1(root->left,root->right);}bool isSymmetric1(TreeNode* a,TreeNode* b) { if(!(a||b))return 1; else if(!(a&&b))return 0; else return a->val==b->val&&isSymmetric1(a->left,b->right)&&isSymmetric1(a->right,b->left);}
方法一:clear
方法二:vector<int>().swap(nums);
方法三:利用代码块和临时变量
{ vector<int> tmp = curLevel; curLevel.swap(tmp); }
clear虽然不会deallocate释放空间,但是会destroy执行析构函数,所以可以用同一个空间构造节点,如果swap了就要重新分配空间再构造节点。由于本题是对同一个vector的重复利用,可以直接用clear();,空间复杂度是单层最大节点数。
如果要每次都释放空间,也可以用res.emplace_back(std::move(curLevel))
,涉及, , ,
class Solution: def maxProfit(self, prices: List[int]) -> int: def max_p(ps): if not ps or len(ps) == 1: return 0, 0, 0 very_low = 0 low = 0 high = 0 profit = 0 for i, p in enumerate(ps): if p < ps[low]: low = i elif p - ps[low] > profit: high = i very_low = low profit = p - ps[low] return very_low, high, profit low, high, profit = max_p(prices) _, _, profit_right = max_p(prices[0:low]) _, _, profit_left = max_p(prices[high+1:]) _, _, profit_middle = max_p(prices[low:high+1][::-1]) return profit + max(profit_left, profit_middle, profit_right)
def isPalindrome(self, s: str) -> bool: s = ''.join(i for i in s if i.isalnum()).lower() return s == s[::-1]
def singleNumber(self, nums: List[int]) -> int: seen_once = seen_twice = 0 for num in nums: # first appearance: # add num to seen_once, don't add to seen_twice because of presence in seen_once # second appearance: # remove num from seen_once, add num to seen_twice # third appearance: # don't add to seen_once because of presence in seen_twice, remove num from seen_twice seen_once = ~seen_twice & (seen_once ^ num) seen_twice = ~seen_once & (seen_twice ^ num) return seen_once
struct WTreeNode{TreeNode* TNode;bool tag;};vector postorderTraversal(TreeNode* root) { vector ret; stacks; TreeNode* p=root; if(root==NULL) return ret; WTreeNode* l; while(!s.empty()||p){ if (p!=NULL){ //左子树不断入栈 l=new WTreeNode; l->TNode=p;l->tag=0; s.push(l); p=p->left; } else{ l=s.top(); s.pop(); if(l->tag==1){ ret.push_back(l->TNode->val); } else{ //右子树没输出 l->tag=1; s.push(l); p=l->TNode->right; } } } return ret;}
while(!s.empty()||p){ if (p!=NULL){ } else{ }}
vector postorderTraversal1(TreeNode* root) { vector ret; stacks; vector invert; s.push(root); while(!s.empty()){ TreeNode *p=s.top();s.pop(); if(p!=NULL){ invert.push_back(p->val); s.push(p->left); s.push(p->right); } } reverse(invert.begin(),invert.end()); return invert;}
int findMin(vector & numbers) { //if(numbers.size()==0)return -1; int l=0,r=numbers.size()-1; int mid; while(lnumbers[r]) l=mid; else if(numbers[mid]
方法一:hashmap
方法二:随机算法,先取再验证
方法三::核心是利用这个数据的前缀特性,用军队打仗理解;每个非众数都会和一个数配对
int majorityElement(vector nums) { int solider = nums[0]; int count=0; for(int i=0;i
与两数之和联系的数据结构
uint32_t reverseBits(uint32_t n) { //注意定义uint32_t,如果是int要小心算术移位 unsigned int t = 0; int i = 0; while (n >= 1) { t = (t << 1) | (n & 1); i++; n = n >> 1; } if (i == 0) //细节:左移不要越界 return t; return t << (32 - i);}
def reverseByte(byte): return (byte * 0x0202020202 & 0x010884422010) % 1023
return n&1+hammingWeight(n>>=1);
class Solution {public: ListNode* deleteNode(ListNode* head, int val) { if(head==NULL)return NULL; if(head->val==val)return head->next; ListNode *p=head; while(p->next){ if(p->next->val==val){ p->next=p->next->next; break; } p=p->next; } return head; }};
ListNode* reverseList(ListNode* head) { if(!head||!head->next) return head; ListNode *p=head,*q=head->next; p->next=NULL; ListNode* temp; while(q!=NULL){ temp=q->next; q->next=p; p=q; q=temp; } return p;}
方法1: 快排
方法2: 利用小顶堆,保证size不大于k; C++中用priority_queue, 配合unordered_map
方法3: 快排变体,
引申:堆的实现
#define ElemType pairclass CMaxHeap { //小顶堆private: ElemType *heap; int heapSize, MaxHeapSize;public: CMaxHeap(int size) { heapSize = 0; MaxHeapSize = size; heap = new ElemType[size + 1]; } ~CMaxHeap() { delete[] heap; } void ClearHeap() { heapSize = 0; } bool IsEmpty() { return heapSize == 0; } bool IsFull() { return heapSize == MaxHeapSize; } int getLength() { return heapSize; } ElemType top() { return heap[0]; } void push(ElemType e); ElemType pop(); //去堆顶元素 void FixUp(int k); void FixDown(int k);};void CMaxHeap::FixDown(int k) { int i; i = 2 * k + 1; while (i < heapSize) { if (i < heapSize - 1 && heap[i] > heap[i + 1]) i++; //取孩子结点中较小者 if (heap[k] < heap[i]) break; swap(heap[k], heap[i]); k = i; i = 2 * k + 1; }}void CMaxHeap::FixUp(int k) { int i; i = (k - 1) / 2; while (k > 0 && heap[i] > heap[k]) { swap(heap[k], heap[i]); k = i; i = (k - 1) / 2; }}void CMaxHeap::push(ElemType e) { heap[heapSize] = e; heapSize++; FixUp(heapSize - 1);}ElemType CMaxHeap::pop() { //去掉堆顶 swap(heap[0], heap[heapSize - 1]); heapSize--; FixDown(0); return heap[heapSize];}void heap_sort(ElemType *a, int l, int r) {// int N = r - l; ElemType *p = a + l; for (int k = (N - 1) / 2; k >= 0; k--) FixDown(p, k, N); while (N > 0) { exch(p, p + N); N--; FixDown(p, 0, N); }}
templatevoid quick_sort(T*a, int l, int r) {//递归实现 if (r <= l)return; int i = partition(a, l, r);//划分操作 quick_sort(a, l, i - 1); quick_sort(a, i + 1, r);}template int partition(T*a, int l, int r) {//思想,从两边向中间扫描 int i = l - 1,j = r; T e = a[r];//最右端元素为划分元素 while (1) { while (a[++i] < e); while (e < a[--j])if (j == left)break; if (i >= j)break; exch(a + i, a + j); } exch(a + i, a + right); return i;}//改进:中间元素法和小序列处理template void quick_sort(T*a, int l, int r) {//递归实现 if (r-l< 5)return; exch(a[(l + r) / 2], a[r - 1]); compExch(a[l], a[r - 1]);//取三个值的中间值 compExch(a[l], a[r]); compExch(a[r], a[r - 1]); int i = partition(a, l, r);//划分操作 quick_sort(a, l, i - 1); quick_sort(a, i + 1, r);}
bool containsNearbyAlmostDuplicate(vector & nums, int k, int t) { int size = nums.size(); if(size <= 1) return false; if(k==0) return t==0; multisetwindow; //construct the first window for(int i=0; i & window, int val, int t) { auto itlower = window.lower_bound(val); if(itlower != window.end() && (*itlower)-val <= t) return true; if(itlower != window.begin()) { --itlower; if(val - (*itlower) <= t) return true; } return false;}
public class Solution { // Get the ID of the bucket from element value x and bucket width w // In Java, `-3 / 5 = 0` and but we need `-3 / 5 = -1`. private long getID(long x, long w) { return x < 0 ? (x + 1) / w - 1 : x / w; } public boolean containsNearbyAlmostDuplicate(int[] nums, int k, int t) { if (t < 0) return false; Mapd = new HashMap<>(); long w = (long)t + 1; for (int i = 0; i < nums.length; ++i) { long m = getID(nums[i], w); // check if bucket m is empty, each bucket may contain at most one element if (d.containsKey(m)) return true; // check the neighbor buckets for almost duplicate if (d.containsKey(m - 1) && Math.abs(nums[i] - d.get(m - 1)) < w) return true; if (d.containsKey(m + 1) && Math.abs(nums[i] - d.get(m + 1)) < w) return true; // now bucket m is empty and no almost duplicate in neighbor buckets d.put(m, (long)nums[i]); if (i >= k) d.remove(getID(nums[i - k], w)); } return false; }}
priority_queue<int, vector<int>, greater<int>> hi;
min.push_back(num);push_heap(min.begin(),min.end(),greater ());
class Codec {public: // Encodes a tree to a single string. string serialize(TreeNode* root) { if(root==NULL)return ""; ostringstream ostr; queueq; TreeNode*temp; q.push(root); int curNum=1; while(!q.empty()){ temp=q.front(); q.pop(); if(!temp){ if(curNum) ostr<<"null,"; } else { ostr< val<<","; curNum--; q.push(temp->left); if(temp->left)curNum++; q.push(temp->right); if(temp->right)curNum++; } } return ostr.str(); } // Decodes your encoded data to tree. TreeNode* deserialize(string data) { if(data=="")return NULL; istringstream istr(data); queue q; TreeNode* root=new TreeNode; TreeNode **number=new TreeNode*; if(ReadStream (istr,number)){ root=number[0]; if(!root)return NULL; q.push(root); } else return NULL; TreeNode *temp; while(!q.empty()){ temp=q.front(); q.pop(); if(!temp)continue; if(ReadStream(istr,number)){ temp->left=number[0]; q.push(temp->left); } else break; if(ReadStream(istr,number)){ temp->right=number[0]; q.push(temp->right); } else break; } return root; } bool ReadStream(istringstream &istr,TreeNode **number){ string s; if(getline(istr,s,',')){ if(s=="null")number[0]=NULL; else number[0]=new TreeNode(stoi(s)); return 1; } return 0; }};
Node *treeToDoublyList(Node *root){ root = treeToDoublyList(root, 0); if (root == NULL) return NULL; Node *p = root; while (p->right) p = p->right; p->right = root; root->left = p; return root;}Node *treeToDoublyList(Node *root, int flag){ //flag=0:left, flag=1:right if (root == NULL) return NULL; Node *l = treeToDoublyList(root->left, 1); Node *r = treeToDoublyList(root->right, 0); root->left = l; root->right = r; if (l) l->right = root; if (r) r->left = root; Node *p = root; if (!flag) while (p->left) p = p->left; else while (p->right) p = p->right; return p;}
class Solution {public: typedef TreeNode* Link; bool searchR(Link p,int x){ //判断子树内是否存在x if(p==NULL)return 0; int key=p->val; if(x==key)return 1; if(xleft,x); else return searchR(p->right,x); } bool findTarget(TreeNode* root, int k) { int result=0; if(root==NULL)return 0; if(k>2*root->val){ //利用BST特性减少运算量 if(findTarget(root->right,k))return 1; } else if(k<2*root->val){ if(findTarget(root->left,k))return 1; } if(searchR(root->left,k-root->val))return 1; if(searchR(root->right,k-root->val))return 1; if(findinAB(root->left,root->right,k)) return 1; return 0; } bool findinAB(TreeNode* A,TreeNode* B,int k){ //在A、B中各取一个节点 int result=0; if(A==NULL||B==NULL) return 0; if(searchR(B,k-A->val))return 1; if(findinAB(A->left,B,k))return 1; if(findinAB(A->right,B,k))return 1; return 0; }};
转载地址:http://drtv.baihongyu.com/