When to use “and” operator in Ocaml AST

Multi tool use
up vote
0
down vote
favorite
I am translating the rules of my grammar into an AST.
Is it necessary to use the "and" operator in defining our AST?
For instance, I have translated my grammar thus far like so:
type program =
| Decls of typ * identifier * decls_prime
type typ =
| INT
| BOOL
| VOID
type identifier = string
(* decls_prime = vdecl decls | fdecl decls *)
type declsprime =
| Vdecl of variabledeclaration * decls
| Fdecl of functiondeclaration * decls
(*“lparen” formals_opt “rparen” “LBRACE” vdecl_list stmt_list “RBRACE”*)
type functiondeclaration =
| Fdecl of variabledeclarationlist * stmtlist
(*formals_opt = formal_list | epsilon *)
type FormalsOpt =
|FormalsOpt of formallist
(* typ “ID” formal_list_prime *)
type formalList =
| FormalList of typ * identifier * formallistprime
type formallistprime =
| FormalListPrime of formalList
type variabledeclarationlist =
| VdeclList of variabledeclaration * variabledeclarationlist
(*stmt stmt_list | epsilon*)
type stmtlist =
| StmtList of stmt * stmtlist
| StmtlistNil
(* stmt = “RETURN” stmt_prime| expr SEMI |“LBRACE” stmt_list RBRACE| IF LPAREN expr RPAREN stmt stmt_prime_prime| FOR LPAREN expr_opt SEMI expr SEMI expr_opt RPAREN stmt| WHILE LPAREN expr RPAREN stmt*)
type Stmt
| Return of stmtprime
| Expression of expr
| StmtList of stmtlist
| IF of expr * stmt * stmtprimeprime
| FOR of expropt * expr * expropt * stmt
| WHILE of expr * stmt
(*stmt_prime = SEMI| expr SEMI*)
type stmtprime
| SEMI
| Expression of expr
(*NOELSE | ELSE stmt*)
type stmtprimeprime
| NOELSE
| ELSE of stmt
(* Expr_opt = expr | epsilon *)
type expropt =
| Expression of expr
| ExprNil
type Expr
type ExprPrime
(* Actuals_opt = actuals_list | epsilon *)
type ActualsOpt=
| ActualsList of actualslist
| ActualsNil
type ActualsList =
| ActualsList of expr * actualslistprime
(*actualslistprime = COMMA expr actuals_list_prime | epsilon*)
type actualslistprime =
| ActualsListPrime of expr * actualslistprime
| ALPNil
But it looks as though this example from Illinois uses a slightly different structure:
type program = Program of (class_decl list)
and class_decl = Class of id * id * (var_decl list) * (method_decl list)
and method_decl = Method....
Is it necessary to use "and" when defining my AST? And moreover, is it wrong for me to use a StmtList type rather than (stmt list) even though I call the AST StmtList method correctly in my parser?
parsing ocaml abstract-syntax-tree
add a comment |
up vote
0
down vote
favorite
I am translating the rules of my grammar into an AST.
Is it necessary to use the "and" operator in defining our AST?
For instance, I have translated my grammar thus far like so:
type program =
| Decls of typ * identifier * decls_prime
type typ =
| INT
| BOOL
| VOID
type identifier = string
(* decls_prime = vdecl decls | fdecl decls *)
type declsprime =
| Vdecl of variabledeclaration * decls
| Fdecl of functiondeclaration * decls
(*“lparen” formals_opt “rparen” “LBRACE” vdecl_list stmt_list “RBRACE”*)
type functiondeclaration =
| Fdecl of variabledeclarationlist * stmtlist
(*formals_opt = formal_list | epsilon *)
type FormalsOpt =
|FormalsOpt of formallist
(* typ “ID” formal_list_prime *)
type formalList =
| FormalList of typ * identifier * formallistprime
type formallistprime =
| FormalListPrime of formalList
type variabledeclarationlist =
| VdeclList of variabledeclaration * variabledeclarationlist
(*stmt stmt_list | epsilon*)
type stmtlist =
| StmtList of stmt * stmtlist
| StmtlistNil
(* stmt = “RETURN” stmt_prime| expr SEMI |“LBRACE” stmt_list RBRACE| IF LPAREN expr RPAREN stmt stmt_prime_prime| FOR LPAREN expr_opt SEMI expr SEMI expr_opt RPAREN stmt| WHILE LPAREN expr RPAREN stmt*)
type Stmt
| Return of stmtprime
| Expression of expr
| StmtList of stmtlist
| IF of expr * stmt * stmtprimeprime
| FOR of expropt * expr * expropt * stmt
| WHILE of expr * stmt
(*stmt_prime = SEMI| expr SEMI*)
type stmtprime
| SEMI
| Expression of expr
(*NOELSE | ELSE stmt*)
type stmtprimeprime
| NOELSE
| ELSE of stmt
(* Expr_opt = expr | epsilon *)
type expropt =
| Expression of expr
| ExprNil
type Expr
type ExprPrime
(* Actuals_opt = actuals_list | epsilon *)
type ActualsOpt=
| ActualsList of actualslist
| ActualsNil
type ActualsList =
| ActualsList of expr * actualslistprime
(*actualslistprime = COMMA expr actuals_list_prime | epsilon*)
type actualslistprime =
| ActualsListPrime of expr * actualslistprime
| ALPNil
But it looks as though this example from Illinois uses a slightly different structure:
type program = Program of (class_decl list)
and class_decl = Class of id * id * (var_decl list) * (method_decl list)
and method_decl = Method....
Is it necessary to use "and" when defining my AST? And moreover, is it wrong for me to use a StmtList type rather than (stmt list) even though I call the AST StmtList method correctly in my parser?
parsing ocaml abstract-syntax-tree
add a comment |
up vote
0
down vote
favorite
up vote
0
down vote
favorite
I am translating the rules of my grammar into an AST.
Is it necessary to use the "and" operator in defining our AST?
For instance, I have translated my grammar thus far like so:
type program =
| Decls of typ * identifier * decls_prime
type typ =
| INT
| BOOL
| VOID
type identifier = string
(* decls_prime = vdecl decls | fdecl decls *)
type declsprime =
| Vdecl of variabledeclaration * decls
| Fdecl of functiondeclaration * decls
(*“lparen” formals_opt “rparen” “LBRACE” vdecl_list stmt_list “RBRACE”*)
type functiondeclaration =
| Fdecl of variabledeclarationlist * stmtlist
(*formals_opt = formal_list | epsilon *)
type FormalsOpt =
|FormalsOpt of formallist
(* typ “ID” formal_list_prime *)
type formalList =
| FormalList of typ * identifier * formallistprime
type formallistprime =
| FormalListPrime of formalList
type variabledeclarationlist =
| VdeclList of variabledeclaration * variabledeclarationlist
(*stmt stmt_list | epsilon*)
type stmtlist =
| StmtList of stmt * stmtlist
| StmtlistNil
(* stmt = “RETURN” stmt_prime| expr SEMI |“LBRACE” stmt_list RBRACE| IF LPAREN expr RPAREN stmt stmt_prime_prime| FOR LPAREN expr_opt SEMI expr SEMI expr_opt RPAREN stmt| WHILE LPAREN expr RPAREN stmt*)
type Stmt
| Return of stmtprime
| Expression of expr
| StmtList of stmtlist
| IF of expr * stmt * stmtprimeprime
| FOR of expropt * expr * expropt * stmt
| WHILE of expr * stmt
(*stmt_prime = SEMI| expr SEMI*)
type stmtprime
| SEMI
| Expression of expr
(*NOELSE | ELSE stmt*)
type stmtprimeprime
| NOELSE
| ELSE of stmt
(* Expr_opt = expr | epsilon *)
type expropt =
| Expression of expr
| ExprNil
type Expr
type ExprPrime
(* Actuals_opt = actuals_list | epsilon *)
type ActualsOpt=
| ActualsList of actualslist
| ActualsNil
type ActualsList =
| ActualsList of expr * actualslistprime
(*actualslistprime = COMMA expr actuals_list_prime | epsilon*)
type actualslistprime =
| ActualsListPrime of expr * actualslistprime
| ALPNil
But it looks as though this example from Illinois uses a slightly different structure:
type program = Program of (class_decl list)
and class_decl = Class of id * id * (var_decl list) * (method_decl list)
and method_decl = Method....
Is it necessary to use "and" when defining my AST? And moreover, is it wrong for me to use a StmtList type rather than (stmt list) even though I call the AST StmtList method correctly in my parser?
parsing ocaml abstract-syntax-tree
I am translating the rules of my grammar into an AST.
Is it necessary to use the "and" operator in defining our AST?
For instance, I have translated my grammar thus far like so:
type program =
| Decls of typ * identifier * decls_prime
type typ =
| INT
| BOOL
| VOID
type identifier = string
(* decls_prime = vdecl decls | fdecl decls *)
type declsprime =
| Vdecl of variabledeclaration * decls
| Fdecl of functiondeclaration * decls
(*“lparen” formals_opt “rparen” “LBRACE” vdecl_list stmt_list “RBRACE”*)
type functiondeclaration =
| Fdecl of variabledeclarationlist * stmtlist
(*formals_opt = formal_list | epsilon *)
type FormalsOpt =
|FormalsOpt of formallist
(* typ “ID” formal_list_prime *)
type formalList =
| FormalList of typ * identifier * formallistprime
type formallistprime =
| FormalListPrime of formalList
type variabledeclarationlist =
| VdeclList of variabledeclaration * variabledeclarationlist
(*stmt stmt_list | epsilon*)
type stmtlist =
| StmtList of stmt * stmtlist
| StmtlistNil
(* stmt = “RETURN” stmt_prime| expr SEMI |“LBRACE” stmt_list RBRACE| IF LPAREN expr RPAREN stmt stmt_prime_prime| FOR LPAREN expr_opt SEMI expr SEMI expr_opt RPAREN stmt| WHILE LPAREN expr RPAREN stmt*)
type Stmt
| Return of stmtprime
| Expression of expr
| StmtList of stmtlist
| IF of expr * stmt * stmtprimeprime
| FOR of expropt * expr * expropt * stmt
| WHILE of expr * stmt
(*stmt_prime = SEMI| expr SEMI*)
type stmtprime
| SEMI
| Expression of expr
(*NOELSE | ELSE stmt*)
type stmtprimeprime
| NOELSE
| ELSE of stmt
(* Expr_opt = expr | epsilon *)
type expropt =
| Expression of expr
| ExprNil
type Expr
type ExprPrime
(* Actuals_opt = actuals_list | epsilon *)
type ActualsOpt=
| ActualsList of actualslist
| ActualsNil
type ActualsList =
| ActualsList of expr * actualslistprime
(*actualslistprime = COMMA expr actuals_list_prime | epsilon*)
type actualslistprime =
| ActualsListPrime of expr * actualslistprime
| ALPNil
But it looks as though this example from Illinois uses a slightly different structure:
type program = Program of (class_decl list)
and class_decl = Class of id * id * (var_decl list) * (method_decl list)
and method_decl = Method....
Is it necessary to use "and" when defining my AST? And moreover, is it wrong for me to use a StmtList type rather than (stmt list) even though I call the AST StmtList method correctly in my parser?
parsing ocaml abstract-syntax-tree
parsing ocaml abstract-syntax-tree
asked Nov 9 at 21:12
Matt
4731624
4731624
add a comment |
add a comment |
1 Answer
1
active
oldest
votes
up vote
2
down vote
accepted
You only need and
when your definitions are mutually recursive. That is, if a statement could contain an expression and an expression could in turn contain a statement, then Expr
and Stmt
would have to be connected with an and
. If your code compiles without and
, you don't need the and
.
PS: This is unrelated to your question, but I think it would make a lot more sense to use the list
and option
types than to define your own versions for specific types (such as stmntlist
, expropt
etc.). stmtprime
is another such case: You could just define Return
as Return of expr option
and get rid of the stmtprime
type. Same with stmtprimeprime
.
I had to write the parser by hand (no parser generator) which is why I created StmtPrime, etc AST types. How would I define option? Very new to OCaml
– Matt
Nov 9 at 21:30
@Matt I don't see why writing the parser by hand would affect what the AST looks like. "How would I define option?"option
is a type from the standard library - you don't need to define it. You can just writeReturn of stmt option
instead ofReturn of stmtprime
and that's it.
– sepp2k
Nov 9 at 21:34
But doesn't Stmt prime need to be built up? Because StmtPrime = Semi | Expression
– Matt
Nov 9 at 21:38
so technically, its not optional?
– Matt
Nov 9 at 21:38
1
@Matt I'm not following you.stmtprime
is a type that you defined that could either contain an expression or nothing.expr option
is also a type that could either contain an expression or nothing. Therefore these two types are completely interchangeable and thestmtprime
type is unnecessary. Whether the semicolon is optional or not is a property of your parser, not your AST. The fact that you happened to name your no-expression constructorSEMI
does not affect whether or not the parser expects a semicolon.
– sepp2k
Nov 9 at 21:48
|
show 3 more comments
1 Answer
1
active
oldest
votes
1 Answer
1
active
oldest
votes
active
oldest
votes
active
oldest
votes
up vote
2
down vote
accepted
You only need and
when your definitions are mutually recursive. That is, if a statement could contain an expression and an expression could in turn contain a statement, then Expr
and Stmt
would have to be connected with an and
. If your code compiles without and
, you don't need the and
.
PS: This is unrelated to your question, but I think it would make a lot more sense to use the list
and option
types than to define your own versions for specific types (such as stmntlist
, expropt
etc.). stmtprime
is another such case: You could just define Return
as Return of expr option
and get rid of the stmtprime
type. Same with stmtprimeprime
.
I had to write the parser by hand (no parser generator) which is why I created StmtPrime, etc AST types. How would I define option? Very new to OCaml
– Matt
Nov 9 at 21:30
@Matt I don't see why writing the parser by hand would affect what the AST looks like. "How would I define option?"option
is a type from the standard library - you don't need to define it. You can just writeReturn of stmt option
instead ofReturn of stmtprime
and that's it.
– sepp2k
Nov 9 at 21:34
But doesn't Stmt prime need to be built up? Because StmtPrime = Semi | Expression
– Matt
Nov 9 at 21:38
so technically, its not optional?
– Matt
Nov 9 at 21:38
1
@Matt I'm not following you.stmtprime
is a type that you defined that could either contain an expression or nothing.expr option
is also a type that could either contain an expression or nothing. Therefore these two types are completely interchangeable and thestmtprime
type is unnecessary. Whether the semicolon is optional or not is a property of your parser, not your AST. The fact that you happened to name your no-expression constructorSEMI
does not affect whether or not the parser expects a semicolon.
– sepp2k
Nov 9 at 21:48
|
show 3 more comments
up vote
2
down vote
accepted
You only need and
when your definitions are mutually recursive. That is, if a statement could contain an expression and an expression could in turn contain a statement, then Expr
and Stmt
would have to be connected with an and
. If your code compiles without and
, you don't need the and
.
PS: This is unrelated to your question, but I think it would make a lot more sense to use the list
and option
types than to define your own versions for specific types (such as stmntlist
, expropt
etc.). stmtprime
is another such case: You could just define Return
as Return of expr option
and get rid of the stmtprime
type. Same with stmtprimeprime
.
I had to write the parser by hand (no parser generator) which is why I created StmtPrime, etc AST types. How would I define option? Very new to OCaml
– Matt
Nov 9 at 21:30
@Matt I don't see why writing the parser by hand would affect what the AST looks like. "How would I define option?"option
is a type from the standard library - you don't need to define it. You can just writeReturn of stmt option
instead ofReturn of stmtprime
and that's it.
– sepp2k
Nov 9 at 21:34
But doesn't Stmt prime need to be built up? Because StmtPrime = Semi | Expression
– Matt
Nov 9 at 21:38
so technically, its not optional?
– Matt
Nov 9 at 21:38
1
@Matt I'm not following you.stmtprime
is a type that you defined that could either contain an expression or nothing.expr option
is also a type that could either contain an expression or nothing. Therefore these two types are completely interchangeable and thestmtprime
type is unnecessary. Whether the semicolon is optional or not is a property of your parser, not your AST. The fact that you happened to name your no-expression constructorSEMI
does not affect whether or not the parser expects a semicolon.
– sepp2k
Nov 9 at 21:48
|
show 3 more comments
up vote
2
down vote
accepted
up vote
2
down vote
accepted
You only need and
when your definitions are mutually recursive. That is, if a statement could contain an expression and an expression could in turn contain a statement, then Expr
and Stmt
would have to be connected with an and
. If your code compiles without and
, you don't need the and
.
PS: This is unrelated to your question, but I think it would make a lot more sense to use the list
and option
types than to define your own versions for specific types (such as stmntlist
, expropt
etc.). stmtprime
is another such case: You could just define Return
as Return of expr option
and get rid of the stmtprime
type. Same with stmtprimeprime
.
You only need and
when your definitions are mutually recursive. That is, if a statement could contain an expression and an expression could in turn contain a statement, then Expr
and Stmt
would have to be connected with an and
. If your code compiles without and
, you don't need the and
.
PS: This is unrelated to your question, but I think it would make a lot more sense to use the list
and option
types than to define your own versions for specific types (such as stmntlist
, expropt
etc.). stmtprime
is another such case: You could just define Return
as Return of expr option
and get rid of the stmtprime
type. Same with stmtprimeprime
.
edited Nov 9 at 21:42
answered Nov 9 at 21:21
sepp2k
290k36592604
290k36592604
I had to write the parser by hand (no parser generator) which is why I created StmtPrime, etc AST types. How would I define option? Very new to OCaml
– Matt
Nov 9 at 21:30
@Matt I don't see why writing the parser by hand would affect what the AST looks like. "How would I define option?"option
is a type from the standard library - you don't need to define it. You can just writeReturn of stmt option
instead ofReturn of stmtprime
and that's it.
– sepp2k
Nov 9 at 21:34
But doesn't Stmt prime need to be built up? Because StmtPrime = Semi | Expression
– Matt
Nov 9 at 21:38
so technically, its not optional?
– Matt
Nov 9 at 21:38
1
@Matt I'm not following you.stmtprime
is a type that you defined that could either contain an expression or nothing.expr option
is also a type that could either contain an expression or nothing. Therefore these two types are completely interchangeable and thestmtprime
type is unnecessary. Whether the semicolon is optional or not is a property of your parser, not your AST. The fact that you happened to name your no-expression constructorSEMI
does not affect whether or not the parser expects a semicolon.
– sepp2k
Nov 9 at 21:48
|
show 3 more comments
I had to write the parser by hand (no parser generator) which is why I created StmtPrime, etc AST types. How would I define option? Very new to OCaml
– Matt
Nov 9 at 21:30
@Matt I don't see why writing the parser by hand would affect what the AST looks like. "How would I define option?"option
is a type from the standard library - you don't need to define it. You can just writeReturn of stmt option
instead ofReturn of stmtprime
and that's it.
– sepp2k
Nov 9 at 21:34
But doesn't Stmt prime need to be built up? Because StmtPrime = Semi | Expression
– Matt
Nov 9 at 21:38
so technically, its not optional?
– Matt
Nov 9 at 21:38
1
@Matt I'm not following you.stmtprime
is a type that you defined that could either contain an expression or nothing.expr option
is also a type that could either contain an expression or nothing. Therefore these two types are completely interchangeable and thestmtprime
type is unnecessary. Whether the semicolon is optional or not is a property of your parser, not your AST. The fact that you happened to name your no-expression constructorSEMI
does not affect whether or not the parser expects a semicolon.
– sepp2k
Nov 9 at 21:48
I had to write the parser by hand (no parser generator) which is why I created StmtPrime, etc AST types. How would I define option? Very new to OCaml
– Matt
Nov 9 at 21:30
I had to write the parser by hand (no parser generator) which is why I created StmtPrime, etc AST types. How would I define option? Very new to OCaml
– Matt
Nov 9 at 21:30
@Matt I don't see why writing the parser by hand would affect what the AST looks like. "How would I define option?"
option
is a type from the standard library - you don't need to define it. You can just write Return of stmt option
instead of Return of stmtprime
and that's it.– sepp2k
Nov 9 at 21:34
@Matt I don't see why writing the parser by hand would affect what the AST looks like. "How would I define option?"
option
is a type from the standard library - you don't need to define it. You can just write Return of stmt option
instead of Return of stmtprime
and that's it.– sepp2k
Nov 9 at 21:34
But doesn't Stmt prime need to be built up? Because StmtPrime = Semi | Expression
– Matt
Nov 9 at 21:38
But doesn't Stmt prime need to be built up? Because StmtPrime = Semi | Expression
– Matt
Nov 9 at 21:38
so technically, its not optional?
– Matt
Nov 9 at 21:38
so technically, its not optional?
– Matt
Nov 9 at 21:38
1
1
@Matt I'm not following you.
stmtprime
is a type that you defined that could either contain an expression or nothing. expr option
is also a type that could either contain an expression or nothing. Therefore these two types are completely interchangeable and the stmtprime
type is unnecessary. Whether the semicolon is optional or not is a property of your parser, not your AST. The fact that you happened to name your no-expression constructor SEMI
does not affect whether or not the parser expects a semicolon.– sepp2k
Nov 9 at 21:48
@Matt I'm not following you.
stmtprime
is a type that you defined that could either contain an expression or nothing. expr option
is also a type that could either contain an expression or nothing. Therefore these two types are completely interchangeable and the stmtprime
type is unnecessary. Whether the semicolon is optional or not is a property of your parser, not your AST. The fact that you happened to name your no-expression constructor SEMI
does not affect whether or not the parser expects a semicolon.– sepp2k
Nov 9 at 21:48
|
show 3 more comments
Sign up or log in
StackExchange.ready(function ()
StackExchange.helpers.onClickDraftSave('#login-link');
);
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
StackExchange.ready(
function ()
StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fstackoverflow.com%2fquestions%2f53233330%2fwhen-to-use-and-operator-in-ocaml-ast%23new-answer', 'question_page');
);
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function ()
StackExchange.helpers.onClickDraftSave('#login-link');
);
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function ()
StackExchange.helpers.onClickDraftSave('#login-link');
);
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function ()
StackExchange.helpers.onClickDraftSave('#login-link');
);
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
nWR 04CMsvnmCM rRg91GpVPyKkn9dv F4YgMhJHCJsT6jPZ,crwZ8TRYwgSA2 1