- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
为了理解实现、操作等,我从 JDK 中复制了 LinkedList 类,并以非泛型(仅限字符串)方式进行了修改。您可以找到实际代码 here .与问题相关的是
addLast()
public void addLast(String e){
System.out.println("Invoked:addLast()");
linkLast(e);
}
节点内部类
private static class Node {
String item;
Node next;
Node prev;
Node(Node prev, String element, Node next) {
this.item = element;
this.next = next;
this.prev = prev;
}
@Override public String toString() {
return "Node [item=" + item + ", next=" + next + ", prev=" + prev+ "]";
}
}
linkLast()
void linkLast(String e) {
System.out.println("Invoked:linkLast(" + e + ")");
final Node l = last;
final Node newNode = new Node(l, e, null);
System.out.println("\tCreatd new node as:" + newNode);
last = newNode;
System.out.println("\tSetting this node as last");
if (l == null){
// Applicable for 1st node. Last & First point to newNode
first = newNode;
System.out.println("\tLast was null, setting this node as first.");
}
else{
// Set newNode as next of previous Last.
System.out.println("\tLast was not null, seting this node as next node of previous Last.");
l.next = newNode;
}
size++;
//System.out.println("Size of list:" + size);
modCount++;
System.out.println("\tMod Count:" + modCount + "\n");
}
当我执行这个类时,出现 stackoverflow 错误,
public static void main(String args[]){
LinkedListDemo obj = new LinkedListDemo();
obj.add("B");
obj.addFirst("A");
obj.addLast("C");
}
输出
Invoked:add(B)
Invoked:linkLast(B)
Creatd new node as:Node [item=B, next=null, prev=null]
Setting this node as last Last was null, setting this node as first.
Mod Count:1
Invoked:addFirst()
Invoked:linkFirst(A)
Creatd new node asNode [item=A, next=Node [item=B, next=null, prev=null], prev=null]
Seting this node as first
First was not null, seting this node as next node of previous First.
Mod Count:2
Invoked:addLast()
Invoked:linkLast(C)
Exception in thread "main" java.lang.StackOverflowError
at java.lang.String.getChars(String.java:826)
at java.lang.AbstractStringBuilder.append(AbstractStringBuilder.java:416)at java.lang.StringBuilder.append(StringBuilder.java:132)
at java.lang.StringBuilder.<init>(StringBuilder.java:110)
at collections.LinkedListDemo$Node.toString(LinkedListDemo.java:570)
at java.lang.String.valueOf(String.java:2847)
at java.lang.StringBuilder.append(StringBuilder.java:128)
at collections.LinkedListDemo$Node.toString(LinkedListDemo.java:570)
然后最后 3 行不断重复。 我无法弄清楚是什么导致了这种情况。
更新 - 基于收到的回复
所以,如果理解正确的话,
Invoked:add(A)
Invoked:linkLast(A)
Creatd new node as:Node [item=A, next=null, prev=null]
Setting this node as last
Last was null, setting this node as first.
Mod Count:1
Invoked:add(B)
Invoked:linkLast(B)
Creatd new node as:Node [item=B, next=null, prev=Node [item=A, next=null, prev=null]]
Setting this node as last
Last was not null, seting this node as next node of previous Last.
Mod Count:2
在这一阶段“节点 B”仅创建了单侧链接(上一个),其中调用了 toString()。只有在链接另一侧链接(下一个)之后。而这个双重链接是无限递归的原因,在下一次插入时。
也许这就是为什么 toString() 方法没有在两个父类(super class) AbstractSequentialList 和 AbstractList 中实现,而是在 AbstractCollection 中实现的原因。该实现提取一个 Iterator,它迭代以打印集合。
那么,为什么它只发生在第 3 次插入时。两次插入不会导致此问题。
最佳答案
问题是您的 toString
方法,它在尝试打印 prev
和 next
节点时递归调用自身。
如果当前打印的Node
的next
不为空,则调用next.toString()
,当打印那个
,再次调用原Node
的prevNode
的toString()
,所以没有end递归。
可能的解决方案:
@Override
public String toString()
{
return "Node [item=" + item + ", next=" + (next==null?"null":next.item) + ", prev=" + (prev==null?"null":prev.item) + "]";
}
关于评论:
添加第一个节点后,列表如下所示:
prev next
null <- Node1 -> null
first
和last
都是指 Node1
在 toString 中没有无限递归的风险。
添加第二个节点后,列表如下所示:
null <- Node1 -> <- Node2 -> null
first
指的是 Node1,last
指的是 Node2
如您所见,Node1 和 Node2 相互引用(Node1 是 Node2 的 prev
,Node2 是 Node1 的 next
)。因此,在第二次插入后尝试使用原始 toString()
方法打印其中一个节点将导致无限递归。
但是,由于您在执行 l.next = newNode;
行之前打印了新节点,所以您不会在第二次插入时得到无限递归,而只会在第三次插入时得到。如果将 System.out.println("\tCreatd new node as:"+ newNode);
移动到 linkLast
的末尾,第二次插入将导致无限递归。
关于java - 解码 LinkedList 实现,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33890112/
我想添加 LinkedList (我们称之为列表 A)到 LinkedList> (称之为列表 B)。执行此操作后,我需要更改列表 A 的值并将其再次添加到列表 B,但不更改已存储在列表 B 中的值。
更新:感谢所有的回答。我发现的最干净的解决方案是这个: if ( k(Arrays.asList(new LinkedList<>())); 我有一个递归方法,可以从列表中生成所有“n 选 k”组合。
在我的 Java 应用程序中,以下两个都将编译和运行,并产生所需的结果。 //"Rotate" the list items one place to the left. myLinkedList.a
我写了一个LinkedList接受 Nodes 的类存储 Integers . 然后我创建了一个 LinkedList stack = new LinkedList() ,并添加了 Node s 如果
这个问题在这里已经有了答案: What does it mean to "program to an interface"? (33 个答案) 关闭 9 年前。 新手 Java 问题: 谁能解释一下
我有一个问题。我无法并排输出我的 LinkedList。我问了这个问题,但遗憾的是我的老师告诉我不要更改方法头或使用 java 库(如日历)。我得到了很多关于使用它或更改方法头的建议。我是根据年级而定
这里有什么问题?。我正在尝试使用邻接列表,通过利用 util 包中的集合来实现图形数据结构。这里 LinkedList array which holds some integer. Each ele
这个问题已经有答案了: Reversing a linked list in Java, recursively (33 个回答) 已关闭10 年前。 如何使用 linkedList 类中的方法以相反
我需要实现一个 LinkedList,到目前为止,我已经编写了在列表中按顺序插入值的方法。我有我的节点 front 作为我的类的实例数据,当创建我的第一个值并尝试将 front 的 next 值设置为
目前,我的 LinkedList(不是 Java 的)类中有一个方法,可以将单个节点添加到 LinkedList 中,如下所示: public void add(int index, T v) {
我正在编写一个读取 XML 文件的类,该 XML 使用“sax”类进行解析。在我的 XML 文件中,我创建了“for”标签和“宏”,使 for 循环能够写入 XML,例如: Th
我正在处理一个 C++ 作业,我将在一个链表的链表上创建一个搜索引擎。根据要求,我不能使用其他库和 STL。 基本上它会是这样的(我从小列表中删除了变量,因为它们是不相关的): 我的结构是这些: st
老实说,我现在真的很困惑这个问题,并且真的不知道如何解决这个问题。我需要编写一个方法,其中给定一个字符链接列表(例如:{'a','A','d','X'})并返回仅包含大写字符的列表(返回:{'A','
我正在尝试获取可执行文件中的两个链表,并在交替位置将它们合并到一起。前任。 ListOne 1,2,3 和 ListTwo 4,5 新的 ListOne 应该是 1,4,2,5,3。 链表.h文件:
这个问题在这里已经有了答案: Is List a subclass of List? Why are Java generics not implicitly polymorphic? (19 个回答
在尝试了解如何将哈希表插入LinkedLists时,我遇到了麻烦。我失去了尝试过的不同事物的数量。我知道我可以使用ArrayList或其他东西,但是我想使它与LinkedLists一起工作,以便可以对
我一直在尝试编写一种方法,不仅可以从 LinkedList(allUsers) 中删除对象(User),还可以从所有用户拥有的单个 LinkedList 中删除。谁能向我解释为什么这是错误的?我已经包
我有一个列表结构和一个名为树的递归函数。在下面的代码中,它永远不会到达 current == null 语句,因此它将永远运行。 如果我无法使用null,解决方案是什么? private void t
这个问题在这里已经有了答案: How does one add a LinkedList to a LinkedList in C#? (3 个答案) 关闭 9 年前。 假设我有以下内容: Link
我正在尝试为 LinkedList 创建一个反向 ListIterator,并且打算将其实现为 linkedList.listIterator(linkedList. size()) 交换了 next
我是一名优秀的程序员,十分优秀!