Функции проходят все тесты

This commit is contained in:
alexey
2025-12-11 11:32:06 +03:00
parent daae42bfd8
commit 3c9bd4a703

213
list.c
View File

@@ -1,136 +1,239 @@
#include <assert.h> #include <assert.h>
#include <string.h> #include <string.h>
#include "util.h" #include "util.h"
#include "list.h" #include "list.h"
void void
list_init(list *l) list_init(list *l)
{ {
assert(l); assert(l);
memset(l, 0, sizeof(*l)); memset(l, 0, sizeof(*l));
} }
static inline list_node * static inline list_node *
list_node_orphan(list_node *li) list_node_orphan(list_node *li)
{ {
li->next = NULL; li->next = NULL;
li->prev = NULL; li->prev = NULL;
return li; return li;
} }
//NOTE: Можем ли мы расчитывать на то, что node->next валидно после вызова list_free_cb??? //NOTE: Можем ли мы расчитывать на то, что node->next валидно после вызова list_free_cb???
void void
list_clear(list *l, list_free_cb cb) list_clear(list *l, list_free_cb cb)
{ {
assert(l); assert(l);
// <YOURCODE>
list_node *cur = l->head;
while (cur) {
list_node *next_node = cur->next;
if (cb)
cb(cur);
cur = next_node;
}
l->head = NULL;
l->tail = NULL;
} }
void void
list_push_front(list *l, list_node *item) list_push_front(list *l, list_node *item)
{ {
assert(l && item); assert(l && item);
// <YOURCODE>
item->prev = NULL;
item->next = l->head;
if (l->head)
l->head->prev = item;
else
l->tail = item;
l->head = item;
} }
void void
list_push_back(list *l, list_node *item) list_push_back(list *l, list_node *item)
{ {
assert(l && item); assert(l && item);
// <YOURCODE>
item->next = NULL;
item->prev = l->tail;
if (l->tail)
l->tail->next = item;
else
l->head = item;
l->tail = item;
} }
list_node * list_node *
list_first(list *l) list_first(list *l)
{ {
assert(l); assert(l);
return l->head; return l->head;
} }
list_node * list_node *
list_last(list *l) list_last(list *l)
{ {
assert(l); assert(l);
return l->tail; return l->tail;
} }
size_t size_t
list_len(list *l) list_len(list *l)
{ {
assert(l); assert(l);
// <YOURCODE>
return 0; size_t cnt = 0;
for (list_node *cur = l->head; cur; cur = cur->next)
cnt++;
return cnt;
} }
list_node * list_node *
list_get(list *l, int idx) list_get(list *l, int idx)
{ {
assert(l); assert(l);
// <YOURCODE>
return NULL; if (idx < 0)
return NULL;
int i = 0;
for (list_node *cur = l->head; cur; cur = cur->next, i++)
if (i == idx)
return cur;
return NULL;
} }
void void
list_insert(list *l, list_node *item, int idx) list_insert(list *l, list_node *item, int idx)
{ {
assert(l && item); assert(l && item);
// <YOURCODE>
int len = (int)list_len(l);
if (idx < 0 || idx > len)
xerror(1, "index out of range\n");
if (idx == 0) {
list_push_front(l, item);
return;
}
if (idx == len) {
list_push_back(l, item);
return;
}
list_node *pos = list_get(l, idx);
assert(pos);
item->prev = pos->prev;
item->next = pos;
pos->prev->next = item;
pos->prev = item;
} }
list_node * list_node *
list_pop_front(list *l) list_pop_front(list *l)
{ {
assert(l); assert(l);
// <YOURCODE>
return NULL; if (!l->head)
return NULL;
list_node *res = l->head;
l->head = res->next;
if (l->head)
l->head->prev = NULL;
else
l->tail = NULL;
return list_node_orphan(res);
} }
list_node * list_node *
list_pop_back(list *l) list_pop_back(list *l)
{ {
assert(l); assert(l);
// <YOURCODE>
return NULL; if (!l->tail)
return NULL;
list_node *res = l->tail;
l->tail = res->prev;
if (l->tail)
l->tail->next = NULL;
else
l->head = NULL;
return list_node_orphan(res);
} }
int int
list_index(list *l, list_node *n) list_index(list *l, list_node *n)
{ {
assert(l && n); assert(l && n);
list_node *tmp;
int i = 0; int i = 0;
for (tmp = l->head; tmp != NULL; tmp = tmp->next) { for (list_node *cur = l->head; cur; cur = cur->next, i++)
if (tmp == n) if (cur == n)
return i; return i;
i++;
} return -1;
return -1;
} }
list_node * list_node *
list_remove(list *l, int idx) list_remove(list *l, int idx)
{ {
assert(l); assert(l);
// <YOURCODE>
return NULL; list_node *cur = list_get(l, idx);
if (!cur)
return NULL;
if (cur->prev)
cur->prev->next = cur->next;
else
l->head = cur->next;
if (cur->next)
cur->next->prev = cur->prev;
else
l->tail = cur->prev;
return list_node_orphan(cur);
} }
void void
list_reverse(list *l) list_reverse(list *l)
{ {
assert(l); assert(l);
// <YOURCODE>
list_node *cur = l->head;
list_node *tmp = NULL;
while (cur) {
tmp = cur->next;
cur->next = cur->prev;
cur->prev = tmp;
cur = tmp;
}
tmp = l->head;
l->head = l->tail;
l->tail = tmp;
} }
void
list_pass(list *l, list_pass_cb cb, void *opaq)
{
struct list_node *iter;
list_for_each(iter, l) {
cb(iter, opaq);
}
}