双方向連結リストをC言語でやってみる

AOJ ALDS1 3-C Doubly linked listをやったのでノート代わりに書く。

 

テンプレートパワーで実装

 

双方向連結リスト

・やりたいことの図を書いて動きを追ってそれをプログラムに落とし込む

・deleteLastの処理を最初tail部分を作ってやろうとしたが循環リストにすると(カンニング)とてもスッキリした。

nilがわかりづらいのでheadに変えるととても良し

 

コード

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

struct node{
  unsigned int key;
  struct node *next;
  struct node *prev;
};

typedef struct node * NodePointer;

NodePointer head;

NodePointer listSearch(int key){
  /* your code */
  NodePointer cur = head->next;
  while(cur->key!=key && cur!= head){
      cur=cur->next;
  }
  return cur;
}

void init(){
  head = malloc(sizeof(struct node));
  head->next = head;
  head->prev = head;
}

void printList(){
  NodePointer cur = head->next;
  int isf = 1;
  while(1){
    if ( cur == head ) break;
    if ( isf == 0)  printf(" ");
    printf("%d", cur->key);
    cur = cur->next;
    isf = 0;
  }
  printf("\n");
}

void deleteNode(NodePointer t){
  /* your code */
  t->prev->next=t->next;
  t->next->prev=t->prev;
  free(t);
}

void deleteFirst(){
  NodePointer t = head->next;
  if ( t == head ) return;
  deleteNode(t);
}

void deleteLast(){
  /* your code */
  NodePointer t = head->prev;
  if ( t == head ) return;
  deleteNode(t);
}

void delete(int key){
  /* your code */
  NodePointer t = listSearch(key);
  if ( t == head ) return;
  deleteNode(t);
}


void insert(int key){
  NodePointer x;
  x = malloc(sizeof(struct node));
  x->key = key;
 //一回目の挿入でhead->prevが末尾xに行ってる
  x->next=head->next;
  x->next->prev=x;
  head->next=x;
  x->prev=head;

}

int main(){
  int key, n, i;
  int size = 0;
  char com[20];
  int np = 0, nd = 0;
  scanf("%d", &n);
  init();
  for ( i = 0; i < n; i++ ){
    scanf("%s%d", com, &key);
    if ( com[0] == 'i' ) {insert(key); np++; size++;}
    else if ( com[0] == 'd' ) {
      if (strlen(com) > 6){
    if ( com[6] == 'F' ) deleteFirst();
    else if ( com[6] == 'L' ) deleteLast();
      } else {
    delete(key); nd++; 
      }
      size--;
    }
  }

  printList();
  return 0;
}