LeetCode 链表章节
简单
21. 合并两个有序链表
将两个升序链表合并为一个新的 升序 链表并返回。新链表是通过拼接给定的两个链表的所有节点组成的。
示例 1:
![]()
输入:l1 = [1,2,4], l2 = [1,3,4] 输出:[1,1,2,3,4,4]
示例 2:
输入:l1 = [], l2 = [] 输出:[]
示例 3:
输入:l1 = [], l2 = [0] 输出:[0]
提示:
- 两个链表的节点数目范围是
[0, 50]
-100 <= Node.val <= 100
l1
和l2
均按 非递减顺序 排列
递归,问题可以化为重复子问题,既不断合并两个链表,所以可以采用递归
ListNode* mergeTwoLists(ListNode* list1, ListNode* list2) {if (list1 == nullptr)return list2;if (list2 == nullptr)return list1;if (list1->val < list2->val) {list1->next = mergeTwoLists(list1->next, list2);return list1;} else {list2->next = mergeTwoLists(list1, list2->next);return list2; }
}
迭代
ListNode* mergeTwoLists(ListNode* list1, ListNode* list2) {ListNode* newhead = new ListNode;ListNode* cur = newhead;while (list1 && list2) {if (list1->val < list2->val) {cur->next = list1;list1 = list1->next;} else {cur->next = list2;list2 = list2->next;}cur = cur->next;}cur->next = (list1 ? list1 : list2);ListNode* ans = newhead->next;delete newhead;return ans;
}
141. 环形链表
给你一个链表的头节点
head
,判断链表中是否有环。如果链表中有某个节点,可以通过连续跟踪
next
指针再次到达,则链表中存在环。 为了表示给定链表中的环,评测系统内部使用整数pos
来表示链表尾连接到链表中的位置(索引从 0 开始)。注意:pos
不作为参数进行传递 。仅仅是为了标识链表的实际情况。如果链表中存在环 ,则返回
true
。 否则,返回false
。示例 1:
输入:head = [3,2,0,-4], pos = 1 输出:true 解释:链表中有一个环,其尾部连接到第二个节点。
示例 2:
输入:head = [1,2], pos = 0 输出:true 解释:链表中有一个环,其尾部连接到第一个节点。
示例 3:
输入:head = [1], pos = -1 输出:false 解释:链表中没有环。
提示:
- 链表中节点的数目范围是
[0, 10^4]
-10^5 <= Node.val <= 10^5
pos
为-1
或者链表中的一个 有效索引 。**进阶:**你能用
O(1)
(即,常量)内存解决此问题吗?
哈希
- 如果链表中存在环,链表中有重复的地址出现;如果没有环,则退出循环。
bool hasCycle(ListNode *head) {unordered_set<ListNode* > hash;while (head) {if (hash.count(head))return true;hash.insert(head);head = head->next;}return false;
}
快慢指针
- 如果链表中存在环,那么
fast
指针会先进入环,然后在环中不断循环。由于fast
指针移动速度比slow
指针快,最终fast
指针会追上slow
指针,即slow
指针和fast
指针会相遇。 - 如果链表中不存在环,那么
fast
指针会先到达链表的末尾(即fast->next
或fast->next->next
为nullptr
),此时循环结束,说明链表中不存在环。
bool hasCycle(ListNode *head) {if (head == nullptr)return false;ListNode* slow = head, *fast = head;while (fast->next && fast->next->next) {slow = slow->next;fast = fast->next->next;if (slow == fast)return true;}return false;
}
160. 相交链表
给你两个单链表的头节点
headA
和headB
,请你找出并返回两个单链表相交的起始节点。如果两个链表不存在相交节点,返回null
。图示两个链表在节点
c1
开始相交**:**
题目数据 保证 整个链式结构中不存在环。
注意,函数返回结果后,链表必须 保持其原始结构 。
自定义评测:
评测系统 的输入如下(你设计的程序 不适用 此输入):
intersectVal
- 相交的起始节点的值。如果不存在相交节点,这一值为0
listA
- 第一个链表listB
- 第二个链表skipA
- 在listA
中(从头节点开始)跳到交叉节点的节点数skipB
- 在listB
中(从头节点开始)跳到交叉节点的节点数评测系统将根据这些输入创建链式数据结构,并将两个头节点
headA
和headB
传递给你的程序。如果程序能够正确返回相交节点,那么你的解决方案将被 视作正确答案 。示例 1:
输入:intersectVal = 8, listA = [4,1,8,4,5], listB = [5,6,1,8,4,5], skipA = 2, skipB = 3 输出:Intersected at '8' 解释:相交节点的值为 8 (注意,如果两个链表相交则不能为 0)。 从各自的表头开始算起,链表 A 为 [4,1,8,4,5],链表 B 为 [5,6,1,8,4,5]。 在 A 中,相交节点前有 2 个节点;在 B 中,相交节点前有 3 个节点。 — 请注意相交节点的值不为 1,因为在链表 A 和链表 B 之中值为 1 的节点 (A 中第二个节点和 B 中第三个节点) 是不同的节点。换句话说,它们在内存中指向两个不同的位置,而链表 A 和链表 B 中值为 8 的节点 (A 中第三个节点,B 中第四个节点) 在内存中指向相同的位置。
示例 2:
输入:intersectVal = 2, listA = [1,9,1,2,4], listB = [3,2,4], skipA = 3, skipB = 1 输出:Intersected at '2' 解释:相交节点的值为 2 (注意,如果两个链表相交则不能为 0)。 从各自的表头开始算起,链表 A 为 [1,9,1,2,4],链表 B 为 [3,2,4]。 在 A 中,相交节点前有 3 个节点;在 B 中,相交节点前有 1 个节点。
示例 3:
输入:intersectVal = 0, listA = [2,6,4], listB = [1,5], skipA = 3, skipB = 2 输出:No intersection 解释:从各自的表头开始算起,链表 A 为 [2,6,4],链表 B 为 [1,5]。 由于这两个链表不相交,所以 intersectVal 必须为 0,而 skipA 和 skipB 可以是任意值。 这两个链表不相交,因此返回 null 。
提示:
listA
中节点数目为m
listB
中节点数目为n
1 <= m, n <= 3 * 10^4
1 <= Node.val <= 10^5
0 <= skipA <= m
0 <= skipB <= n
- 如果
listA
和listB
没有交点,intersectVal
为0
- 如果
listA
和listB
有交点,intersectVal == listA[skipA] == listB[skipB]
**进阶:**你能否设计一个时间复杂度
O(m + n)
、仅用O(1)
内存的解决方案?
哈希
- 找到第一个重复出现的地址。
ListNode *getIntersectionNode(ListNode *headA, ListNode *headB) {unordered_set<ListNode* > hash;while (headA) {hash.insert(headA);headA = headA->next;}while (headB) {if (hash.count(headB))return headB;headB = headB->next;}return nullptr;
}
双指针
-
指针
curA
先遍历链表A
的不相交部分,长度为a - c
;然后遍历相交部分,当遍历到相交部分末尾时,如果还没有和curB
相遇,它会跳到链表B
的头部继续遍历。此时它总共遍历的节点数为a - c + b - c
。 -
指针
curB
先遍历链表B
的不相交部分,长度为b - c
;然后遍历相交部分,当遍历到相交部分末尾时,如果还没有和curA
相遇,它会跳到链表A
的头部继续遍历。此时它总共遍历的节点数为b - c + a - c
。 -
如果没有相交节点,
c = 0
,curA
和curB
会在都为空时相遇,并退出循环。
图片原题解
ListNode *getIntersectionNode(ListNode *headA, ListNode *headB) {if (headA == nullptr || headB == nullptr)return nullptr;ListNode* curA = headA, * curB = headB;while (curA != curB) {curA = curA == nullptr ? headB : curA->next;curB = curB == nullptr ? headA : curB->next;}return curA;
}
206. 反转链表
给你单链表的头节点
head
,请你反转链表,并返回反转后的链表。示例 1:
![]()
输入:head = [1,2,3,4,5] 输出:[5,4,3,2,1]
示例 2:
![]()
输入:head = [1,2] 输出:[2,1]
示例 3:
输入:head = [] 输出:[]
提示:
- 链表中节点的数目范围是
[0, 5000]
-5000 <= Node.val <= 5000
**进阶:**链表可以选用迭代或递归方式完成反转。你能否用两种方法解决这道题?
迭代
ListNode* reverseList(ListNode* head) {ListNode* cur = head;ListNode* prev = nullptr;while (cur) {ListNode* next = cur->next;cur->next = prev;prev = cur;cur = next;}return prev;
}
递归
ListNode* reverseList(ListNode* head) {if (head == nullptr || head->next == nullptr)return head;ListNode* new_head = reverseList(head->next);head->next->next = head;head->next = nullptr;return new_head;
}
234. 回文链表
给你一个单链表的头节点
head
,请你判断该链表是否为回文链表如果是,返回true
;否则,返回false
。回文 序列是向前和向后读都相同的序列
示例 1:
![]()
输入:head = [1,2,2,1] 输出:true
示例 2:
![]()
输入:head = [1,2] 输出:false
提示:
- 链表中节点数目在范围
[1, 10^5]
内0 <= Node.val <= 9
**进阶:**你能否用
O(n)
时间复杂度和O(1)
空间复杂度解决此题?
数组模拟
bool isPalindrome(ListNode* head) {vector<int> nums;while (head) {nums.push_back(head->val);head = head->next;}int left = 0, right = nums.size() - 1;while (left < right) {if (nums[left] != nums[right])return false;left++;right--;}return true;
}
找到链表的中间节点,再逆置后比对


// 876. 链表的中间结点
ListNode* middleNode(ListNode* head) {ListNode* slow = head, * fast = head;while (fast && fast->next) {slow = slow->next;fast = fast->next->next;}return slow;
}// 206. 反转链表
ListNode* reverseList(ListNode* head) {ListNode* cur = head;ListNode* prev = nullptr;while (cur) {ListNode* next = cur->next;cur->next = prev;prev = cur;cur = next;}return prev;
}bool isPalindrome(ListNode* head) {ListNode* mid = middleNode(head);mid = reverseList(mid);while (mid) {if (head->val != mid->val)return false;head = head->next;mid = mid->next;}return true;
}
876. 链表的中间结点
给你单链表的头结点
head
,请你找出并返回链表的中间结点。如果有两个中间结点,则返回第二个中间结点。
示例 1:
输入:head = [1,2,3,4,5] 输出:[3,4,5] 解释:链表只有一个中间结点,值为 3 。
示例 2:
输入:head = [1,2,3,4,5,6] 输出:[4,5,6] 解释:该链表有两个中间结点,值分别为 3 和 4 ,返回第二个结点。
提示:
- 链表的结点数范围是
[1, 100]
1 <= Node.val <= 100
快慢指针
ListNode* middleNode(ListNode* head) {ListNode* slow = head, * fast = head;while (fast && fast->next) {slow = slow->next;fast = fast->next->next;}return slow;
}
中等
2. 两数相加
给你两个 非空 的链表,表示两个非负的整数。它们每位数字都是按照 逆序 的方式存储的,并且每个节点只能存储 一位 数字。
请你将两个数相加,并以相同形式返回一个表示和的链表。
你可以假设除了数字 0 之外,这两个数都不会以 0 开头。
示例 1:
输入:l1 = [2,4,3], l2 = [5,6,4] 输出:[7,0,8] 解释:342 + 465 = 807.
示例 2:
输入:l1 = [0], l2 = [0] 输出:[0]
示例 3:
输入:l1 = [9,9,9,9,9,9,9], l2 = [9,9,9,9] 输出:[8,9,9,9,0,0,0,1]
提示:
- 每个链表中的节点数在范围
[1, 100]
内0 <= Node.val <= 9
- 题目数据保证列表表示的数字不含前导零
根据题意,将链表中的两数相加即可;通过创建newhead,可以避免新链表判断是否有头节点。
ListNode* addTwoNumbers(ListNode* l1, ListNode* l2) {int t = 0, sum = 0;ListNode* newhead = new ListNode;ListNode* cur = newhead;ListNode* cur1 = l1,* cur2 = l2;while (cur1 && cur2) {sum = cur1->val + cur2->val + t;cur->next = new ListNode(sum % 10); cur = cur->next;t = sum / 10;cur1 = cur1->next;cur2 = cur2->next;}while (cur1) {sum = cur1->val + t;cur->next = new ListNode(sum % 10); cur = cur->next;t = sum / 10;cur1 = cur1->next;}while (cur2) {sum = cur2->val + t;cur->next = new ListNode(sum % 10); cur = cur->next; t = sum / 10;cur2 = cur2->next;}if (t)cur->next = new ListNode(t);return newhead->next;
}
对上面代码进行优化。
ListNode* addTwoNumbers(ListNode* l1, ListNode* l2) {int sum = 0;ListNode* newhead = new ListNode;ListNode* cur = newhead;ListNode* cur1 = l1,* cur2 = l2;while (cur1 || cur2 || sum) {if (cur1) {sum += cur1->val;cur1 = cur1->next;}if (cur2) {sum += cur2->val;cur2 = cur2->next;}cur->next = new ListNode(sum % 10); cur = cur->next;sum /= 10;}return newhead->next;
}
19. 删除链表的倒数第 N 个结点
给你一个链表,删除链表的倒数第
n
个结点,并且返回链表的头结点。示例 1:
![]()
输入:head = [1,2,3,4,5], n = 2 输出:[1,2,3,5]
示例 2:
输入:head = [1], n = 1 输出:[]
示例 3:
输入:head = [1,2], n = 1 输出:[1]
提示:
- 链表中结点的数目为
sz
1 <= sz <= 30
0 <= Node.val <= 100
1 <= n <= sz
**进阶:**你能尝试使用一趟扫描实现吗?
计算链表长度,转化为删除链表第(len - n)个节点。
ListNode* removeNthFromEnd(ListNode* head, int n) {int len = 0;ListNode* cur = head;while (cur) {cur = cur->next;len++;}int k = len - n;cur = head;ListNode* prev = nullptr;while (k--) {prev = cur;cur = cur->next;}if (prev == nullptr)return head->next;prev->next = cur->next;return head;
}
快慢指针,fast
指针快slow
指针(n + 1)个,当fast
为空时,slow
刚好为要删除的节点前一个
ListNode* removeNthFromEnd(ListNode* head, int n) {ListNode* new_head = new ListNode(0, head);ListNode* fast = head;ListNode* slow = new_head;while (n--) {fast = fast->next;}while (fast) {slow = slow->next;fast = fast->next;}slow->next = slow->next->next;ListNode* ans = new_head->next;delete new_head;return ans;
}
24. 两两交换链表中的节点
给你一个链表,两两交换其中相邻的节点,并返回交换后链表的头节点。你必须在不修改节点内部的值的情况下完成本题(即,只能进行节点交换)。
示例 1:
![]()
输入:head = [1,2,3,4] 输出:[2,1,4,3]
示例 2:
输入:head = [] 输出:[]
示例 3:
输入:head = [1] 输出:[1]
提示:
- 链表中节点的数目在范围
[0, 100]
内0 <= Node.val <= 100
递归
ListNode* swapPairs(ListNode* head) {if (head == nullptr || head->next == nullptr)return head;ListNode* newhead = head->next;// 先交换当前两个节点后面的节点,再交换当前两个节点head->next = swapPairs(newhead->next);newhead->next = head;return newhead;
}
迭代,将需要交换的左右节点,以及左节点的前节点和后节点定义,便于交换操作。
ListNode* swapPairs(ListNode* head) {if (head == nullptr || head->next == nullptr)return head;ListNode* newhead = new ListNode(0, head);ListNode* prev = newhead;ListNode* left = head;ListNode* right = head->next;ListNode* next = right->next;while (left && right) {left->next = next;right->next = left;prev->next = right;prev = left;left = prev->next;if (left)right = left->next;if (right)next = right->next;}return newhead->next;
}
138. 随机链表的复制
给你一个长度为
n
的链表,每个节点包含一个额外增加的随机指针random
,该指针可以指向链表中的任何节点或空节点。构造这个链表的 深拷贝。 深拷贝应该正好由
n
个 全新 节点组成,其中每个新节点的值都设为其对应的原节点的值。新节点的next
指针和random
指针也都应指向复制链表中的新节点,并使原链表和复制链表中的这些指针能够表示相同的链表状态。复制链表中的指针都不应指向原链表中的节点 。例如,如果原链表中有
X
和Y
两个节点,其中X.random --> Y
。那么在复制链表中对应的两个节点x
和y
,同样有x.random --> y
。返回复制链表的头节点。
用一个由
n
个节点组成的链表来表示输入/输出中的链表。每个节点用一个[val, random_index]
表示:
val
:一个表示Node.val
的整数。random_index
:随机指针指向的节点索引(范围从0
到n-1
);如果不指向任何节点,则为null
。你的代码 只 接受原链表的头节点
head
作为传入参数。示例 1:
输入:head = [[7,null],[13,0],[11,4],[10,2],[1,0]] 输出:[[7,null],[13,0],[11,4],[10,2],[1,0]]
示例 2:
输入:head = [[1,1],[2,1]] 输出:[[1,1],[2,1]]
示例 3:
输入:head = [[3,null],[3,0],[3,null]] 输出:[[3,null],[3,0],[3,null]]
提示:
0 <= n <= 1000
-10^4 <= Node.val <= 10^4
Node.random
为null
或指向链表中的节点。
递归回溯
// 旧节点对应新节点
unordered_map<Node*, Node*> cache_node;Node* copyRandomList(Node* head) {if (head == nullptr) return nullptr;if (!cache_node.count(head)) { // 未复制这个节点Node* new_node = new Node(head->val);cache_node[head] = new_node;new_node->next = copyRandomList(head->next);new_node->random = copyRandomList(head->random);}return cache_node[head];
}
迭代 + 节点拆分,对于链表 A→B→C
,我们可以将其拆分为 A→A ′ →B→B ′ →C→C ′
。对于任意一个原节点 S,其拷贝节点 S ′即为其后继节点。
Node* copyRandomList(Node* head) {if (head == nullptr)return nullptr;// 插入新节点到原链表中for (Node* cur = head; cur != nullptr; cur = cur->next->next) {Node* new_node = new Node(cur->val);new_node->next = cur->next;cur->next = new_node;}// 复制随机指针for (Node* cur = head; cur != nullptr; cur = cur->next->next) {Node* new_node = cur->next;new_node->random = (cur->random ? cur->random->next : nullptr);}// 拆分链表Node* new_head = head->next;for (Node* cur = head; cur != nullptr; cur = cur->next) {Node* new_node = cur->next;cur->next = cur->next->next;new_node->next = (new_node->next ? new_node->next->next : nullptr);}return new_head;
}
142. 环形链表 II
给定一个链表的头节点
head
,返回链表开始入环的第一个节点。 如果链表无环,则返回null
。如果链表中有某个节点,可以通过连续跟踪
next
指针再次到达,则链表中存在环。 为了表示给定链表中的环,评测系统内部使用整数pos
来表示链表尾连接到链表中的位置(索引从 0 开始)。如果pos
是-1
,则在该链表中没有环。注意:pos
不作为参数进行传递,仅仅是为了标识链表的实际情况。不允许修改 链表。
示例 1:
![]()
输入:head = [3,2,0,-4], pos = 1 输出:返回索引为 1 的链表节点 解释:链表中有一个环,其尾部连接到第二个节点。
示例 2:
输入:head = [1,2], pos = 0 输出:返回索引为 0 的链表节点 解释:链表中有一个环,其尾部连接到第一个节点。
示例 3:
输入:head = [1], pos = -1 输出:返回 null 解释:链表中没有环。
提示:
- 链表中节点的数目范围在范围
[0, 10^4]
内-105 <= Node.val <= 10^5
pos
的值为-1
或者链表中的一个有效索引**进阶:**你是否可以使用
O(1)
空间解决此题?
哈希
- 如果链表中存在环,链表中第一次重复出现的地址就是入环的第一个节点;如果没有环,则退出循环。
ListNode *detectCycle(ListNode *head) {unordered_set<ListNode* > hash;while (head) {if (hash.count(head))return head;hash.insert(head);head = head->next;}return nullptr;
}
快慢指针
设链表头节点到环入口节点的距离为 a
,环入口节点到快慢指针相遇节点的距离为 b
,相遇节点再到环入口节点的距离为 c
。
- 当快慢指针相遇时,
slow
指针走过的距离为a + b
,fast
指针走过的距离为a + b + k * (b + c)
(k
为fast
指针在环内绕的圈数,k >= 1
)。 - 由于
fast
指针速度是slow
指针的两倍,所以有2 * (a + b) = a + b + k * (b + c)
,化简可得a = (k - 1) * (b + c) + c
,既a
等于c
加上k - 1
圈的长度。 - 这意味着从链表头节点和快慢指针相遇节点同时出发,以相同速度前进,它们会在环的入口节点相遇。

ListNode *detectCycle(ListNode *head) {if (head == nullptr)return nullptr;ListNode* slow = head, *fast = head;while (fast->next && fast->next->next) {slow = slow->next;fast = fast->next->next;if (slow == fast) {while (head != slow) {head = head->next;slow = slow->next;}return slow;}}return nullptr;
}
143. 重排链表
给定一个单链表
L
的头节点head
,单链表L
表示为:L0 → L1 → … → Ln - 1 → Ln
请将其重新排列后变为:
L0 → Ln → L1 → Ln - 1 → L2 → Ln - 2 → …
不能只是单纯的改变节点内部的值,而是需要实际的进行节点交换。
示例 1:
输入:head = [1,2,3,4] 输出:[1,4,2,3]
示例 2:
输入:head = [1,2,3,4,5] 输出:[1,5,2,4,3]
提示:
- 链表的长度范围为
[1, 5 * 104]
1 <= node.val <= 1000
876. 链表的中间结点 + 206. 反转链表 + 合并链表
ListNode* reverseList(ListNode* head) {ListNode* prev = nullptr;ListNode* cur = head;while (cur) {ListNode* next = cur->next;cur->next = prev;prev = cur;cur = next;}return prev;
}ListNode* middleNode(ListNode* head) {ListNode* slow = head;ListNode* fast = head;while (fast && fast->next) {slow = slow->next;fast = fast->next->next;}return slow;
}void reorderList(ListNode* head) {ListNode* mid = middleNode(head);ListNode* l1 = head;ListNode* l2 = mid->next;mid->next = nullptr;l2 = reverseList(l2);// 合并while (l1 && l2) {ListNode* l1_tmp = l1->next;ListNode* l2_tmp = l2->next;l1->next = l2;l1 = l1_tmp;l2->next = l1;l2 = l2_tmp;}
}
146. LRU 缓存
请你设计并实现一个满足 LRU (最近最少使用) 缓存 约束的数据结构。
实现
LRUCache
类:
LRUCache(int capacity)
以 正整数 作为容量capacity
初始化 LRU 缓存int get(int key)
如果关键字key
存在于缓存中,则返回关键字的值,否则返回-1
。void put(int key, int value)
如果关键字key
已经存在,则变更其数据值value
;如果不存在,则向缓存中插入该组key-value
。如果插入操作导致关键字数量超过capacity
,则应该 逐出 最久未使用的关键字。函数
get
和put
必须以O(1)
的平均时间复杂度运行。示例:
输入 ["LRUCache", "put", "put", "get", "put", "get", "put", "get", "get", "get"] [[2], [1, 1], [2, 2], [1], [3, 3], [2], [4, 4], [1], [3], [4]] 输出 [null, null, null, 1, null, -1, null, -1, 3, 4]解释 LRUCache lRUCache = new LRUCache(2); lRUCache.put(1, 1); // 缓存是 {1=1} lRUCache.put(2, 2); // 缓存是 {1=1, 2=2} lRUCache.get(1); // 返回 1 lRUCache.put(3, 3); // 该操作会使得关键字 2 作废,缓存是 {1=1, 3=3} lRUCache.get(2); // 返回 -1 (未找到) lRUCache.put(4, 4); // 该操作会使得关键字 1 作废,缓存是 {4=4, 3=3} lRUCache.get(1); // 返回 -1 (未找到) lRUCache.get(3); // 返回 3 lRUCache.get(4); // 返回 4
提示:
1 <= capacity <= 3000
0 <= key <= 10000
0 <= value <= 105
- 最多调用
2 * 105
次get
和put
哈希表 + 双向链表
get
和put
方法 要求O(1)
的平均时间复杂度,所以想到用哈希表;用带伪头节点和伪尾节点双向链表来维护缓存,便于不断更新头节点,删除节点;
get
方法
- 尝试从哈希表中查找键为
key
的节点。 - 如果节点不存在,返回 -1。
- 如果节点存在,将该节点移动到双向链表的头部(表示最近使用),然后返回节点的值。
put
方法
-
如果键
key
不存在于缓存中:- 创建一个新的节点,并将其插入到双向链表的头部。
- 将该节点的信息存入哈希表。
- 如果缓存大小超过容量,删除双向链表的尾节点(即最近最少使用的节点),并从哈希表中移除对应的记录。
-
如果键
key
已经存在于缓存中:- 更新该节点的值。
- 将该节点移动到双向链表的头部
struct DLinkNode {int key, value;DLinkNode* next;DLinkNode* prev;DLinkNode() : key(0), value(0), next(nullptr), prev(nullptr) {}DLinkNode(int key_, int value_) : key(key_), value(value_), next(nullptr), prev(nullptr) {}
};class LRUCache {
private:DLinkNode* head_; // 伪头部DLinkNode* tail_; // 伪尾部unordered_map<int, DLinkNode*> cache_;int size_;int capacity_;
public:LRUCache(int capacity) : size_(0), capacity_(capacity) {head_ = new DLinkNode();tail_ = new DLinkNode();head_->next = tail_;tail_->prev = head_;}int get(int key) {if (!cache_.count(key))return -1;DLinkNode* node = cache_[key];moveToHead(node);return node->value;}void put(int key, int value) {if (!cache_.count(key)) { // key不存在,创建新节点DLinkNode* node = new DLinkNode(key, value);cache_[key] = node;addToHead(node);++size_;if (size_ > capacity_) { // 超出缓存容量,删除尾节点DLinkNode* removed = removeTail();cache_.erase(removed->key);delete removed;--size_;}} else { // key存在,更新value,移至头部DLinkNode* node = cache_[key];node->value = value;moveToHead(node);}}private:void removeNode(DLinkNode* node) {node->prev->next = node->next;node->next->prev = node->prev;}void addToHead(DLinkNode* node) {DLinkNode* head = head_->next;head->prev = node;node->next = head;head_->next = node;node->prev = head_;}void moveToHead(DLinkNode* node) {removeNode(node);addToHead(node);}DLinkNode* removeTail() {DLinkNode* node = tail_->prev;removeNode(node);return node;}
};
147. 对链表进行插入排序
给定单个链表的头
head
,使用 插入排序 对链表进行排序,并返回 排序后链表的头 。插入排序 算法的步骤:
- 插入排序是迭代的,每次只移动一个元素,直到所有元素可以形成一个有序的输出列表。
- 每次迭代中,插入排序只从输入数据中移除一个待排序的元素,找到它在序列中适当的位置,并将其插入。
- 重复直到所有输入数据插入完为止。
下面是插入排序算法的一个图形示例。部分排序的列表(黑色)最初只包含列表中的第一个元素。每次迭代时,从输入数据中删除一个元素(红色),并就地插入已排序的列表中。
对链表进行插入排序。
![]()
示例 1:
![]()
输入: head = [4,2,1,3] 输出: [1,2,3,4]
示例 2:
![]()
输入: head = [-1,5,3,4,0] 输出: [-1,0,3,4,5]
提示:
- 列表中的节点数在
[1, 5000]
范围内-5000 <= Node.val <= 5000
ListNode* insertionSortList(ListNode* head) {ListNode* new_head = new ListNode(0, head);ListNode* cur = head->next;ListNode* prev = head;while (cur) {if (cur->val < prev->val) {ListNode* pos = new_head;while (pos->next->val <= cur->val)pos = pos->next;prev->next = cur->next;cur->next = pos->next;pos->next = cur;}prev = cur;cur = cur->next;}ListNode* ans = new_head->next;delete new_head;return ans;
}
148. 排序链表
给你链表的头结点
head
,请将其按 升序 排列并返回 排序后的链表 。示例 1:
![]()
输入:head = [4,2,1,3] 输出:[1,2,3,4]
示例 2:
![]()
输入:head = [-1,5,3,4,0] 输出:[-1,0,3,4,5]
示例 3:
输入:head = [] 输出:[]
提示:
- 链表中节点的数目在范围
[0, 5 * 104]
内-105 <= Node.val <= 105
**进阶:**你可以在
O(n log n)
时间复杂度和常数级空间复杂度下,对链表进行排序吗?
归并排序,通过快慢指针找到中间节点,递归地对链表进行分治排序。
ListNode* merge(ListNode* list1, ListNode* list2) {ListNode* new_head = new ListNode;ListNode* cur = new_head;while (list1 && list2) {if (list1->val < list2->val) {cur->next = list1;list1 = list1->next;} else {cur->next = list2;list2 = list2->next;}cur = cur->next;}cur->next = (list1 ? list1 : list2);ListNode* head = new_head->next;delete new_head;return head;
}ListNode* sortList(ListNode* head, ListNode* tail) {if (head == nullptr)return head;if (head->next == tail) {head->next = nullptr;return head;}ListNode* slow = head, * fast = head;while (fast != tail && fast->next != tail) {slow = slow->next;fast = fast->next->next;}ListNode* mid = slow;return merge(sortList(head, mid), sortList(mid, tail));
}ListNode* sortList(ListNode* head) {return sortList(head, nullptr);
}
困难
23. 合并 K 个升序链表
给你一个链表数组,每个链表都已经按升序排列。
请你将所有链表合并到一个升序链表中,返回合并后的链表。
示例 1:
输入:lists = [[1,4,5],[1,3,4],[2,6]] 输出:[1,1,2,3,4,4,5,6] 解释:链表数组如下: [1->4->5,1->3->4,2->6 ] 将它们合并到一个有序链表中得到。 1->1->2->3->4->4->5->6
示例 2:
输入:lists = [] 输出:[]
示例 3:
输入:lists = [[]] 输出:[]
提示:
k == lists.length
0 <= k <= 10^4
0 <= lists[i].length <= 500
-10^4 <= lists[i][j] <= 10^4
lists[i]
按 升序 排列lists[i].length
的总和不超过10^4
暴力解法
根据21. 合并两个有序链表这题,不断合并所有链表。
ListNode* mergeTwoLists(ListNode* list1, ListNode* list2) {ListNode* newhead = new ListNode;ListNode* cur = newhead;while (list1 && list2) {if (list1->val < list2->val) {cur->next = list1;list1 = list1->next;} else {cur->next = list2;list2 = list2->next;}cur = cur->next;}cur->next = (list1 ? list1 : list2);return newhead->next;
}ListNode* mergeKLists(vector<ListNode*>& lists) {ListNode* ans = nullptr;for (ListNode* list : lists)ans = mergeTwoLists(ans, list);return ans;
}
分治合并
和归并排序的思想一样,不断将链表的头节点分成两部分,先分别对这两部分合并,再将这两部分合并。
ListNode* mergeTwoLists(ListNode* list1, ListNode* list2) {ListNode* new_head = new ListNode;ListNode* cur = new_head;while (list1 && list2) {if (list1->val < list2->val) {cur->next = list1;list1 = list1->next;} else {cur->next = list2;list2 = list2->next;}cur = cur->next;}cur->next = (list1 ? list1 : list2);return new_head->next;
}ListNode* merge(vector<ListNode*>& lists, int left, int right) {if (left > right) return nullptr;if (left == right) return lists[left];int mid = left + (right - left) / 2;return mergeTwoLists(merge(lists, left, mid), merge(lists, mid + 1, right));
}ListNode* mergeKLists(vector<ListNode*>& lists) {int n = lists.size();return merge(lists, 0, n - 1);
}
优先级队列
借助优先级队列找到最小的未被合并的头节点,不断合并。
struct cmp {bool operator()(ListNode* list1, ListNode* list2) {return list1->val > list2->val;}
};ListNode* mergeKLists(vector<ListNode*>& lists) {// 建立小根堆priority_queue<ListNode*, vector<ListNode*>, cmp> heap;// 所有头节点入队for (ListNode* list : lists)if (list)heap.push(list);// 合并k个有序链表ListNode* new_head = new ListNode;ListNode* cur = new_head;while (!heap.empty()) {ListNode* t = heap.top();heap.pop();if (t->next)heap.push(t->next);cur->next = t;cur = cur->next;}return new_head->next;
}
25. K 个一组翻转链表
给你链表的头节点
head
,每k
个节点一组进行翻转,请你返回修改后的链表。
k
是一个正整数,它的值小于或等于链表的长度。如果节点总数不是k
的整数倍,那么请将最后剩余的节点保持原有顺序。你不能只是单纯的改变节点内部的值,而是需要实际进行节点交换。
示例 1:
![]()
输入:head = [1,2,3,4,5], k = 2 输出:[2,1,4,3,5]
示例 2:
![]()
输入:head = [1,2,3,4,5], k = 3 输出:[3,2,1,4,5]
提示:
- 链表中的节点数目为
n
1 <= k <= n <= 5000
0 <= Node.val <= 1000
**进阶:**你可以设计一个只用
O(1)
额外内存空间的算法解决此问题吗?
模拟
每k个节点进行一次206. 反转链表,并记录新的头和尾用于整个链表的连接。
pair<ListNode*, ListNode*> reverseList(ListNode* head, ListNode* tail) {ListNode* prev = tail->next;ListNode* cur = head;while (prev != tail) {ListNode* next = cur->next;cur->next = prev;prev = cur;cur = next;}return {tail, head};
}ListNode* reverseKGroup(ListNode* head, int k) {ListNode* new_head = new ListNode(0, head);ListNode* cur = new_head;while (cur) {ListNode* tail = cur;for (int i = 0; i < k; ++i) {tail = tail->next;if (tail == nullptr)return new_head->next;}auto p = reverseList(cur->next, tail);head = p.first;tail = p.second;cur->next = head;cur = tail;}return new_head->next;
}
k
是一个正整数,它的值小于或等于链表的长度。如果节点总数不是k
的整数倍,那么请将最后剩余的节点保持原有顺序。你不能只是单纯的改变节点内部的值,而是需要实际进行节点交换。
示例 1:
![]()
输入:head = [1,2,3,4,5], k = 2 输出:[2,1,4,3,5]
示例 2:
![]()
输入:head = [1,2,3,4,5], k = 3 输出:[3,2,1,4,5]
提示:
- 链表中的节点数目为
n
1 <= k <= n <= 5000
0 <= Node.val <= 1000
**进阶:**你可以设计一个只用
O(1)
额外内存空间的算法解决此问题吗?
模拟
每k个节点进行一次206. 反转链表,并记录新的头和尾用于整个链表的连接。
pair<ListNode*, ListNode*> reverseList(ListNode* head, ListNode* tail) {ListNode* prev = tail->next;ListNode* cur = head;while (prev != tail) {ListNode* next = cur->next;cur->next = prev;prev = cur;cur = next;}return {tail, head};
}ListNode* reverseKGroup(ListNode* head, int k) {ListNode* new_head = new ListNode(0, head);ListNode* cur = new_head;while (cur) {ListNode* tail = cur;for (int i = 0; i < k; ++i) {tail = tail->next;if (tail == nullptr)return new_head->next;}auto p = reverseList(cur->next, tail);head = p.first;tail = p.second;cur->next = head;cur = tail;}return new_head->next;
}
相关文章:
LeetCode 链表章节
简单 21. 合并两个有序链表 将两个升序链表合并为一个新的 升序 链表并返回。新链表是通过拼接给定的两个链表的所有节点组成的。 示例 1: 输入:l1 [1,2,4], l2 [1,3,4] 输出:[1,1,2,3,4,4]示例 2: 输入:l1 [], l2…...
JavaScript实现倒计时函数
函数代码 /*** 倒计时* param {function} callback 回调函数,参数为当前剩余时间(time)* param {number} count 倒计时开始时间(s)* param {number} interval 间隔时间(ms)*/ function countDown(callback, count 60, interval 1000) {callback(count);const sta…...
Spring Boot全局异常处理:“危机公关”团队
目录 一、全局异常处理的作用二、Spring Boot 实现全局异常处理(附上代码实例)三、总结: 🌟我的其他文章也讲解的比较有趣😁,如果喜欢博主的讲解方式,可以多多支持一下,感谢…...
Vue 调用摄像头扫描条码
以下是一个基于 Vue.js 的页面代码示例,用于调用摄像头并扫描条码。我们将使用 jsQR 库来解析二维码(或条形码),这是一个轻量级的 JavaScript 库。 实现步骤: 安装依赖:需要引入 jsQR 库。调用摄像头&…...
springboot3.x下集成hsqldb数据库
springboot3.x下集成hsqldb数据库 本文使用目前最新的sringboot3.4.3 和HyperSQL 2.7.4演示 HSQLDB数据库简介 HSQLDB(HyperSQL DataBase)是一个开放源代码的JAVA数据库。 可以透过 jdbc driver 来存取, 支持 ANSI-92 标准的 SQL 语法, 而且他占的空…...
网络流算法: Edmonds-Karp算法
图论相关帖子 基本概念图的表示: 邻接矩阵和邻接表图的遍历: 深度优先与广度优先拓扑排序图的最短路径:Dijkstra算法和Bellman-Ford算法最小生成树二分图多源最短路径强连通分量欧拉回路和汉密尔顿回路网络流算法: Edmonds-Karp算法网络流算法: Dinic算法 环境要求 本文所用…...
ArcGIS Pro可见性分析:精通地形视线与视域分析
在地理信息系统(GIS)的广泛应用中,可见性分析作为一项关键技术,发挥着不可替代的作用。 无论是城市规划、环境监测,还是军事侦察、景观设计,可见性分析都能提供精确的数据支持,帮助我们更好地理…...
jenkens使用笔记
jenkens使用笔记 笔记使用版本是2.492.1 git仓库ssh证书配置 已开始配置一直不行,然后下载插件,多次重启等一些列操作, 后来配置就可以工作了,原因不祥,不知道哪个配置起效了。 等回来闹明白了,再补充笔记…...
解决跨域请求的问题(CORS)
目录 解决跨域请求问题的方法 1. 服务器端配置响应头 2. JSONP(JSON with Padding) 3. 代理服务器 场景示例 前端代码(使用 Fetch API) 后端代码(使用 Node.js Express 并设置 CORS 响应头) 跨域资…...
未来经济范式争夺战:AR眼镜为何成为下一代交互终端的制高点?
未来经济范式争夺战:AR眼镜为何成为下一代交互终端的制高点? 在蒸汽机轰鸣的工业革命时代,煤炭、铁路、电报构建了第一个现代经济范式;互联网时代,电力、光纤、物流网络重构了全球经济版图。当前,我们正站…...
CentOS 7 安装Nginx-1.26.3
无论安装啥工具、首先认准了就是官网。Nginx Nginx官网下载安装包 Windows下载: http://nginx.org/download/nginx-1.26.3.zipLinxu下载 wget http://nginx.org/download/nginx-1.26.3.tar.gzLinux安装Nginx-1.26.3 安装之前先安装Nginx依赖包、自行选择 yum -y i…...
基于opencv消除图片马赛克
以下是一个基于Python的图片马赛克消除函数实现,结合了图像处理和深度学习方法。由于马赛克消除涉及复杂的图像重建任务,建议根据实际需求选择合适的方法: import cv2 import numpy as np from PIL import Imagedef remove_mosaic(image_pat…...
MongoDB Compass中MONGOSH常用查询整理
MongoDB Compass中MONGOSH常用查询整理 选择数据库基本的查找指令find() 方法findOne() 方法 高级查询条件比较操作符逻辑操作符投影操作排序操作限制和跳过操作limit() 方法skip() 方法 正则表达式查询数组查询 MongoDB Compass 是一款可视化的 MongoDB 数据库管理工具&#x…...
SSM笔记
一、获取对象 Scop 单例在容器启动时就直接创建,如果不希望这样,那就使用Lazy懒加载,只能在单例模式下 3、4不常用 FactoryBean创建 对象 创建对象比较复杂时,可以实现创建一个类实现FactoryBean,实现3个方法来创建…...
5G学习笔记之BWP
我们只会经历一种人生,我们选择的人生。 参考:《5G NR标准》、《5G无线系统指南:如微见著,赋能数字化时代》 目录 1. 概述2. BWP频域位置3. 初始与专用BWP4. 默认BWP5. 切换BWP 1. 概述 在LTE的设计中,默认所有终端均能处理最大2…...
MongoDB Chunks核心概念与机制
1. 基础定义 Chunk(块):MongoDB分片集群中数据的逻辑存储单元,由一组连续的片键(Shard Key)范围数据组成,默认大小为64MB(可调整范围为1-1024MB)。数据分…...
el-table 手动选择展示列
需求: 由于表格的列过多,用滚动条进行滚动对比数据不方便,所以提出,手动选择展示列 实现思路: 表格默认展示所有字段,每个字段通过 v-if 属性来进行判断是否显示;点击设置按钮图标(表格右上角࿰…...
Netty笔记3:NIO编程
Netty笔记1:线程模型 Netty笔记2:零拷贝 Netty笔记3:NIO编程 Netty笔记4:Epoll Netty笔记5:Netty开发实例 Netty笔记6:Netty组件 Netty笔记7:ChannelPromise通知处理 Netty笔记8…...
LeetCode Hot 100
1.两数之和 暴力解法:时间/空间复杂度 O(N),O(1) class Solution {public int[] twoSum(int[] nums, int target) {for(int i0;i<nums.length;i){for(int ji1;j<nums.length;j){if(nums[i] nums[j] target){return new int[]{i,j};}}}return new int[0];}…...
Vue.js 学习笔记
文章目录 前言一、Vue.js 基础概念1.1 Vue.js 简介1.2 Vue.js 的特点1.3 Vue.js 基础示例 二、Vue.js 常用指令2.1 双向数据绑定(v-model)2.2 条件渲染(v-if 和 v-show)2.3 列表渲染(v-for)2.4 事件处理&am…...
MySQL表连接详解
MySQL表连接详解 在 MySQL 中,表连接(Join)用于将多个表中的数据组合在一起,基于它们之间的关系进行查询。常见的表连接类型包括内连接、左连接、右连接和全外连接。以下是这些连接类型的详细说明: 1. 内连接&#x…...
【JAVA】ThreadPoolTaskExecutor 线程池学习、后端异步、高并发处理
ThreadPoolTaskExecutor 是 Spring 框架提供的一个线程池实现类,基于 Java 原生的 ThreadPoolExecutor 进行了封装和扩展,支持更灵活的配置,并与 Spring 的依赖注入、生命周期管理等功能无缝集成。它常用于异步任务处理、定时任务调度和高并发…...
PPT 小黑第38套
对应大猫40 幻灯片母板-最后一页-重命名为奇数页 奇偶页-点中标题-形状格式-形状填充-青色 最后一页页码左对齐 更换幻灯片背景:设计-设置背景格式-图片填充 【开始】-段落居中,对齐文本-中部对齐,排列-对齐-底端,-再水平居中…...
安卓开发相机功能
相机功能 安卓中的相机调用功能也经历了很多的方案升级,目前可选的官方方案是CameraX、Camera2、Camera(废弃),还有一些第三方免费或者是付费的相机库。对于大多数开发者,建议使用 CameraX。 CameraX CameraX 是 An…...
宝塔找不到php扩展swoole,服务器编译安装
1. 在php7.4中安装swoole,但找不到这个扩展安装 2. 服务器下载源码解压安装 http://pecl.php.net/package/swoole 下载4.8.0版本 解压到/www/server/php/74/下 3. 发现报错问题; 更新一下依赖 yum update yum -y install gcc gcc-c autoconf libjpe…...
Spring Web MVC
前言 今天来复习 Spring Web MVC 框架。它提供了一套高效、便捷的方式来构建 Web 应用程序。今天,就让我们一同深入 Spring Web MVC,从基础概念到实际应用,好好补补. 一、Spring Web MVC 是什么? 官方定义解读 根据官方描述&…...
蓝桥杯备考:动态规划线性dp之下楼梯问题进阶版
老规矩,按照dp题的顺序 step1 定义状态表达 f[i]表示到第i个台阶的方案数 step2:推导状态方程 step3:初始化 初始化要保证 1.数组不越界 2.推导结果正确 如图这种情况就越界了,我们如果把1到k的值全初始化也不现实,会增加程序的时间复杂度…...
机器视觉开发教程——封装Halcon通用模板匹配工具【含免费教程源码】
目录 引言前期准备Step1 设计可序列化的输入输出集合【不支持多线程】Step2 设计程序框架1、抽象层【IProcess】2、父类【HAlgorithm】3、子类【HFindModelTool】 Step3 设计UI结果展示 引言 通过仿照VisionPro软件二次开发Halcon的模板匹配工具,便于在客户端软件中…...
UDP透传程序
UDP透传程序 本脚本用于在 设备 A 和 设备 B 之间建立 UDP 数据转发桥梁,适用于 A 和 B 设备无法直接通信的情况。 流程: A --> 电脑 (中继) --> B B --> 电脑 (中继) --> A 需要修改参数: B_IP “192.168.1.123” # 设备 B 的…...
【USRP】NVIDIA Sionna:用于 6G 物理层研究的开源库
目录 Sionna:用于 6G 物理层研究的开源库主要特点实现6G研究的民主化支持 5G、6G 等模块化、可扩展、可伸缩快速启动您的研究 好处原生人工智能支持综合研究平台开放生态系统 安装笔记使用 pip 安装基于Docker的安装从源代码安装“你好世界!”探索锡奥纳…...
Spring WebFlux 中 WebSocket 使用 DataBuffer 的注意事项
以下是修改后的完整文档,包含在多个多线程环境中使用 retain() 和 release() 方法的示例,且确保在 finally 块中调用 release(): 在 Spring WebFlux 中,WebSocketMessage 主要用于表示 WebSocket 的消息载体,其中 getP…...
SQL经典常用查询语句
1. 基础查询语句 1.1 查询表中所有数据 在SQL中,查询表中所有数据是最基本的操作之一。通过使用SELECT * FROM table_name;语句,可以获取指定表中的所有记录和列。例如,假设有一个名为employees的表,包含员工的基本信息…...
0005__PyTorch 教程
PyTorch 教程 | 菜鸟教程 离线包:torch-1.13.1cpu-cp39-cp39-win_amd64.whl https://download.pytorch.org/whl/torch_stable.html...
高并发场景下的数据库优化
在高并发系统中,数据库通常是性能瓶颈。面对高并发请求,我们需要采用合适的优化策略,以保证数据库的稳定性和高效性。本文将介绍数据库高并发问题的成因,并结合 Mybatis-Plus,探讨 乐观锁、悲观锁、高并发优化及数据库…...
Linux:同步
目录 一、同步概念 条件变量 二、生产者消费者模型 三、环形队列 一、同步概念 互斥用来解决 访问临界资源 的非原子性,通俗来说,由于互斥锁的实现,保证了在用户角度看,同一个时间内访问临界资源的代码只有一个线程在执行。 而…...
GB28181开发--ZLMediaKit+WVP+Jessibuca
一、核心组件功能 1、ZLMediaKit 定位:基于 C++11 的高性能流媒体服务框架,支持 RTSP/RTMP/HLS/HTTP-FLV 等协议互转,具备低延迟(最低 100ms)、高并发(单机 10W 级连接)特性,适用于商用级流媒体服务器部署。 特性:跨平台(Linux/Windows/ARM 等)、支持 …...
23种设计模式之《备忘录模式(Memento)》在c#中的应用及理解
程序设计中的主要设计模式通常分为三大类,共23种: 1. 创建型模式(Creational Patterns) 单例模式(Singleton):确保一个类只有一个实例,并提供全局访问点。 工厂方法模式࿰…...
Oracle删除重复数据保留其中一条
Oracle删除重复数据保留其中一条 在Oracle数据库中,要删除重复数据并保留其中一条记录,可以使用多种方法。这里介绍两种常见的方法:使用ROWID或使用ROW_NUMBER()窗口函数。 方法1:使用ROWID ROWID是Oracle中用来唯一标识表中每…...
deepseek助力运维和监控自动化
将DeepSeek与Agent、工作流及Agent编排技术结合,可实现IT运维与监控的智能化闭环管理。以下是具体应用框架和场景示例: 一、智能Agent体系设计 多模态感知Agent 日志解析Agent:基于DeepSeek的NLP能力,实时解析系统日志中的语义&a…...
16.1STM32_ADC
STM32_ADC 数字信号分为高/低电平两种状态 模拟信号就是任意的电压值 STM32芯片内就是一整套的数字逻辑电路,来实现我们的程序执行,以及各种各样的外设功能, ADC(模拟-数字转换技术)的功能就是将模拟信号转化为数字…...
神经网络 - 激活函数(Swish函数、GELU函数)
一、Swish 函数 Swish 函数是一种较新的激活函数,由 Ramachandran 等人在 2017 年提出,其数学表达式通常为 其中 σ(x) 是 Sigmoid 函数(Logistic 函数)。 如何理解 Swish 函数 自门控特性 Swish 函数可以看作是对输入 x 进行“…...
VS2015 c++和cmake配置编程
Visual Studio 2015:确保安装了C开发工具,并安装“使用C的桌面开发”工作负载。CMake:可以从 CMake官网 下载并安装,并将其添加到系统环境变量中。vs加载项目启动Visual Studio。选择“继续但无代码”。点击“文件”。选择 “打开…...
如何为 Web 前端开发面试做好准备
大家好!我是 [数擎AI],一位热爱探索新技术的前端开发者,在这里分享前端和Web3D、AI技术的干货与实战经验。如果你对技术有热情,欢迎关注我的文章,我们一起成长、进步! 开发领域:前端开发 | AI 应…...
深入探索像ChatGPT这样的大语言模型
参考 【必看珍藏】2月6日,安德烈卡帕西最新AI普及课:深入探索像ChatGPT这样的大语言模型|Andrej Karpathy fineweb知乎翻译介绍 fineweb-v1原始连接 fineweb中文翻译版本 Chinese Fineweb Edu数据集 查看网络的内部结果,可以参…...
代码贴——堆(二叉树)数据结构
头文件Heap.h #pragma once #include<bits/stdc.h> typedef int HPDataType;typedef struct Heap {HPDataType* a;int size;int capacity; }HP;void HPInit(HP* php); void HPDestory(HP* php); //出入后保持数据是堆 void HPPush(HP* php,HPDataType x); HPDataType HP…...
office或者word排版中,复制/黏贴进来文字不会自动换行,如何处理?
李升伟 整理 一、思考与分析 在Office或Word中复制粘贴文字时,文字不会自动换行,需要处理这个问题。首先,我得回想一下常见的原因和解决方法。可能的情况有很多,比如文本带有硬回车、段落格式设置问题,或者文本框的自…...
最新!!!DeepSeek开源周发布内容汇总
本周,人工智能领域的新锐力量DeepSeek宣布将于本周举办“开源周”(Open Source Week),连续五天每日开源一个核心代码库,以透明的方式与全球开发者分享其在通用人工智能(AGI)探索中的最新成果。以…...
【MySQL】(2) 库的操作
SQL 关键字,大小写不敏感。 一、查询数据库 show databases; 注意加分号,才算一句结束。 二、创建数据库 {} 表示必选项,[] 表示可选项,| 表示任选其一。 示例:建议加上 if not exists 选项。 三、字符集编码和排序…...
记一次渗透测试实战:SQL注入漏洞的挖掘与利用
0x01 漏洞发现 在对某网站进行安全测试时,发现以下URL存在异常: https://******.com/search.php?keyword1&zt1954&dw1885&zz& 当参数keyword和zt被赋值为-1时页面返回特殊内容,初步判断存在SQL注入漏洞。 0x02 注入验证…...
Gin框架从入门到实战:核心用法与最佳实践
为什么选择Gin框架? Gin 是一个基于 Go 语言的高性能 Web 框架,具备以下优势: 轻量高效:底层依赖 net/http,性能接近原生。简洁优雅:API 设计友好,支持路由分组、中间件链、参数绑定等特性。生…...