LeetCode382-链表随机节点
原题链接:【382. 链表随机节点】:https://leetcode-cn.com/problems/linked-list-random-node/
题目描述:
给定一个单链表,随机选择链表的一个节点,并返回相应的节点值。保证每个节点被选的概率一样。
进阶:
如果链表十分大且长度未知,如何解决这个问题?你能否使用常数级空间复杂度实现?
示例:
// 初始化一个单链表 [1,2,3].
ListNode head = new ListNode(1);
head.next = new ListNode(2);
head.next.next = new ListNode(3);
Solution solution = new Solution(head); // getRandom()方法应随机返回1,2,3中的一个,保证每个元素被返回的概率相等。
solution.getRandom();
相关知识点:水塘抽样 链表 数学 随机化
解题思路:
一般的想法就是,我先遍历一遍链表,得到链表的总长度 n,再生成一个 [1,n] 之间的随机数为索引,然后再遍历链表,找到索引对应的节点,不就是一个随机的节点了吗?
但是由题目可知,链表十分大,且长度未知,也就说,我们不知道链表长度n,也就没办法取中间的随机数,那么我们应该怎么去取值呢?
由于是链表,所以我们可以一个节点一个节点遍历加载进入内存。虽然我们无法知道链表总长度,但是我们可以知道我们当前遍历的节点的长度i。
比如我们已经遍历了i个元素,可以从这i个元素中随机取了一个元素a,那如果现在再给你一个新元素b,你是继续留着a呢?还是抽取b作为样本结果呢?以什么样的原则来选择a和b呢?你的选择能够保证概率相等吗?
这里就用到了著名的蓄水池抽样算法。
蓄水池抽样算法
假设给定一个数据流,数据流长度N很大,如何从中随机选取k个数据,并且要保证每个数据被抽取到的概率相等。
当k = 1,只取一个元素时,要保证每条数据被抽取到的概率相等,那么每个数被抽取的概率应该为1/N,只要采取这种策略,只需要遍历一遍数据流就可以得到采样值,并且保证所有数据被选中的概率均为1/N
当k > 1,取k个元素时,要保证每条数据被抽取到的概率相等,那么每个数被抽取的概率应该为k / N。
算法实现思路为:
一个一个遍历数据流,在取第i个数据的时候,生成一个0到1的随机数p,如果p < k / i,替换池中任意一个数替换为第i个数;当p > k / i,继续保留前面的数。直到数据流结束,返回此k个数。但是为了保证计算准确性,一般是生成一个0到i的随机数,跟k相比。
图解如下:

本题题解:
注意这里其实就是K=1的情况,取随机数rand,范围为【0,i),随机数+1,变成【1,i】范围,两边都是闭合的,更容易理解。所以,原来的小于k,即小于1,现在+1,四舍五入都变成了1,所以边界判断范围就是,随机数rand=1,则样本替换为当前遍历的节点,否则保留之前的样本节点,继续往下遍历。
其余就都是链表的操作了。
代码实现如下:
//给定一个单链表,随机选择链表的一个节点,并返回相应的节点值。保证每个节点被选的概率一样。
//
// 进阶:
//如果链表十分大且长度未知,如何解决这个问题?你能否使用常数级空间复杂度实现?
//
// 示例:
//
//
//// 初始化一个单链表 [1,2,3].
//ListNode head = new ListNode(1);
//head.next = new ListNode(2);
//head.next.next = new ListNode(3);
//Solution solution = new Solution(head);
//
//// getRandom()方法应随机返回1,2,3中的一个,保证每个元素被返回的概率相等。
//solution.getRandom();
//
// Related Topics 水塘抽样 链表 数学 随机化
// 172 0 //leetcode submit region begin(Prohibit modification and deletion) import java.util.Random; /**
* Definition for singly-linked list.
* public class ListNode {
* int val;
* ListNode next;
* ListNode() {}
* ListNode(int val) { this.val = val; }
* ListNode(int val, ListNode next) { this.val = val; this.next = next; }
* }
*/
class Solution { // 需要将链表传递给getRandom方法,所以只能提取作为类变量
ListNode head;
public Solution(ListNode head) {
this.head = head;
} public int getRandom() {
// 定义一个变量,存储采样的结果值
int result = 0;
// 定义一个变量i,标记遍历到了第几个节点
int i = 0;
// 将当前链表head指针赋值给cur
ListNode cur = head;
// 循环遍历链表
while (cur != null) {
i++;
// 取随机数rand 范围 [1, i]
int rand = new Random().nextInt(i) + 1;
// 因为rand最小值为1,这个边界只能取rand = 1
if (rand == 1) {
result = cur.val;
}
// 指针往后移动,遍历下一个节点
cur = cur.next;
} // 返回采样结果
return result;
}
} /**
* Your Solution object will be instantiated and called as such:
* Solution obj = new Solution(head);
* int param_1 = obj.getRandom();
*/
//leetcode submit region end(Prohibit modification and deletion)
LeetCode382-链表随机节点的更多相关文章
- [Swift]LeetCode382. 链表随机节点 | Linked List Random Node
		
Given a singly linked list, return a random node's value from the linked list. Each node must have t ...
 - Java实现 LeetCode 382 链表随机节点
		
382. 链表随机节点 给定一个单链表,随机选择链表的一个节点,并返回相应的节点值.保证每个节点被选的概率一样. 进阶: 如果链表十分大且长度未知,如何解决这个问题?你能否使用常数级空间复杂度实现? ...
 - [LeetCode] Linked List Random Node 链表随机节点
		
Given a singly linked list, return a random node's value from the linked list. Each node must have t ...
 - 382 Linked List Random Node 链表随机节点
		
给定一个单链表,随机选择链表的一个节点,并返回相应的节点值.保证每个节点被选的概率一样.进阶:如果链表十分大且长度未知,如何解决这个问题?你能否使用常数级空间复杂度实现?示例:// 初始化一个单链表 ...
 - [LeetCode] 382. Linked List Random Node 链表随机节点
		
Given a singly linked list, return a random node's value from the linked list. Each node must have t ...
 - [CareerCup] 2.3 Delete Node in a Linked List 删除链表的节点
		
2.3 Implement an algorithm to delete a node in the middle of a singly linked list, given only access ...
 - 【链表】在O(1)的时间删除链表的节点
		
/** * 在O(1)的时间删除链表的节点 * * @author * */ public class Solution { public static void deleteNode(Node he ...
 - 剑指Offer:删除链表的节点【18】
		
剑指Offer:删除链表的节点[18] 题目描述 在一个排序的链表中,存在重复的结点,请删除该链表中重复的结点,重复的结点不保留,返回链表头指针. 例如,链表1->2->3->3-& ...
 - (CSDN迁移) 输入一个链表,从尾到头打印链表每个节点的值
		
题目描述 输入一个链表,从尾到头打印链表每个节点的值. 思路1. 翻转链表,使用java自带的翻转函数或者从头到尾依次改变链表的节点指针 /** * public class ListNode { * ...
 
随机推荐
- hdu 4521 小明序列(线段树,DP思想)
			
题意: ①首先定义S为一个有序序列,S={ A1 , A2 , A3 , ... , An },n为元素个数 : ②然后定义Sub为S中取出的一个子序列,Sub={ Ai1 , Ai2 , Ai3 , ...
 - linux 文件描述符和inode 的理解和区别
			
inode 或i节点是指对文件的索引.如一个系统,所有文件是放在磁盘或flash上,就要编个目录来说明每个文件在什么地方,有什么属性,及大小等.就像书本的目录一样,便于查找和管理.这目录是操作系统需要 ...
 - Python推导式详解,带你写出比较精简酷炫的代码
			
Python推导式详解,带你写出比较精简酷炫的代码 前言 1.推导式分类与用法 1.1 列表推导 1.2 集合推导 1.3 字典推导 1.4 元组推导?不存在的 2.推导式的性能 2.1 列表推导式与 ...
 - CCCC-exercise
			
CCCC-exercise 1.L1 总结L1 1-27里面我觉得有东西可以总结的题目 贴了部分的代码 L1-006(20) 一个正整数 N 的因子中可能存在若干连续的数字.例如 630 可以分解为 ...
 - Python 数据类型常用的内置方法(一)
			
目录 Python 数据类型常用的内置方法 1.整型 int 2.浮点型 float 字符串转浮点型: 3.字符串 str 多种类型转字符型: 索引 切片 len( )方法:统计字符串长度/个数 移除 ...
 - R数据分析:跟随top期刊手把手教你做一个临床预测模型
			
临床预测模型也是大家比较感兴趣的,今天就带着大家看一篇临床预测模型的文章,并且用一个例子给大家过一遍做法. 这篇文章来自护理领域顶级期刊的文章,文章名在下面 Ballesta-Castillejos ...
 - Scrapy入门到放弃06:Spider中间件
			
前言 写一写Spider中间件吧,都凌晨了,一点都不想写,主要是也没啥用...哦不,是平时用得少.因为工作上的事情,已经拖更好久了,这次就趁着半夜写一篇. Scrapy-deltafetch插件是在S ...
 - python实现分水岭算法
			
目录: 问题:分水岭算法对图像分割很有作用,怎么把对象分割开来的?分水岭算法是比较完美的分割,跟前面的讲的轮廓不一样! (一)原理 (二)实现 (一)原理 opencv中的分水岭算法是基于距离变换的, ...
 - Android LayoutInflater(布局填充器)
			
先来看一下LayoutInflater的基本用法吧,它的用法非常简单,首先需要获取到LayoutInflater的实例,有两种方法可以获取到,第一种写法如下: LayoutInflater layou ...
 - JDBC数据库的使用操作总结
			
JDBC是一组能够执行SQL语句的API 由于传统的数据库操作方式需要程序员掌握各个不同的数据库的API,极其不便 因此java定义了JDBC这一标准的接口和类,为程序员操作数据库提供了统一的方式 J ...