Mercurial > hg > Gears > GearsAgda
view doc/Interface.mm @ 466:831b7f6fd687
Fix warning pointer type
author | Tatsuki IHA <innparusu@cr.ie.u-ryukyu.ac.jp> |
---|---|
date | Thu, 21 Dec 2017 18:52:21 +0900 |
parents | 6f873aad1b06 |
children |
line wrap: on
line source
Interfaceのtypedef はコールフレームを定義する Interfaceの呼び出しの時に使える引数はtypedefに定義されている必要がある ... は呼び出し側のコールフレームを保存するのに使う typedef struct Stack<Type, Impl>{ Type* stack; Type* data; Type* data1; __code whenEmpty(...); __code clear(Impl* stack,__code next(...)); __code push(Impl* stack,Type* data, __code next(...)); __code pop(Impl* stack, __code next(Type* data, ...)); __code pop2(Impl* stack, __code next(Type* data, Type* data1, ...)); __code isEmpty(Impl* stack, __code next(...), __code whenEmpty(...)); __code get(Impl* stack, __code next(Type* data, ...)); __code get2(Impl* stack, __code next(Type* data, Type* data1, ...)); __code next(...); } Stack; 呼び出し方の例 goto nodeStack->push(newNode, replaceNode1); newNode はdataに対応する replaceNode1はnextに対応する。 replaceNode1のコールフレームは...に格納される。 つまり、replaceNode1はCodeGearのクロージャに相当する。 Interfaceから値を返す場合は継続経由で値を返す __code get2(Impl* stack, __code next(Type* data, Type* data1, ...)); 継続の型はInterfaceで定義されていて、この型に合うCodeGearを引数として渡す goto nodeStack->get2(insertCase1,tree) //意味的にはtreeの後ろに... __code insertCase1(struct Node *parent, struct Node *grandparent, struct RedBlackTree* tree) { //こっちも後ろに...があるはず goto next(data, data1, ...); createはinterfaceの実装を定義する interfaceのメソッドの番号をここで指定する implimentation側のDataGearは格納される実装依存の状態を持つ struct SingleLinkedStack { struct Element* top; } SingleLinkedStack; Stack* createSingleLinkedStack(struct Context* context) { struct Stack* stack = new Stack(); struct SingleLinkedStack* singleLinkedStack = new SingleLinkedStack(); stack->stack = (union Data*)singleLinkedStack; singleLinkedStack->top = NULL; stack->push = C_pushSingleLinkedStack; stack->pop = C_popSingleLinkedStack; stack->pop2 = C_pop2SingleLinkedStack; stack->get = C_getSingleLinkedStack; stack->get2 = C_get2SingleLinkedStack; stack->isEmpty = C_isEmptySingleLinkedStack; stack->clear = C_clearSingleLinkedStack; return stack; } 実装内部で使うCodeGearの引数はコールフレームで決まる コールフレームに含まれない中間変数を使っても良いが、辻褄は合ってる必要はある 一般的にはコールフレームに含まれている引数しか書けない 実装側の引数を書いても良い(ようにするか?) 実装の状態にアクセスする時にはコールフレーム内の実装を表すDataGearから取り出してくる __code replaceNode1(struct RedBlackTree* tree, struct Node* node, __code next(...)) { 呼び出しの例 goto insertNode(tree, tree->nodeStack, node);