- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
给定以下删除双向链表中元素的代码:
struct list
{
int info;
struct list *prev;
struct list *next;
};
struct list *insert(struct list *top,int k)
{
struct list *tmp=NULL;
if(!top)
{
tmp=(struct list *)malloc(sizeof(struct list));
if(tmp)
{
tmp->info=k;
tmp->next=NULL;
tmp->prev=NULL;
top=tmp;
}
else
printf("error\n");
}
else
{
top->next=insert(top->next,k);
if(top->next)
top->next->prev=top;
}
return top;
}
void delete(struct list **top,int k)
{
struct list *tmp=NULL;
if(*top)
{
if((*top)->info==k)
{
tmp=*top;
if((*top)->next)
{
(*top)->next->prev=(*top)->prev;
}
if((*top)->prev)
{
(*top)->prev->next=(*top)->next;
}
*top=(*top)->next;
free(tmp);
}
else
delete(&((*top)->next),k);
}
}
/* correct function
void delete(struct list **top,int k)
{
struct list *tmp=NULL;
if(*top)
{
if((*top)->info==k)
{
tmp=*top;
if((*top)->next)
{
(*top)->next->prev=(*top)->prev;
}
if((*top)->prev)
{
(*top)->prev->next=(*top)->next;
}
else
*top=(*top)->next;
free(tmp);
}
else
delete(&((*top)->next),k);
}
}*/
int main()
{
int i,k;
struct list *top=NULL;
for(i=1;i<11;i++)
top=insert(top,i);
printf("insert a key\n");
scanf("%d",&k);
delete(&top,k);
// ...
}
问题是当节点k被删除时,k的前一个节点的next字段并不指向k的后继者,而是指向k的后继者的后继者。
例如:
给定以下序列:1 2 3 4 5 6 7 8 9 10。
删除节点5;
结果:1 2 3 4 7 8 9 10。
为什么会这样?
编辑:我在评论中添加了 delete 函数,其中只有当 *top 是头部时指针才会前进,现在它可以工作了。但问题总是悬而未决,即,为什么更改 *top 的值也会修改 (*top)->prev->next=(*top)->next 之前更改的值。
最佳答案
我没有检查 chat room ,但这是我使用的测试代码的工作解决方案。它使用迭代而不是递归方法。我没有更改 insert()
代码(虽然我将它移到了 delete()
函数之后),但如果它成为“我的”代码,我会删除递归。
#include <stdio.h>
#include <stdlib.h>
struct list
{
int info;
struct list *prev;
struct list *next;
};
static
void delete(struct list **top, int k)
{
struct list *node = *top;
while (node)
{
if (node->info == k)
{
if (node->next)
node->next->prev = node->prev;
if (node->prev)
node->prev->next = node->next;
if (node == *top)
*top = node->next;
free(node);
return;
}
node = node->next;
}
}
static
struct list *insert(struct list *top, int k)
{
struct list *tmp = NULL;
if (!top)
{
tmp = (struct list *)malloc(sizeof(struct list));
if (tmp)
{
tmp->info = k;
tmp->next = NULL;
tmp->prev = NULL;
top = tmp;
}
else
printf("error\n");
}
else
{
top->next = insert(top->next, k);
if (top->next)
top->next->prev = top;
}
return top;
}
static void dump_list_fwd(const char *tag, struct list *top)
{
printf("List %p: %s\n", (void *)top, tag);
while (top != 0)
{
printf(" Item %p: %2d (next %p, prev %p)\n", (void *)top,
top->info, (void *)top->next, (void *)top->prev);
top = top->next;
}
}
static void free_list(struct list *top)
{
while (top != 0)
{
struct list *next = top->next;
free(top);
top = next;
}
}
int main(void)
{
struct list *top = NULL;
for (int i = 1; i < 11; i++)
top = insert(top, i);
dump_list_fwd("After insert", top);
for (int i = 1; i < 11; i++)
{
int k = (i * 3 + 5) % 10 + 1;
printf("Delete %d\n", k);
delete(&top, k);
dump_list_fwd("After delete", top);
}
free_list(top);
return 0;
}
使用命令行可以干净地编译:
gcc -O3 -g -std=c11 -Wall -Wextra -Wmissing-prototypes -Wstrict-prototypes \
-Wold-style-definition -Werror dellink.c -o dellink
除main()
以外的功能是 static
满足-Wmissing-prototypes
选项。如果要在这个源文件之外使用这些函数,它们将在 header 中声明——但是没有其他源文件也没有 header ,所以我将它们设为 static
.
delete()
函数被重写为迭代函数。关键观察(修复)是存储在传递给函数的指针中的地址应该仅在该节点被删除时才会更改。当代码是递归时,这更难处理。我也避免使用 (*top)
通过将其复制到局部变量中无处不在;这稍微简化了代码。
print 和 free 函数是简单的代码。我几乎总是通过 tag
字符串打印函数,以便可以简单地识别调用它的不同位置。如果我彻底完成这项工作,我会使用 uintptr_t
和 PRIXPTR
来自 <inttypes.h>
的宏避免 0x0
为空指针打印(我会使用 "0x%.12" PRIXPTR
打印地址,完全意识到指针可能需要 16 个字节,但不经常遇到这个问题)。
main()
代码连续添加条目 1..10。删除循环以不同的顺序处理条目(9、2、5、8、1、4、7、10、3、6),因此“从开始删除”和“从结束删除”都执行为以及“从中间移除”。
List 0x7f9460c03180: After insert
Item 0x7f9460c03180: 1 (next 0x7f9460c031a0, prev 0x0)
Item 0x7f9460c031a0: 2 (next 0x7f9460c031c0, prev 0x7f9460c03180)
Item 0x7f9460c031c0: 3 (next 0x7f9460c031e0, prev 0x7f9460c031a0)
Item 0x7f9460c031e0: 4 (next 0x7f9460c03200, prev 0x7f9460c031c0)
Item 0x7f9460c03200: 5 (next 0x7f9460c03220, prev 0x7f9460c031e0)
Item 0x7f9460c03220: 6 (next 0x7f9460c03240, prev 0x7f9460c03200)
Item 0x7f9460c03240: 7 (next 0x7f9460c03260, prev 0x7f9460c03220)
Item 0x7f9460c03260: 8 (next 0x7f9460c03280, prev 0x7f9460c03240)
Item 0x7f9460c03280: 9 (next 0x7f9460c032a0, prev 0x7f9460c03260)
Item 0x7f9460c032a0: 10 (next 0x0, prev 0x7f9460c03280)
Delete 9
List 0x7f9460c03180: After delete
Item 0x7f9460c03180: 1 (next 0x7f9460c031a0, prev 0x0)
Item 0x7f9460c031a0: 2 (next 0x7f9460c031c0, prev 0x7f9460c03180)
Item 0x7f9460c031c0: 3 (next 0x7f9460c031e0, prev 0x7f9460c031a0)
Item 0x7f9460c031e0: 4 (next 0x7f9460c03200, prev 0x7f9460c031c0)
Item 0x7f9460c03200: 5 (next 0x7f9460c03220, prev 0x7f9460c031e0)
Item 0x7f9460c03220: 6 (next 0x7f9460c03240, prev 0x7f9460c03200)
Item 0x7f9460c03240: 7 (next 0x7f9460c03260, prev 0x7f9460c03220)
Item 0x7f9460c03260: 8 (next 0x7f9460c032a0, prev 0x7f9460c03240)
Item 0x7f9460c032a0: 10 (next 0x0, prev 0x7f9460c03260)
Delete 2
List 0x7f9460c03180: After delete
Item 0x7f9460c03180: 1 (next 0x7f9460c031c0, prev 0x0)
Item 0x7f9460c031c0: 3 (next 0x7f9460c031e0, prev 0x7f9460c03180)
Item 0x7f9460c031e0: 4 (next 0x7f9460c03200, prev 0x7f9460c031c0)
Item 0x7f9460c03200: 5 (next 0x7f9460c03220, prev 0x7f9460c031e0)
Item 0x7f9460c03220: 6 (next 0x7f9460c03240, prev 0x7f9460c03200)
Item 0x7f9460c03240: 7 (next 0x7f9460c03260, prev 0x7f9460c03220)
Item 0x7f9460c03260: 8 (next 0x7f9460c032a0, prev 0x7f9460c03240)
Item 0x7f9460c032a0: 10 (next 0x0, prev 0x7f9460c03260)
Delete 5
List 0x7f9460c03180: After delete
Item 0x7f9460c03180: 1 (next 0x7f9460c031c0, prev 0x0)
Item 0x7f9460c031c0: 3 (next 0x7f9460c031e0, prev 0x7f9460c03180)
Item 0x7f9460c031e0: 4 (next 0x7f9460c03220, prev 0x7f9460c031c0)
Item 0x7f9460c03220: 6 (next 0x7f9460c03240, prev 0x7f9460c031e0)
Item 0x7f9460c03240: 7 (next 0x7f9460c03260, prev 0x7f9460c03220)
Item 0x7f9460c03260: 8 (next 0x7f9460c032a0, prev 0x7f9460c03240)
Item 0x7f9460c032a0: 10 (next 0x0, prev 0x7f9460c03260)
Delete 8
List 0x7f9460c03180: After delete
Item 0x7f9460c03180: 1 (next 0x7f9460c031c0, prev 0x0)
Item 0x7f9460c031c0: 3 (next 0x7f9460c031e0, prev 0x7f9460c03180)
Item 0x7f9460c031e0: 4 (next 0x7f9460c03220, prev 0x7f9460c031c0)
Item 0x7f9460c03220: 6 (next 0x7f9460c03240, prev 0x7f9460c031e0)
Item 0x7f9460c03240: 7 (next 0x7f9460c032a0, prev 0x7f9460c03220)
Item 0x7f9460c032a0: 10 (next 0x0, prev 0x7f9460c03240)
Delete 1
List 0x7f9460c031c0: After delete
Item 0x7f9460c031c0: 3 (next 0x7f9460c031e0, prev 0x0)
Item 0x7f9460c031e0: 4 (next 0x7f9460c03220, prev 0x7f9460c031c0)
Item 0x7f9460c03220: 6 (next 0x7f9460c03240, prev 0x7f9460c031e0)
Item 0x7f9460c03240: 7 (next 0x7f9460c032a0, prev 0x7f9460c03220)
Item 0x7f9460c032a0: 10 (next 0x0, prev 0x7f9460c03240)
Delete 4
List 0x7f9460c031c0: After delete
Item 0x7f9460c031c0: 3 (next 0x7f9460c03220, prev 0x0)
Item 0x7f9460c03220: 6 (next 0x7f9460c03240, prev 0x7f9460c031c0)
Item 0x7f9460c03240: 7 (next 0x7f9460c032a0, prev 0x7f9460c03220)
Item 0x7f9460c032a0: 10 (next 0x0, prev 0x7f9460c03240)
Delete 7
List 0x7f9460c031c0: After delete
Item 0x7f9460c031c0: 3 (next 0x7f9460c03220, prev 0x0)
Item 0x7f9460c03220: 6 (next 0x7f9460c032a0, prev 0x7f9460c031c0)
Item 0x7f9460c032a0: 10 (next 0x0, prev 0x7f9460c03220)
Delete 10
List 0x7f9460c031c0: After delete
Item 0x7f9460c031c0: 3 (next 0x7f9460c03220, prev 0x0)
Item 0x7f9460c03220: 6 (next 0x0, prev 0x7f9460c031c0)
Delete 3
List 0x7f9460c03220: After delete
Item 0x7f9460c03220: 6 (next 0x0, prev 0x0)
Delete 6
List 0x0: After delete
valgrind
为代码提供了健康证明— 没有访问错误,也没有未释放的内存。
让我们画一幅画——ASCII 艺术脱颖而出!
+-----+
| top |
+-----+
|
v
+--------+ +--------+ +--------+ +--------+
| |---->| |---->| |---->| |
| Node 1 | | Node 2 | | Node 3 | | Node 4 |
| |<----| |<----| |<----| |
+--------+ +--------+ +--------+ +--------+
top
在递归调用中实际存储在'Node 1'->next
!
+-----+
| top |
+-----+
|
v
+--------+ +--------+ +--------+ +--------+
| |---->| |---->| |---->| |
| Node 1 | | Node 2 | | Node 3 | | Node 4 |
| |<----| |<----| |<----| |
+--------+ +--------+ +--------+ +--------+
(*top)->next
:分配:(*top)->next->prev=(*top)->prev;
+-----+
| top |
+-----+
|
v
+--------+ +--------+ +--------+ +--------+
| |---->| |---->| |---->| |
| Node 1 | | Node 2 | | Node 3 | | Node 4 |
| |<-------------------| |<----| |
+--------+ +--------+ +--------+ +--------+
(*top)->prev
:分配:(*top)->prev->next=(*top)->next;
请注意,因为 top
实际上是指向'Node 1'->next
的指针,作业 (*top)->prev->next = (*top)->next
也改变 (*top)
.
+-----+
| top |
+-----+
|
v
+--------+ +--------+ +--------+ +--------+
| |------------------->| |---->| |
| Node 1 | | Node 2 | | Node 3 | | Node 4 |
| |<-------------------| |<----| |
+--------+ +--------+ +--------+ +--------+
*top
免费正在分配 (*top) = (*top)->next;
因为 *top
也是'Node 1'->next
, 分配给 *top
也改变了'Node 1'->next
指向,跳过'Node 3'
.请注意,向后遍历仍会经过 'Node 3'
.
+-----+
| top |
+-----+
|
v
+--------+ +--------+ +--------+
| |---------------------------------->| |
| Node 1 | | Node 3 | | Node 4 |
| |<-------------------| |<----| |
+--------+ +--------+ +--------+
现在解释损坏的原因。
关于c - 在 C 中通过引用传递删除列表中的节点,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37011095/
这个问题在这里已经有了答案: 关闭 10 年前。 Possible Duplicate: template pass by value or const reference or…? 以下对于将函数
我用相同的参数列表重载了一个运算符两次。但返回类型不同: T& operator()(par_list){blablabla} const T& operator()(par_list){bla
假设我有实现接口(interface) I 的 Activity A。我的 ViewModel 类 (VM) 持有对实现接口(interface) I 的对象的引用: class A extends
PHP 如何解释 &$this ?为什么允许? 我遇到了以下问题,这看起来像是 PHP 7.1 和 7.2 中的错误。它与 &$this 引用和跨命名空间调用以及 call_user_func_arr
谁能解释一下下面“&”的作用: class TEST { } $abc =& new TEST(); 我知道这是引用。但是有人可以说明我为什么以及什么时候需要这样的东西吗?或者给我指向一个对此有很好解
引用变量是一个别名,也就是说,它是某个已存在变量的另一个名字。一旦把引用初始化为某个变量,就可以使用该引用名称或变量名称来指向变量。 C++ 引用 vs 指针 引用很容易与指针混淆,它们之间有三
目录 引言 背景 结论 引言 我选择写C++中的引用是因为我感觉大多数人误解了引用。而我之所以有这个感受是因为我主持过很多C++的面试,并且我很少
Perl 中的引用是指一个标量类型可以指向变量、数组、哈希表(也叫关联数组)甚至函数,可以应用在程序的任何地方 创建引用 定义变量的时候,在变量名前面加个 \,就得到了这个变量的一个引用 $sc
我编写了一个将从主脚本加载的 Perl 模块。该模块使用在主脚本中定义的子程序(我不是维护者)。 对于主脚本中的一个子例程,需要扩展,但我不想修补主脚本。相反,我想覆盖我的模块中的函数并保存对原始子例
我花了几个小时试图掌握 F# Quotations,但我遇到了一些障碍。我的要求是从可区分的联合类型中取出简单的函数(只是整数、+、-、/、*)并生成一个表达式树,最终将用于生成 C 代码。我知道使用
很多时候,问题(尤其是那些标记为 regex 的问题)询问验证密码的方法。似乎用户通常会寻求密码验证方法,包括确保密码包含特定字符、匹配特定模式和/或遵守最少字符数。这篇文章旨在帮助用户找到合适的密码
我想通过 MIN 函数内的地址(例如,C800)引用包含文本的最后一个单元格。你能帮忙吗? Sub Set_Formula() ' ----------------------------- Dim
使用常规的 for 循环,我可以做类似的事情: for (let i = 0; i < objects.length; i++) { delete objects[i]; } 常规的 for-
在 Cucumber 中,您定义了定义 BDD 语法的步骤;例如,您的测试可能有: When I navigate to step 3 然后你可以定义一个步骤: When /^I navigate t
这是什么UnaryExpression的目的,以及应该怎样使用? 最佳答案 它需要一个 Expression对象并用另一个 Expression 包裹它.例如,如果您有一个用于 lambda 的表达式
给出以下内容 $("#identifier div:first, #idetifier2").fadeOut(300,function() { // I need to reference jus
我不知道我要找的东西的正确术语,但我要找的是一个完整的引用,可以放在双引号之间的语句,比如 *, node()、@* 以及所有列出的 here加上任何其他存在的。 我链接到的答案提供了一些细节,但还
This question's answers are a community effort。编辑现有答案以改善此职位。它当前不接受新的答案或互动。 这是什么? 这是常见问答的集合。这也是一个社区Wi
Closed. This question does not meet Stack Overflow guidelines。它当前不接受答案。 想改善这个问题吗?更新问题,以便将其作为on-topic
考虑下一个代码: fn get_ref(slice: &'a Vec, f: fn(&'a Vec) -> R) -> R where R: 'a, { f(slice) } fn m
我是一名优秀的程序员,十分优秀!