offer_JZ25

题目:输入一个复杂链表(每个节点中有节点值,以及两个指针,一个指向下一个节点,另一个特殊指针random指向一个随机节点),请对此链表进行深拷贝,并返回拷贝后的头结点。(注意,输出结果中请不要返回参数中的节点引用,否则判题程序会直接返回空)

思路:

具体做法可以分为四步

第一:输入一个复杂链表,返回结果为复制后复杂链表的head

第二:根据原始链表的每个结点N创建对应的结点N'

第三:设置复制出来结点的random.假设原始结点的随机指向S,复制出来结点的random指向S后一个

第四:把这个长链表拆分成两个链表,奇数位置连接起来就是原始链表,偶数结点连接起来就是复制结点

	public RandomListNode Clone(RandomListNode pHead) {
CloneNodes(pHead);//根据所给的头结点进行拷贝
ConnectRandomNodes(pHead);//设置随机值
return ReConnectNodes(pHead);//返回分好的偶数结点
} // 第一步:根据原始链表的每个结点N创建对应的结点N'
private void CloneNodes(RandomListNode pHead) {
RandomListNode node = pHead;//定义一个结点,指向头原来的头结点
while (node != null) {//只要 原来的 当前结点 不空
RandomListNode cloneNode = new RandomListNode(node.label);//new一个新的结点,并且结点的值域与原来的结点相同
cloneNode.next = node.next;//将新的结点的next指向原来的当前结点的下一个结点
cloneNode.random = null;//random先不管,置空 node.next = cloneNode;//把原来当前结点的next指向新的结点
node = cloneNode.next;//把原来的当前结点往后移一个位置,也就是把新的结点的next指向赋值给node
}
} // 第二步:设置复制出来结点的random.假设原始结点的随机指向S,复制出来结点的random指向S后一个
private void ConnectRandomNodes(RandomListNode pHead) {
RandomListNode node = pHead;
while (node != null) {
RandomListNode clone = node.next;//偶数位置就是克隆出来的新的结点,也是原来结点的next的指向
if (node.random != null) {//如果原来的结点的random不是空
clone.random = node.random.next;//把原来random里面的东西赋值给克隆结点的random;
}
node = clone.next;//原来的当前位置往后移
}
} // 第三步:把这个长链表拆分成两个链表,奇数位置连接起来就是原始链表,偶数结点连接起来就是复制结点
private RandomListNode ReConnectNodes(RandomListNode pHead) {
RandomListNode node = pHead;//当前结点指向原来的头结点
RandomListNode cloneHead = null;//复制的头结点指向空
RandomListNode cloneNode = null;//当前的复制结点指向空 // 设置第一个节点
if (node != null) {
cloneHead = node.next;//偶数位置就是克隆出来的新的结点,也是原来结点的next的指向,用克隆的头结点指向这个克隆的结点,标记原来结点的下一个结点是克隆结点的头结点
cloneNode = node.next;//把当前的克隆结点指向原来的当前结点的下一个结点
node.next = cloneNode.next;//把原来的当前结点的next的指向换成克隆结点指向的结点
node = node.next;//后移
} while (node != null) {
cloneNode.next = node.next;//把当前克隆结点的next指向换成原来的当前结点的下一个结点
cloneNode = cloneNode.next;//当前的克隆结点的位置后移一个结点
node.next = cloneNode.next;//把原来的当前结点的next指向换成当前克隆结点的下一个结点
node = node.next;//原来的当前结点后移一个结点
}
return cloneHead;
}

图示:第三步中的

  1. 设置第一个结点

  2. // 设置第一个节点
    if (node != null) {
    cloneHead = node.next;//偶数位置就是克隆出来的新的结点,也是原来结点的next的指向,用克隆的头结点指向这个克隆的结点,标记原来结点的下一个结点是克隆结点的头结点
    cloneNode = node.next;//把当前的克隆结点指向原来的当前结点的下一个结点
    node.next = cloneNode.next;//把原来的当前结点的next的指向换成克隆结点指向的结点
    node = node.next;//后移
    }

  1. 继续

// 设置第一个节点
if (node != null) {
cloneHead = node.next;//偶数位置就是克隆出来的新的结点,也是原来结点的next的指向,用克隆的头结点指向这个克隆的结点,标记原来结点的下一个结点是克隆结点的头结点
cloneNode = node.next;//把当前的克隆结点指向原来的当前结点的下一个结点
node.next = cloneNode.next;//把原来的当前结点的next的指向换成克隆结点指向的结点
node = node.next;//后移
} while (node != null) {
cloneNode.next = node.next;//把当前克隆结点的next指向换成原来的当前结点的下一个结点
cloneNode = cloneNode.next;//当前的克隆结点的位置后移一个结点
node.next = cloneNode.next;//把原来的当前结点的next指向换成当前克隆结点的下一个结点
node = node.next;//原来的当前结点后移一个结点
}

总结

xxx.next 在等号的左边 代表 xxx的next这个尾巴的指向

​ 在等号的右边 代表 xxx的next的下一个结点实体

offer_JZ25的更多相关文章

随机推荐

  1. GitLab集成kubernetes

    创建GitLab源码项目并上传示例代码 1. 创建GitLab源码项目 本示例中创建的GitLab源码项目地址为:https://gitee.com/SunHarvey/helloworld_java ...

  2. Azure Storage 系列(六)使用Azure Queue Storage

    一,引言 在之前介绍到 Azure Storage 第一篇文章中就有介绍到 Azure Storage 是 Azure 上提供的一项存储服务,Azure 存储包括 对象.文件.磁盘.队列和表存储.这里 ...

  3. 能否使用GHDL+GTKWave代替Quartus ii

    能否使用GHDL+GTKWave代替Quartus ii macOS High Sierra系统 10.13.6 [toc] 先给出答案 可以替代一部分功能 如果你是一个学工科的学生,正在学习EDA. ...

  4. @ComponentScan比较

    ComponetScan 定义扫描规则 value:指定要扫描的包 excludeFilters=Filter[] 指定扫描的时候按照什么规则排除哪些组件. includeFilters=Filter ...

  5. 动态代理:jdk动态代理和cglib动态代理

    /** * 动态代理类:先参考代理模式随笔,了解代理模式的概念,分为jdk动态代理和cglib,jdk动态代理是通过实现接口的方式创建代理类的,cglib是通过继承类的方式实现的代理类的 * jdk动 ...

  6. Python numpy总结(3)——常用函数用法

    1,np.ceil(x, y) 限制元素范围,进一法,即向上取整. x 表示输入的数据  y float类型 表示每个元素的上限. a = np.array([-1.7, -1.5, -0.2, 0. ...

  7. 图文并茂C++精华总结 复习和进阶

    字面常量不可以有引用,因为这也不需要使用符号来引用了,但是字面常量却可以初始化const引用,这将生成一个只读变量: 对变量的const修饰的引用是只读属性的: 也就是说,const修饰的引用,不管是 ...

  8. SDK测试操作文档

    准备所需材料 先把下列所需压缩包和文件传到虚拟机中. crypto-config压缩包存放order和peer节点所需要的证书文件(需要的是申请联盟链中的order和peer的证书文件) m2压缩包是 ...

  9. 简说Modbus-RTU与Modbus-ASCII

    Modbus在串行总线通信中的协议有RTU和ASCII两种.RTU是Remote Terminal Unit的缩写,意思是远程终端单元.ASCII是American Standard Code for ...

  10. layui+tp5表单提交回调

    layui 前段页面form表单提交数据如果监听表单提交 ,tp5后台操作完成后使用 $this->success('success'); 后前端的页面不会出现layui的layer弹窗提示su ...