答案 0 :(得分:16)
这是一个我以前没见过的有趣的想法。凭借今天相当丰富的内存,它看起来很复杂,但收益微不足道(尽管不是所有平台都充满了内存)。 编辑在完成我的实际工作时,我的思绪不断回归,所以我添加了创建新节点并将其放在给定端的功能。现在比较漂亮。 addnode和traverse函数都是对称的,这很酷。两者都不需要知道方向。只需给它列表的一端,它们就能正常运行。
根据Darron的评论(谢谢),我将int更改为intptr_t
以便于移植。
#include <stdio.h>
#include <malloc.h>
#include <stdint.h> // gcc needs this for intptr_t.
typedef struct xorll {
int value;
struct xorll *np;
} xorll;
// traverse the list given either the head or the tail
void traverse( xorll *start ) // point to head or tail
{
xorll *prev, *cur;
cur = prev = start;
while ( cur )
{
printf( "value = %d\n", cur->value );
if ( cur->np == cur )
// done
break;
if ( cur == prev )
cur = cur->np; // start of list
else {
xorll *save = cur;
cur = (xorll*)((uintptr_t)prev ^ (uintptr_t)cur->np);
prev = save;
}
}
}
// create a new node adding it to the given end and return it
xorll* newnode( xorll *prev, xorll *cur, int value )
{
xorll *next;
next = (xorll*)malloc( sizeof( xorll ));
next->value = value;
next->np = cur; // end node points to previous one
if ( cur == NULL )
; // very first node - we'll just return it
else if ( prev == NULL ) {
// this is the second node (they point at each other)
cur->np = next;
next->np = cur;
}
else {
// do the xor magic
cur->np = (xorll*)((uintptr_t)prev ^ (uintptr_t)next);
}
return next;
}
int main( int argc, char* argv[] )
{
xorll *head, *tail;
int value = 1;
// the first two nodes point at each other. Weird param calls to
// get the list started
head = tail = newnode( NULL, NULL, value++ );
tail = newnode( NULL, tail, value++ );
// now add a couple to the end
tail = newnode( tail->np, tail, value++ );
tail = newnode( tail->np, tail, value++ );
// this is cool - add a new head node
head = newnode( head->np, head, 999 );
printf( "Forwards:\n" );
traverse( head );
printf( "Backwards:\n" );
traverse( tail );
}
答案 1 :(得分:9)
由于无法对指针执行xor操作,因此必须将地址转换为整数类型以执行xor并将结果转换回右指针类型。
据我所知,C99只有两种整数类型,可以保证与具有已定义行为的指针之间的转换(=使原始指针返回):来自intptr_t
的{{1}}和uintptr_t
。请注意,这两种类型都是可选的,因此您的实现可能没有它们。
转化示例,假设<stdint.h>
和a
是指向b
的有效指针:
struct node
我不是百分之百确定需要额外演员#include <stdint.h>
/* creating an xor field */
uintptr_t x = (uintptr_t) (void *) a ^ (uintptr_t) (void *) b;
/* reconstructing an address */
a = (void *) (x ^ (uintptr_t) (void *) b);
,如果不是,请有人纠正我。有关void *
类型的更多信息,请参阅C99标准的第7.18.1.4节。
答案 2 :(得分:5)
是否可以在C中实现它,因为XOR链接列表涉及对地址的操作?
是的。地址是指针,指针是数字*,数字允许XOR(即a ^ b
)。
Look up 完成了什么,您应该能够实施。
*至少,您可以将它们视为数字 - 但可能需要显式强制转换。