用于分离链表中偶数和奇数节点的 C 程序
给定一个整数链表,编写一个函数来修改链表,使所有偶数出现在修改后的链表中所有奇数之前。此外,保持偶数和奇数的顺序相同。
例子:
Input: 17->15->8->12->10->5->4->1->7->6->NULL
Output: 8->12->10->4->6->17->15->5->1->7->NULL
Input: 8->12->10->5->4->1->6->NULL
Output: 8->12->10->4->6->5->1->NULL
// If all numbers are even then do not change the list
Input: 8->12->10->NULL
Output: 8->12->10->NULL
// If all numbers are odd then do not change the list
Input: 1->3->5->7->NULL
Output: 1->3->5->7->NULL
方法:
这个想法是获取指向列表最后一个节点的指针。然后从头节点开始遍历列表,将奇值节点从当前位置移动到列表末尾。
感谢 blunderboy 提出这种方法。
算法:
- 获取指向最后一个节点的指针。
- 将所有奇数节点移动到最后。
- 考虑第一个偶数节点之前的所有奇数节点并将它们移动到末尾。
- 将头指针更改为指向第一个偶数节点。
- 考虑第一个偶数节点之后的所有奇数节点并将它们移动到末尾。
C
// C program to segregate even and
// odd nodes in a Linked List
#include
#include
// A node of the singly linked list
struct Node
{
int data;
struct Node *next;
};
void segregateEvenOdd(struct Node **head_ref)
{
struct Node *end = *head_ref;
struct Node *prev = NULL;
struct Node *curr = *head_ref;
// Get pointer to the last node
while (end->next != NULL)
end = end->next;
struct Node *new_end = end;
/* Consider all odd nodes before the
first even node and move then after
end */
while (curr->data % 2 != 0 &&
curr != end)
{
new_end->next = curr;
curr = curr->next;
new_end->next->next = NULL;
new_end = new_end->next;
}
// 10->8->17->17->15
/* Do following steps only if there
is any even node */
if (curr->data % 2 == 0)
{
/* Change the head pointer to point
to first even node */
*head_ref = curr;
/* Now current points to the first
even node */
while (curr != end)
{
if ( (curr->data) % 2 == 0 )
{
prev = curr;
curr = curr->next;
}
else
{
/* Break the link between prev
and current */
prev->next = curr->next;
// Make next of curr as NULL
curr->next = NULL;
// Move curr to end
new_end->next = curr;
// Make curr as new end of list
new_end = curr;
/* Update current pointer to next
of the moved node */
curr = prev->next;
}
}
}
/* We must have prev set before executing
lines following this statement */
else prev = curr;
/* If there are more than 1 odd nodes and
end of original list is odd then move
this node to end to maintain same order
of odd numbers in modified list */
if (new_end != end &&
(end->data) % 2 != 0)
{
prev->next = end->next;
end->next = NULL;
new_end->next = end;
}
return;
}
// UTILITY FUNCTIONS
/* Function to insert a node at
the beginning */
void push(struct Node** head_ref,
int new_data)
{
// Allocate node
struct Node* new_node =
(struct Node*) malloc(sizeof(struct Node));
// Put in the data
new_node->data = new_data;
// Link the old list off the new node
new_node->next = (*head_ref);
// Move the head to point to the new node
(*head_ref) = new_node;
}
// Function to print nodes in
// a given linked list
void printList(struct Node *node)
{
while (node!=NULL)
{
printf("%d ", node->data);
node = node->next;
}
}
// Driver code
int main()
{
// Start with the empty list
struct Node* head = NULL;
/* Let us create a sample linked list
as following 0->2->4->6->8->10->11 */
push(&head, 11);
push(&head, 10);
push(&head, 8);
push(&head, 6);
push(&head, 4);
push(&head, 2);
push(&head, 0);
printf("Original Linked list ");
printList(head);
segregateEvenOdd(&head);
printf("Modified Linked list ");
printList(head);
return 0;
}
输出:
Original Linked list 0 2 4 6 8 10 11
Modified Linked list 0 2 4 6 8 10 11
时间复杂度: O(n)
有关详细信息,请参阅有关在链接列表中分离偶数和奇数节点的完整文章!