このドキュメントは http://edu.net.c.dendai.ac.jp/ 上で公開されています。
複数のデータを扱うのに従来は配列を使用してきました。 これはあらゆるプログラミング言語に存在するデータ形式です。 メモリーのある区画に連続してデータを配置し、添字を利用してデータにアク セスします。 しかし、配列は前もって領域を確保する必要があるため、未知のサイズのデー タを取り扱うことができません。 また、そのため、コンピュータのメモリ量に応じて処理を行なうこともできま せん。
未知の容量のデータ対してメモリの使用量を制御するためには、 新たなデータ構造を使用する必要があります。 そのためには、まとめて大きな領域を確保するような記憶方法ではなく、デー タ一つずつに対して、一つずつメモリ領域を確保するような記憶法が必要とな ります。
線形リストとは図のような特殊な二分木です。
これを計算機上で取り扱うには通常、左の値が入る葉とそうでないノードを結 合し、頂点に値が入るものとして扱った方が便利です。 つまり次の図のようにして実現するのが便利です。
このような構造を作るに動的なメモリの確保をする必要があります。 まず各ノードに対応する構造体を定義します。
typedef struct lst {
struct lst *next;
char item;
} LIST;
まず、LIST型の変数を利用して、線形リストを表現することを考えます。 管理変数はLISTのポインタ型で、各データを表現しているのをLIST型の変数で 表します。 すると、次のように表せます。
LIST* kanri;
LIST a,b,c,nil;
a.item='a';
b.item='b';
c.item='c';
nil.next=NULL;
kanri=&a;
a.next=&b;
b.next=&c;
c.next=&nil;
これで、図と同等の線形リストを表現できました。
次に、この線形リストに対して、データを順に取り出すことを考えます。 線形リストは、途中から見ても線形リストの構造になってますから、再帰的な プログラムを組むことができます。 つまり、データを取り出すには次のようなアルゴリズムを考えれば良いことに なります。
例えば、各文字を一行ずつ表示するshow(LIST* p)関数は次のように作れます。
void show(LIST* p){
if(p->next == NULL) return;
printf("%c\n", p->item);
show(p->next);
}
次に、データ一つ一つに対して、この構造体を作り、データを入れることを考 えます。 そのためには、ポインタで、次のデータの入っている位置を管理します。 そうすることで、あらか じめデータの数を決めず、いくつでもデータを取り扱うことができます。 そのため、この構造体を malloc を利用してメモリ上に作り、ポインタの操作 でこの線形リスト構造ができるようにします。
例えば、要素を a, b, c の順に与えて、上の図のような線形リストを作るこ とを考えます。つまり、メインルーチンでは add('a'), add('b'), add('c') を順番に呼び出すだけで上の構造を作ることを考えます。 つまり、 add を呼び出すたびに次のようにリストが作られていきます。
そのために、 add() が何をするかを考えます。 add() として次のような処理が考えられます。
但し、 ノードを追加するのに、新たに作ったノードに指定された値を入れ、nil ノー ドの直前にあった頂点のポインタ(図の赤い矢印)を変更する必要があります。 これは nil ノードの前の頂点を覚える必要があります。 また、管理変数を変更する必要があるため、管理変数のポインタを関数に渡す ことになったります。
しかし、次のようにするとそのような複雑な状況を解消できます。 それは、nil 頂点にデータを入れ、新しく nil 頂点を作るという方法です。 この方法だと、もともとの nil 頂点だけに注目して全ての作業ができます。 この方法で作成したのが以下のプログラムです。
#include <stdio.h>
#include <stdlib.h>
typedef struct lst {
struct lst *next;
char item;
} LIST;
LIST* newnode(void){
LIST *p;
p=malloc(sizeof(LIST));
if(p!=NULL){
p->next=NULL;
p->item='\0';
}
return p;
}
LIST* add(LIST *p,char c){
while(p->next != NULL){
p= p->next;
}
if((p->next=newnode())!=NULL){
p->item=c;
}
return p;
}
void show(LIST* p){
if(p->next ==NULL)return;
printf("%c\n",p->item);
show(p=p->next);
}
void delnode(LIST* p){
if(p!=NULL){
delnode(p->next);
free(p);
}
}
int main(void){
LIST *pointer;
if((pointer=newnode())==NULL){
fprintf(stderr,"メモリを確保できませんでした\n");
return 1;
}
if((add(pointer,'a')!=NULL) &&
(add(pointer,'b')!=NULL) &&
(add(pointer,'c')!=NULL)){
show(pointer);
}
delnode(pointer);
return 0;
}
このプログラムで add 関数が && でつながれています。 C 言語の && 演算子は左から値を解釈していきます。 但し、途中で 0 を発見したらそれ以降の解釈は止めます。 したがって、この例ではもし途中で add が NULL を返してきたら、それ以上 他の add は行なわれず、式全体の値が偽(0)に確定します。 つまり if 文として指定した処理は行われないことになり、メモリーオーバ時 の正しい対処になります。
なお、このような先に入れたものを先に出力するような記憶方式を先入 れ先出し方式(First In First Out FIFO) と言います。 一方、これとは逆に先入れ後出し方式(First In Last Out FILO)の記憶方式をスタックと言います。 これは次回に講義します。
なお add 関数、show 関数は再帰処理を使うと次のようにも書けます。
LIST* add(LIST *p,char c){
if(p->next != NULL){
return add(p->next,c);
}
if((p->next=newnode())!=NULL){
p->item=c;
}
return p;
}
void show(LIST *p){
if(p->next == NULL){
return;
}
printf("%c\n",p->item);
show(p->next);
}
上で定義した線形リストに対して、文字列を入れられるように改造しなさい。 add のプロトタイプ宣言は次のようになります。
LIST* add(LIST *p, char *s)
そして次のプログラムを動かしなさい。
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
/* LIST の定義群 */
int main(void){
LIST *list;
if((list=newnode())==NULL){
fprintf(stderr,"メモリを確保できませんでした\n");
return 1;
}
if((add(list,"abc")!=NULL) &&
(add(list,"def")!=NULL) &&
(add(list,"ghi")!=NULL)){
show(list);
}
delnode(list);
return 0;
}
テキストファイルをそのまま線形リストにしまい、出力 時に行を逆順に表示するプログラムを作りなさい。
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#define SIZE 80
char buffer[SIZE+1];
int main(void){
int c;
int i=0;
char filename[]="input.txt";
FILE *fh;
errno_t errorcode;
if((errorcode=fopen_s(&fh,filename,"r"))!=0){
fprintf(stderr,"ファイル %s がありません。エラーコードは%dです。\n",filename,errorcode);
return errorcode;
}
while((c=getc(fh))!=EOF){
if(c=='\n'){
buffer[i]='\0';
printf("%s\n",buffer);
i=0;
}else{
if(i>=SIZE){
fprintf(stderr,"一行の長さを越えました。\n");
return 2;
}
buffer[i++]=c;
}
}
return 0;
}
if(c=='\n'){
buffer[i]='\0';
add(pointer,_strdup(buffer));
i=0;
}else{
void reverseshow(LIST *p){
if(p->next == NULL){
return;
}
reverseshow(p->next); /* 始めに後ろを表示させてから */
printf("%s\n",p->item);/* ポインタの位置の行を表示する */
}
なお、文字列へのポインタを入力として、新しい領域に内容をコピーしてその 領域へのポインタを返す関数 _strdup() を使うと上の処理は次のように簡単に なります。
#include <string.h>
char buffer[81]; /* 文字数+1 */
while(ファイルの終りまで){
bufferに一行(最大80文字)読む;
add(_strdup(buffer));
}
buffer と呼ばれる領域の先頭番地は変化しないため、 add(buffer) では毎回 同じ番地が登録されてしまうことに注意して下さい。
fgets(読み込み用バッファの番地, 文字数, ファイルハンドル) 関数を使うと 一行をまとめて読み込むことができます。 但し、文字数分だけいっぱいに読んでしまったかどうかは最終文字が改行記号 かどうかで判定する必要があります。
前述の線形リストの add は毎回 nil の位置を最初から検索していました。 でも、これは覚えておけば検索する必要はありません。 そこで、先頭のノードの位置と、nil の位置両方を覚えるようにします。 この二つの管理変数を一つの変数で取り扱うようにするため、管理変数自体も 構造体にします。 今まで LIST と呼んでいたデータを入れる部分を NODE と呼ぶことにし、新た に管理変数二つの入った構造体を LIST と呼ぶことにします。 すると下記のようなプログラムになります。 このようにすると、全体のデータ量とは無関係に一定時間でデータの追加がで きます。
#include <stdio.h>
#include <stdlib.h>
typedef struct node {
struct node *next;
char *data;
} NODE;
typedef struct list {
NODE *top,*bottom;
} LIST;
LIST* newlist(void){
LIST* l;
NODE* n;
if((l=malloc(sizeof(LIST)))==NULL){
return NULL;
}
if((n=malloc(sizeof(NODE)))==NULL){
free(l);
return NULL;
}
n->next=NULL;
n->data=NULL;
l->top=l->bottom=n;
return l;
}
int add(LIST *list, char *value){
NODE *newnode;
if((newnode=malloc(sizeof(NODE)))==NULL){
return 0;
}
newnode->next=NULL;
list->bottom->next=newnode;
list->bottom->data=value;
list->bottom=newnode;
return 1;
}
void show(LIST *list){
NODE *p;
for(p=list->top; p!=list->bottom; p=p->next){
printf("%s\n",p->data);
}
}
void dellist(LIST *list){
NODE *p,*q;
p=list->top;
do{
q=p;
p=p->next;
free(q);
}while(p!=NULL);
free(list);
}
int main(void){
LIST *list;
list=newlist();
if(list!=NULL){
add(list,"abc");
add(list,"def");
add(list,"ghi");
show(list);
dellist(list);
}
return 0;
}
C++ では Standard Template Library (STL) により線形リストを 提供しています。 線形リストの(オブジェクト)変数の定義をした後、「変数名.操作()」という 書式で操作します。 また、ポインタを制限した形のイテレータ(反復子)という概念が 導入され、リスト上の操作はイテレータを使用して行われます。 例えば、一番最初の例と同じプログラムを C++ で実現させると次のようにな ります。 なお C++ ではシステムの提供するものには std:: をつける必要があります。
#include <iostream>
#include <list>
int main(){
std::list<char> l;
l.insert(l.end(),'a');
l.insert(l.end(),'b');
l.insert(l.end(),'c');
for(std::list<char>::iterator i=l.begin(),e=l.end();
i!=e; i++){
std::cout << *i << std::endl;
}
return 0;
}
双方向リストとは、次の頂点へのポインタと前の頂点へのポインタの両方を保 持するものです。 特定の頂点に注目して処理する場合に便利です。 エディタなど、特定の行に注目した処理をするのに用いられます。 実は C++ の list は双方向リストです。 C++ では Iterator に対して -- 演算が可能です。
双方向リストを使って less を作りなさい。 less とは more を拡張したコマンドで、次の動作をします。
int main(int argc, char *argv[]){
FILE *input;
BLIST *bl;
char c;
bl=newnode();
/*ファイル名のチェック*/
/*ファイルのオープン*/
readfile(input,bl);
firstline(bl);
show();
while((c=getchar())!='q'){
switch(c){
case ' ': case 'd': addline(bl,24); break;
case 'u': addline(bl,-24); break;
case '<': firstline(bl); break;
case '>': lastline(bl); break;
default: continue;
}
show();
}
fclose(input);
return 0;
}
#include <stdio.h>
#include <stdlib.h>
typedef struct blst {
char *item;
struct blst *next;
struct blst *previous;
} BLIST;
void add(BLIST *base, BLIST *pointer, char *i){
BLIST *newnode;
if((newnode=malloc(sizeof(BLIST)))==NULL){
fprintf(stderr,"メモリーを確保できませんでした\n");
exit(2);
}
newnode->item = i;
newnode->next = pointer->next;
if(pointer->next == NULL){
newnode->previous = base->previous;
base->previous = newnode;
}else{
newnode->previous = (pointer->next)->previous;
(pointer->next)->previous = newnode;
}
pointer->next = newnode;
}
void printlist(BLIST *base){
BLIST *p=base;
while((p=p->next)!=NULL){
printf("%s ",p->item);
}
printf("\n");
}
void printreverse(BLIST *base){
BLIST *p=base;
while((p=p->previous)!=NULL){
printf("%s ",p->item);
}
printf("\n");
}
int empty(BLIST *base){
return base->next == NULL;
}
int size(BLIST *base){
BLIST *p=base;
int i=0;
while((p=p->next) != NULL){
i++;
}
return i;
}
int removenode(BLIST *base, BLIST *pointer){
BLIST *p,*n;
if(base==pointer){
return -1;
}
p = pointer->previous;
n = pointer->next;
if(p==NULL){
base->next = n;
}else{
p->next = n;
}
if(n==NULL){
base->previous = p;
}else{
n->previous = p;
}
/* free(pointer->item) */
free(pointer);
}
int main(void){
BLIST *l;
BLIST *p;
if((l=malloc(sizeof(BLIST)))==NULL){
fprintf(stderr,"メモリーを確保できませんでした\n");
exit(2);
}
l->next=l->previous=NULL;
add(l,l,"abc");
printlist(l); printreverse(l); printf("size = %d\n",size(l));
add(l,l,"def");
printlist(l); printreverse(l); printf("size = %d\n",size(l));
add(l,l->previous,"ghi");
printlist(l); printreverse(l); printf("size = %d\n",size(l));
add(l,l->previous,"jkl");
printlist(l); printreverse(l); printf("size = %d\n",size(l));
add(l,l->next->next->next,"mno");
printlist(l); printreverse(l); printf("size = %d\n",size(l));
while(!empty(l)){
printlist(l); printreverse(l); printf("size = %d\n",size(l));
removenode(l, l->next);
}
return 0;
}
以下はコマンドラインからファイル名を指定すると、そのファイルを線形リス トに入れた後、線形リストの内容を表示するプログラムです。
線形リストに入れる時、毎回全線形リストをたどるという非効率な実装 がされています。線形リストの最後の要素を指すポインタを設定し、そこに対 して挿入するようにするとかなり効率が良くなります。
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define BUFFSIZE 80
typedef struct lst {
char *item;
struct lst *next;
} LIST;
LIST *root;
void add(char *c){
LIST *p=root;
while(p->next != NULL){
p= p->next;
}
p->next=malloc(sizeof(LIST));
p->item=c;
(p->next)->next=NULL;
}
void show(){
LIST *p=root;
while(p->next !=NULL){
printf("%s",p->item);
p=p->next;
}
}
int main(int argc, char *argv[]){
FILE *fp;
char buffer[BUFFSIZE];
char *s;
errno_t errorcode;
if((root=malloc(sizeof(LIST)))==NULL){
fprintf(stderr,"メモリーを確保できませんでした\n");
exit(2);
}
root->next=NULL;
root->item=NULL;
if(argc <2 ){
fprintf(stderr, "Specify the file name.\n");
exit(3);
}
if((errorcode=fopen_s(&fp,argv[1],"r"))!=0){
fprintf(stderr,"File %s Not Found. Error code is %d.\n",argv[1],errorcode);
return errorcode;
}
while(fgets(buffer,BUFFSIZE, fp)!=NULL){
if((s=malloc(strlen(buffer)*sizeof(char)))==NULL){
fprintf(stderr,"メモリーを確保できませんでした\n");
exit(2);
}
strcpy(s,buffer);
printf("%s\n",s);
add(s);
}
show();
return 0;
}
#include <iostream>
#include <fstream>
#include <string>
#include <list>
#define BUFFSIZE 80
using std::cerr;
using std::endl;
int main(int argc, char *argv[]){
if(argc<2){
std::cerr << "Specify the file name." << std::endl;
return 1;
}
std::ifstream ifs(argv[1], std::ios::in);
if(!ifs){
std::cerr << "File Not Found." << std::endl;
return 2;
}
char buffer[BUFFSIZE];
std::list<std::string> l;
while(!ifs.eof()){
ifs.getline(buffer,BUFFSIZE);
l.insert(l.end(),buffer);
}
for(std::list<std::string>::iterator i=l.begin(),j=l.end();i!=j;i++){
std::cout << *i << std::endl;
}
return 0;
}