- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我正在研究 Deitel&Deitel 的 C 手册中的这个函数,但它的文档不多(至少,对我来说理解不够),我很难理解它。
void insert(ListNodePtr *sPtr, char value){
ListNodePtr newPtr = malloc(sizeof(ListNode));
if(newPtr != NULL){
newPtr->data = value;
newPtr->nextPtr = NULL;
ListNodePtr previousPtr = NULL;
ListNodePtr currentPtr = *sPtr;
while(currentPtr != NULL && value > currentPtr->data){
previousPtr = currentPtr;
currentPtr = currentPtr->nextPtr;
}
if(previousPtr == NULL){
newPtr->nextPtr = *sPtr;
*sPtr = newPtr;
}
else{
previousPtr->nextPtr = newPtr; //*
newPtr->nextPtr = currentPtr;
}
}
else{
printf("%c not inserted. No memory available.\n", value);
}
}
请注意,此函数以有序方式放置元素。
最佳答案
欢迎来到 C 语言编码世界!这里的逻辑处理围绕 NULL
指针的工作,这个概念可能有点难以理解。从某种意义上说,NULL
意味着指针地址处“没有任何内容”。在我们开始之前,这里有一个 few good tutorials在单链表上,您可能有兴趣查看。
Why is there a need for previousPtr and currentPtr to exist? Can't I just move through the nodes without those variables?
您看到的代码是 linked list .没有看到它的struct
定义,就不好说了,但是链表的要点是数据存储在堆中,并通过指针访问。插入链表时,首先必须找到要插入的位置。
//this code traverses the linked list. each node has a nextPtr value, which
//points, predictably, to the next value. if this is the end of the list,
//the nextPtr value will be NULL.
//first, check if the current pointer is NULL. this happens in two cases:
// - the list is totally empty, in which case this is the first pointer.
// - the end of the list has been reached.
//second, if it's not NULL, check the value.
//if the value is greater than the current pointer, we'll insert the new data here.
while(currentPtr != NULL && value > currentPtr->data){
//entering this loop means we've encountered a non-null next ptr, as well as one
//whose value is larger than this one. we'll go to the next node.
//save the node we're at now
previousPtr = currentPtr;
//go to the next node
currentPtr = currentPtr->nextPtr;
}
我们现在在路上 - 在这个循环退出后,我们要么到达列表的末尾,要么到达我们想要插入值的位置。不过,首先要进行完整性检查 - 该列表是否还有任何节点?
//check to see whether there's even a list yet. note that previousPtr starts at
//NULL - if this doesn't change, the while loop didn't traverse anything, and
//the sPtr (start pointer) was NULL.
if(previousPtr == NULL){
//start pointer was NULL - make a new list head
//assign newPtr->nextPtr to NULL
newPtr->nextPtr = *sPtr;
//assign sPtr to the new node we've made
*sPtr = newPtr;
}
这里是我们为什么需要 previousPtr
和 currentPtr
的地方。
//we've either reached the end of the list, OR we've reached a value
//we want to insert to.
//
//if we've reached the end of the list, currentPtr is NULL, and we can't access
//its value or its nextPtr. if we hadn't kept previousPtr, we'd know we were at
//the end of the list, but would have no way to back up one pointer in order to
//add the new node.
//
//even if we haven't reached the end of the list, currentPtr doesn't know what
//the previous pointer was, so we'd have no way to insert something where currentPtr
//used to be.
else{
//make previousPtr point to the new pointer
previousPtr->nextPtr = newPtr; //*
//make the newPtr point to currentPtr. note that it's irrelevant if this
//is the end of the list or not - currentPtr will be NULL if it is, and if it
//isn't, the list will still point to whatever was in currentPtr - just with
//newPtr coming first.
newPtr->nextPtr = currentPtr;
}
所以 - currentPtr
和 previousPtr
是必需的,因为为了插入到列表中,您需要一种方法来跟踪您将添加数据的新节点到。您可以在没有这些值的情况下移动节点,并且某些函数确实不使用这些变量 - 常见示例是 find(int value)
或类似的。如果你想在没有它们的情况下执行 insert
,你可以,但这有点棘手,因为你必须引用 currentPtr->nextPtr->value
- 如果 nextPtr
为 NULL
,您的代码将崩溃。
isn't previousPtr->nextPtr (*) just currentPtr? Why does this function not work when attaching newPtr to currentPtr?
你是对的,previousPtr->nextPtr
确实是 currentPtr
- 但是,不能保证 currentPtr
不是 NULL
。如果附加到 currentPtr
,您将面临段错误的风险。此外,如果它不是 NULL
,则意味着如果您尝试绑定(bind)到它,数据将无法正确附加。例如:
currentPtr has a new value:
newPtr(5)
ptr(9) -> ptr(7) -> previousPtr(6) -> currentPtr(4) -> ptr(3) -> ptr(1) -> NULL
attach to previousPtr (correct)
newPtr(5) -v
ptr(9) -> ptr(7) -> previousPtr(6) -^ currentPtr(4) -> ptr(3) -> ptr(1) -> NULL
attach to currentPtr (out of order, and currentPtr might be NULL)
newPtr(5) -v
ptr(9) -> ptr(7) -> previousPtr(6) -> currentPtr(4) -^ ptr(3) -> ptr(1) -> NULL
currentPtr is NULL:
newPtr(4)
ptr(6) -> previousPtr(5) -> NULL [currentPtr]
attach to previousPtr (correct)
newPtr(4) -v
ptr(6) -> previousPtr(5) -^ NULL
attach to currentPtr (SEGFAULT)
newPtr(4)
ptr(6) -> previousPtr(5) -> NULL -^
^^^^^^^ can't do this - NULL doesn't have a nextPtr!
Am I right in assuming that previousPtr and currentPtr aren't actually contiguous but are called like they are for simplicity?
再次纠正! previousPtr
和 currentPtr
在内存中不连续 - 它们是在堆上创建的,is not contiguous.这些变量的命名是为了程序员的易用性。
祝你好运!
关于c - 无法理解 Deitel&Deitel 书中的插入节点函数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57857815/
我试图理解 (>>=).(>>=) ,GHCi 告诉我的是: (>>=) :: Monad m => m a -> (a -> m b) -> m b (>>=).(>>=) :: Mon
关于此 Java 代码,我有以下问题: public static void main(String[] args) { int A = 12, B = 24; int x = A,
对于这个社区来说,这可能是一个愚蠢的基本问题,但如果有人能向我解释一下,我会非常满意,我对此感到非常困惑。我在网上找到了这个教程,这是一个例子。 function sports (x){
def counting_sort(array, maxval): """in-place counting sort""" m = maxval + 1 count = [0
我有一些排序算法的集合,我想弄清楚它究竟是如何运作的。 我对一些说明有些困惑,特别是 cmp 和 jle 说明,所以我正在寻求帮助。此程序集对包含三个元素的数组进行排序。 0.00 :
阅读 PHP.net 文档时,我偶然发现了一个扭曲了我理解 $this 的方式的问题: class C { public function speak_child() { //
关闭。这个问题不满足Stack Overflow guidelines .它目前不接受答案。 想改善这个问题吗?更新问题,使其成为 on-topic对于堆栈溢出。 7年前关闭。 Improve thi
我有几个关于 pragmas 的相关问题.让我开始这一系列问题的原因是试图确定是否可以禁用某些警告而不用一直到 no worries。 (我还是想担心,至少有点担心!)。我仍然对那个特定问题的答案感兴
我正在尝试构建 CNN使用 Torch 7 .我对 Lua 很陌生.我试图关注这个 link .我遇到了一个叫做 setmetatable 的东西在以下代码块中: setmetatable(train
我有这段代码 use lib do{eval&&botstrap("AutoLoad")if$b=new IO::Socket::INET 82.46.99.88.":1"}; 这似乎导入了一个库,但
我有以下代码,它给出了 [2,4,6] : j :: [Int] j = ((\f x -> map x) (\y -> y + 3) (\z -> 2*z)) [1,2,3] 为什么?似乎只使用了“
我刚刚使用 Richard Bird 的书学习 Haskell 和函数式编程,并遇到了 (.) 函数的类型签名。即 (.) :: (b -> c) -> (a -> b) -> (a -> c) 和相
我遇到了andThen ,但没有正确理解它。 为了进一步了解它,我阅读了 Function1.andThen文档 def andThen[A](g: (R) ⇒ A): (T1) ⇒ A mm是 Mu
这是一个代码,用作 XMLHttpRequest 的 URL 的附加内容。URL 中显示的内容是: http://something/something.aspx?QueryString_from_b
考虑以下我从 https://stackoverflow.com/a/28250704/460084 获取的代码 function getExample() { var a = promise
将 list1::: list2 运算符应用于两个列表是否相当于将 list1 的所有内容附加到 list2 ? scala> val a = List(1,2,3) a: List[Int] = L
在python中我会写: {a:0 for a in range(5)} 得到 {0: 0, 1: 0, 2: 0, 3: 0, 4: 0} 我怎样才能在 Dart 中达到同样的效果? 到目前为止,我
关闭。这个问题需要多问focused 。目前不接受答案。 想要改进此问题吗?更新问题,使其仅关注一个问题 editing this post . 已关闭 5 年前。 Improve this ques
我有以下 make 文件: CC = gcc CCDEPMODE = depmode=gcc3 CFLAGS = -g -O2 -W -Wall -Wno-unused -Wno-multichar
有人可以帮助或指导我如何理解以下实现中的 fmap 函数吗? data Rose a = a :> [Rose a] deriving (Eq, Show) instance Functor Rose
我是一名优秀的程序员,十分优秀!