| Size: 22883 Comment:  | Size: 49001 Comment:  | 
| Deletions are marked like this. | Additions are marked like this. | 
| Line 7: | Line 7: | 
| Line 9: | Line 10: | 
| DLT is a front-end for disjunctive datalog, extending the DLV system with Template predicates, Frame Logic and higher order predicate names. Template predicates can be seen as a way to define intensional predicates by means of a subprogram, where the subprogram is generic and reusable as many times is necessary. Frame Logic extends classical logic and it gives consent for frame-based or object–oriented syntax. Higher order predicates allow to treat predicate names as variables (although this semantics has quite nothing to deal with second order logics). The language implemented by the DLT system is named DLP^T^.[[BR]] | DLT is a front-end for disjunctive datalog, extending the DLV system with Template predicates, Frame Logic and higher order predicate names. Template predicates can be seen as a way to define intensional predicates by means of a subprogram, where the subprogram is generic and reusable as many times is necessary. Frame Logic extends classical logic and enables frame-based or object-oriented syntax. Higher order predicates allow to treat predicate names as variables (although this semantics has quite nothing to deal with second order logics). [[BR]] | 
| Line 12: | Line 13: | 
| ---- | |
| Line 15: | Line 16: | 
| You can invoke the DLT system directly from the command-line, after the prompt of the system, as well the DLV system. '''Note:''' All the DLV options must be passed using the option ''-solverparams={}''. In the curly brackets the user may pass the list of options for the solver separated by a space. If you do not specify any options or files, DLT will just print some informational output and an empty model ({ }): | We can invoke the DLT system directly from the command-line, after the prompt of the system, as well the DLV system. Note that DLT relies on an external solver ES, which takes an input generated by DLT. Models generated by ES are then parsed back to the DLT syntax. ES can be chosen from a variety of solvers (see the Compatibility Table below) If you do not specify any options or files, DLT will just print some informational output: | 
| Line 24: | Line 24: | 
| {{{ DLW. Disjunctive Logic Framework for the Web [Build DEV/May 3 2006 gcc 3.3.5 (Debian 1:3.3.5-13)] usage: ./dlt {FRONTEND} {OPTIONS} [-higherorder] [-solver=<path>] [-preparsing] [-postparsing] [-solverparams={options_solver}] [filename [filename [...]]] Specify -help for more detailed usage information. }}} | {{{ DLT. Disjunctive Logic Framework with Templates and more [V1.5.1 - Build DEV/Jul 22 2008 gcc 4.0.1 (Apple Inc. build 5465)] usage: ./dlt {FRONTEND} {OPTIONS} [--higherorder] [--solver=<path>] [--preparsing] [--postparsing] [--framespace] [--uri] [--filter=<predicate>] [--pfilter=<predicate>] [--silent] [--aritycheck] [--safetycheck] [--solverparams={options_solver}] [filename [filename [...]]] Specify -help for more detailed usage information.}}} | 
| Line 32: | Line 32: | 
| The first line shows, first of all, the name of the executable program (DLW), then it gives information about the version of the executable program, the date when the binary was built and the identifier of the compiler. The last three lines give a brief explanation on how to use dlt, the last line informing the user of the ''–help'' option for futher instructions.[[BR]] The ''–help'' or ''-h'' option[[BR]][[BR]] {{{ > }}}'''{{{./dlt -help}}}''' [[BR]][[BR]] or [[BR]][[BR]] {{{ > }}}'''{{{./dlt -h}}}''' | The first line shows, first of all, the name of the executable program (DLT), then it gives information about the version of the executable program, the date when the binary was built and the identifier of the compiler. The last three lines give a brief explanation on how to use dlt, the last line informing the user of the ''-help'' option for futher instructions.[[BR]] [[Anchor(HELP)]]The ''-help'' or ''-h'' option[[BR]][[BR]] {{{ > }}}'''{{{./dlt -help}}}''' {{{ }}} or {{{ }}} {{{ > }}}'''{{{./dlt -h}}}''' | 
| Line 43: | Line 39: | 
| {{{ DLW. Disjunctive Logic Framework for the Web [Build DEV/May 5 2006 gcc 3.3.5 (Debian 1:3.3.5-13)] usage: ./dlt {FRONTEND} {OPTIONS} [-higherorder] [-solver=<path>] [-preparsing] [-postparsing] [-solverparams={options_solver}] [filename [filename [...]]] DLW Specific options: -higherorder Enables higher order predicate names -solver=[path_name] The path of the solver executable file For example: -solver=/dlv/dl -preparsing Print out re-written code and exit -postparsing Maps higher-order atoms into first-order atom -solverparams={ options } The specific solver options For example: -solverparams={ -v -filter=ciao } }}} If you do not want to see this status line, use the -silent option, which suppresses various informational output and blank lines. {{{ $ }}}'''{{{dlt -silent}}}''' {{{ {} }}} From now on, all examples will include the {{{-silent}}} option. | {{{ DLT. Disjunctive Logic Framework with Templates and more [V1.5.1 - Build DEV/Jul 22 2008 gcc 4.0.1 (Apple Inc. build 5465)] usage: ./dlt {FRONTEND} {OPTIONS} [--higherorder] [--solver=<path>] [--preparsing] [--postparsing] [--framespace] [--uri] [--filter=<predicate>] [--pfilter=<predicate>] [--silent] [--aritycheck] [--safetycheck] [--solverparams={options_solver}] [filename [filename [...]]] DLT Specific options: --higherorder Enables higher order predicate names --solver=[path_name] The path of the solver executable file For example: --solver=/dlv/dl --preparsing Print out re-written code and exit --postparsing Maps higher-order atoms into first-order atom --framespace Activates displaying for framespace directives --uri Enables more verbose output for URI constants --filter=name1,..,nameN Output only instances of the specified predicates --pfilter=name1,..,nameN Output only positive of the instances specified predicates --solverparams={options} The specific solver options For example: -solverparams={-v -nofacts} --aritycheck Forces to check same arity for the same predicate name --safetycheck Forces to check safety of rule's head variables }}} [[Anchor(SIL)]] If you do not want to see this status line, use the ''-silent'' option, which suppresses various informational output and blank lines. {{{ > }}}'''{{{./dlt -silent}}}''' {{{ usage: ./dlt {FRONTEND} {OPTIONS} [--higherorder] [--solver=<path>] [--preparsing] [--postparsing] [--framespace] [--uri] [--filter=<predicate>] [--pfilter=<predicate>] [--silent] [--aritycheck] [--safetycheck] [--solverparams={options_solver}] [filename [filename [...]]] Specify -help for more detailed usage information. }}} ==== Section 1.1. DLT Specific options ==== The following table shows the specific options of DLT.[[BR]][[BR]] '''note:''' The ''–filter'' e ''–pfilter'' options are inherited by DLV, but they must to be considered as specific options of DLT. Indeed, DLT can change the predicate names before passing them to DLV ( DLT set them up again it in the post-parsing phase ). Thus, it’s necessary that DLT performs filtering operations directly, before printing the results. ||< : 15% #99CCCC> '''OPTION''' ||< : 35% #99CCCC> '''EXPLANATION''' || ||<: #ADD8E6> ''[#HIGH -higherorder]'' ||< #ADD8E6>Considers the names of the predicates as variables|| ||<: #ADD8E6> ''-solver=[path_name]'' ||< #ADD8E6> Allows to specify the path of the solver || ||<: #ADD8E6> ''[#PREP -preparsing]'' ||< #ADD8E6> Prints to standard output the program rewritten by DLT ans stops without invoking the solver || ||<: #ADD8E6> ''[#POST -postparsing]'' ||< #ADD8E6>Receives the output of the solver from standard input and maps data to the original DLT format (before rewriting) || ||<: #ADD8E6> ''-filter=pred''||< #ADD8E6>Turns on filtering. Predicates pred (or true negated -pred ) will be the only included in the output. This option may be used multiple times, and pred may also be a comma-separated list of predicate names.|| ||<: #ADD8E6> ''-pfilter=pred''||< #ADD8E6>Turns on filtering. Positive instances of predicate pred will be included in the output. This option may be used multiple times, and pred may also be a comma-separated list of predicate names.|| ||<: #ADD8E6> ''[#SOLV -solverparams={options}]''||< #ADD8E6> In the curly brackets the user may pass a space separated list of options to be passed to the solver.|| ||<: #ADD8E6> ''[#SIL -silent]''||< #ADD8E6>Do not output information about the version of the executable program and the compiler. || ||<: #ADD8E6> ''[#ARC -aritycheck]''||< #ADD8E6> Forces to check same arity for the same predicate name|| ||<: #ADD8E6> ''[#SAC -safetycheck]''||< #ADD8E6> Forces to check safety of rule's head variables|| ||<: #ADD8E6> ''[#HELP -help]'' or ''[#HELP -h]'' ||< #ADD8E6>Gives a detailed explanation on how to use DLT.|| ---- | 
| Line 73: | Line 100: | 
| DLT’s native language is ''DLV'' extended with Template predicates. We remember that DLV’s native language is ''Disjunctive Datalog'' extended with constraints, true negation, aggregates and queries. ==== Section 2.1. Template Definition ==== A template definition D consists of a template header and an associated DLP^T^ subprogram closed into parenthesis curly braces. '''Example 2-1.''' The following is a template definition: {{{ #template max[p(1)](1) { max(X) :- p(X), not exceeded(X). exceeded(X) :- p(X), p(Y), Y > X. } }}} | DLT's native language is [http://www.dbai.tuwien.ac.at/proj/dlv/ DLV] extended with Template predicates and other constructs. We remember that DLV's native language is ''Disjunctive Datalog'' extended with constraints, true negation, aggregates and queries. '''How to write a DLP^T^ program''' For a correct writing and usage of the DLP^T^ language here are a few general conditions that must be satisfied. ==== Section 2.1. Writing Template Definition ==== A template definition D consists of a template header and an associated DLP^T^ subprogram closed into curly braces. We consider a generic definition of template: ||< #FFFF99> '''#template''' ''template_name'' { p,,1,,( a,,1,, ) p,,n,, ( a,,n,, )}(arity)[[BR]] { [[BR]][[BR]] < Body of Template > [[BR]][[BR]] } || All the following conditions must be respected: * In the body of a template definition at least a rule must have an atom in its head with the same name of the template in which it is defined; * The formal parameters (p,,1,,(a,,1,,),..., p,,n,,(a,,n,,)) cannot appear in a head; * All the formal parameters should appear in some rules; * In the program cannot exist two or more template definitions having the same name; * The formal parameters cannot be preceded from the true negation symbol and the negation as failure simbol in the template header in which they are uses; * The global predicates defined using the GLOBAL directive cannot appear in the head of a rule defined in the body of the template; * Definitions of template cannot appear in the body of a template; '''Example 2-1. Correct Template Definitions''' ||< #FFFF99>'''c,,1,,''': ''' #template''' max{p(1)}(1)[[BR]] { [[BR]][[BR]] max(X) :- p(X), '''not''' exceeded(X). [[BR]] exceeded(X) :- p(X), p(Y), Y > X. [[BR]][[BR]] } || ||< #FFFF99>'''c,,2,,''': ''' #template'''max{p(1)}(1)'''GLOBAL''' node[[BR]] { [[BR]][[BR]] max(X) :- p(X), '''not''' exceeded(X).[[BR]] exceeded(X) :- p(X), p(Y), Y > X.[[BR]][[BR]] } || '''Example 2-2 Wrong Template Definitions''' ||< #FFFF99>'''t,,1,,''': ''' #template''' wrong_definition1{p(1)}(1)[[BR]] { [[BR]][[BR]] exceeded(X) :- p(X), p(Y), Y > X.[[BR]] p(a).[[BR]][[BR]] } || ||< #FFFF99>'''t,,2,,''': ''' #template''' wrong_definition2{-p(1), not c(1)}(1)[[BR]] { [[BR]][[BR]] < Body of Template Definition > [[BR]][[BR]] } || '''t,,1,,''' is a wrong template definition because the formal parameter ''p''cannot appear in the head of a rule. '''t,,2,,''' is a wrong template definition because the formal parameter ''p'' is preceded from the true negation symbol, while the formal parameter ''c'' is preceded from the negation as failure symbol. [[Anchor(ES23)]] '''Example 2-3.''' The following is a template definition: ||< #FFFF99> '''#template''' max{p(1)}(1)[[BR]] { [[BR]][[BR]]max(X) :- p(X), not exceeded(X).[[BR]]exceeded(X) :- p(X), p(Y), Y > X.[[BR]][[BR]] { || | 
| Line 92: | Line 157: | 
| '''{{{ #template }}}'''{{{max[p(1)](1) }}} | ||< #FFFF99>''' #template '''max{p(1)}(1) || | 
| Line 96: | Line 161: | 
| '''{{{ { }}}''' {{{ max(X) :- p(X), not exceeded(X).}}} {{{ exceeded(X) :- p(X), p(Y), Y > X.}}} '''{{{ } }}}''' | ||< #FFFF99> { [[BR]][[BR]] max(X) :- p(X), not exceeded(X).[[BR]] exceeded(X) :- p(X), p(Y), Y > X.[[BR]][[BR]] } || | 
| Line 106: | Line 165: | 
| ==== Section 2.2. Writing Template Atoms ==== We consider a generic definition of template: ||< #FFFF99>''template_name''{ p,,1,,(X,,1,,),..., p,,n,,(X,,n,,)}(Args)|| All the following conditions must be respected: * For every template atom, defined in some rule of the general program, there must exist a definition having the same name of ''template_name''; * The number of actual predicates (p,,1,,(X,,1,,),..., p,,n,,(X,,n,,)) must be equal to the number of formal predicates of the correspondent definition; * The arity of the output terms must be equal to the output arity of the correspondent definition; * The number of asterisks of every actual parameter must be equal to the arity of the correspondent formal parameter; * The actual parameters cannot appear negated in the atom template that uses them(note that true negation is allowed instead). '''Example 2-4 Correct Template Atoms''' ||< #FFFF99> '''c,,1,,''': template_name{-person($,*,*)}(X).|| ||< #FFFF99> '''c,,2,,''': template_name{person($,*,*)}(X).|| ||< #FFFF99> '''c,,3,,''': template_name{person(Name,*,*)}(X).|| ||< #FFFF99> '''c,,4,,''': template_name{-person(Name,*,*)}(X).|| '''Example 2-5 Wrong Template Atoms''' We consider the template definition of the Example 2-3(definition of template ''max''). The following are wrong template atoms: ||< #FFFF99> '''a,,1,,''': ''max''person(Name,*,*)}(X,Y,Z).|| ||< #FFFF99> '''a,,2,,''': ''max''{not person(Name,*,*)}(X).|| ||< #FFFF99> '''a,,3,,''': ''max''{person(Name,'''*''','''*'''), person('''*''','''*''',$)}(X).|| '''a,,1,,''' is a wrong template atom because the arity of the output terms is not equal to the output arity of the correspondent definition. '''a,,2,,''' is a wrong template atom because the actual parameter ''person'' is preceded by negation as failure. '''a,,3,,''' is a wrong template atom because the number of actual predicates is not equal to the number of formal predicates of the correspondent definition. | |
| Line 108: | Line 215: | 
| '''Example 2-2.''' Wrong Template Definition The following is not a valid template definition: '''{{{ #template }}}'''{{{wrong_definition[p(1)](1) }}} '''{{{ { }}}''' '''{{{ template }}}'''{{{max[h(1)](1) }}} {{{ < }}}'''{{{other rules}}}'''{{{ > }}} '''{{{ } }}}''' ==== Section 2.2. Template Atom ==== | ==== Section 2.3. Negative Rules with Template Atoms and True Negation ==== Template atoms can be used also as negated and negative atoms in the generic rules of the program. Then we can use a template atom after the ''negation as failure'' symbol ('''not''') as it happens for the usual atoms. '''Example 2-6.''' The following is an example of negated and negative template atom, respectively. ||< #FFFF99> ''' not '''max{student(Sex,$,*)}(Age) || Negation for template atoms is treated as well negation for DLV atom. For more information about ''negation as failure'' and true negation we send back at the DLV user manual available at the DLV homepage (http://www.dbai.tuwien.ac.at/proj/dlv/). ==== Section 2.4. Template Definition with Global Predicates ==== Template programming can become a very useful feature in applications where it is necessary to compact repetitive pieces of code. We consider a planning program for the Space Shuttle Reaction Control System. Here is a sketch from the program: ||<#FFFF99>'''template_name''' ready_to_fire_conditions{propulsor_type(2)}(2)[[BR]]''' GLOBAL '''time_slot, tank, is_pressurized_by, is_damaged [[BR]] { [[BR]][[BR]]ready_to_fire_conditions(J,T) :-propulsor_type(J,R),[[BR]]time_slot(T),tank(TK1,R),[[BR]]tank(TK2,R), TK1 !=TK2,[[BR]]is_pressurized_by(J,TK1,T),[[BR]]is_pressurized_by(J,TK2,T),[[BR]]not is_damaged(J).[[BR]][[BR]] } || the above template is employed to remove a set of repeated rules, which are changed this way ||< #FFFF99> fire_jet(J,T) :- ready_to_fire_conditions{jet(*,*)}(J,T). || ||< #FFFF99> fire_vernier(J,T) :- ready_to_fire_conditions{vernier(*,*)}(J,T). || ==== Section 2.5. Group-by ==== | 
| Line 126: | Line 244: | 
| '''Example 2-3.''' The followings are template atoms: {{{ max[weight(}}}'''{{{*}}}'''{{{)](M) }}} {{{ max[student($,Sex,}}}'''{{{*}}}'''{{{)](Age) }}} ''max'' is the template name; ''weight'' and ''student'' are predicate names, called ''actual predicates''; variables and constants are called ''standard'' terms (''Sex'' is a standard term); the dollar ‘$’ symbol is called ''projection term'' and the symbol ‘*’ is called ''parameter term''. ''M'' and ''Age'' are a list of usual terms (i.e. either variables or constants), and are called ''output lists''. We note that the template definitions may be unified with a template atom in many ways. The above example contains a plain invocation (max[weight(*)](M)), and a compound invocation (max[ student($,Sex,*) ](Age)). Intuitively, projection terms (‘$’ symbols) are intended in order to indicate attributes of an actual predicate which have to be ignored. A standard term (a constant or a variable) within an actual atom indicates a ‘group-by’ attribute, whereas parameter terms (‘*’ symbols) indicate attributes to be considered as parameter. For example, the template atom ''max[student($,Sex,*)](Age)'' allows to employ the definition of the template predicate ''max'' on a ternary predicate ''student'', discarding the first attribute of ''student'', and grouping by values of the second attribute. The intuitive meaning of this invocation is to define a predicate computing the student with maximum value of the ‘Age’ attribute (the third attribute of the ''student'' predicate), grouped by the ‘Sex’ attribute (the second one), ignoring the first attribute. The computed values of ''Age'' are returned through the output list. '''Example 2-4.''' We consider the following EDB: {{{ person(riccy,f,26). }}} {{{ person(kali,f,26). }}} {{{ person(peppe,m,26). }}} {{{ person(gibbi,m,29). }}} {{{ person(paddy,f,27). }}} it represents a set of persons of which it is known name, sex and age. We consider the template definition of ''max'' given in Example 2-1. The following: {{{ older_sex(Name,Sex,Age):- max[person($,Sex,*)](Age), }}} {{{ person(Name,Sex,Age). }}} | '''Example 2-7.''' The followings are template atoms: ||< #FFFF99> max{weight('''*''')}(M) || ||< #FFFF99> max{student($,Sex,'''*''')}(Age) || ''max'' is the template name; ''weight'' and ''student'' are predicate names, called ''actual predicates''; variables and constants are called ''standard'' terms (''Sex'' is a standard term); the dollar '''$''' symbol is called ''projection term'' and the symbol '''*''' is called ''parameter term''. ''M'' and ''Age'' are a list of usual terms (i.e. either variables or constants), and are called ''output lists''. We note that the template definitions may be unified with a template atom in many ways. The above example contains a plain invocation ''( max{weight(*)}(M) )'', and a compound invocation ''( max{ student($,Sex,*) }(Age) )''. Intuitively, projection terms ('''$''' symbols) are intended in order to indicate attributes of an actual predicate which have to be ignored. A standard term (a constant or a variable) within an actual atom indicates a '''group-by''' attribute, whereas parameter terms ('''*''' symbols) indicate attributes to be considered as parameter. For example, the template atom ''max{student($,Sex,*)}(Age)'' allows to employ the definition of the template predicate ''max'' on a ternary predicate ''student'', discarding the first attribute of ''student'', and grouping by values of the second attribute. The intuitive meaning of this invocation is to define a predicate computing the student with maximum value of the ''Age'' attribute (the third attribute of the ''student'' predicate), grouped by the ''Sex'' attribute (the second one), ignoring the first attribute. The computed values of ''Age'' are returned through the output list. '''Example 2-8.''' We consider the following EDB: ||< #FFFF99> person(riccy,f,26). [[BR]] person(kali,f,26). [[BR]] person(peppe,m,26). [[BR]] person(gibbi,m,29). [[BR]] person(paddy,f,27). || it represents a set of persons of which it is known name, sex and age. We consider the template definition of ''max'' given in [#ES23 Example 2-3]. The following: ||< #FFFF99> older_sex(Name,Sex,Age):- max{person($,Sex,*)}(Age),person(Name,Sex,Age). || | 
| Line 158: | Line 267: | 
| {{{ $ dlt -silent -nofacts programFileName }}} | {{{ > }}} '''{{{./dlt -silent -solverparams={-nofacts} programFileName }}}''' | 
| Line 162: | Line 271: | 
| {{{ {older_sex(paddy,f,27),older_sex(gibbi,m,29)} }}} | {{{ {older_sex(paddy,f,27),older_sex(gibbi,m,29)} }}} | 
| Line 166: | Line 276: | 
| ==== Section 2.3. Negative Rules with Template Atoms and True Negation ==== Template atoms can be used also as negated and negative atoms in the generic rules of the program. Then we can use a template atom after the “negation as failure” symbol ('''not''') as it happens for the usual atoms. '''Example 2-5.''' The following is an example of negated and negative template atom, respectively. '''{{{ not }}}'''{{{max[student(Sex,$,*)](Age) }}} Negation for template atoms is treated as well negation for DLV atom. For more information about “negation as failure” and true negation we send back at the DLV user manual available at the DLV homepage (http://www.dbai.tuwien.ac.at/proj/dlv/). ==== Section 2.4. Template Definition with Global Predicates ==== Template programming can become a very useful feature in applications where it is necessary to compact repetitive pieces of code. We consider a planning program for the Space Shuttle Reaction Control System. Here is a sketch from the program: '''{{{ #template }}}'''{{{ready_to_fire_conditions[propulsor_type(2)](2) }}} '''{{{ GLOBAL }}}'''{{{time_slot, tank, is_pressurized_by, is_damaged }}} {{{ { }}} {{{ ready_to_fire_conditions(J,T) :- propulsor_type(J,R), }}}} {{{ time_slot(T),tank(TK1,R), }}} {{{ tank(TK2,R), TK1 != TK2, }}} {{{ is_pressurized_by(J,TK1,T), }}} {{{ is_pressurized_by(J,TK2,T), }}} {{{ not is_damaged(J). }}} {{{ } }}} the above template is employed to remove a set of repeated rules, which are changed this way {{{ fire_jet(J,T) :- ready_to_fire_conditions[jet(*,*)](J,T). }}} {{{ fire_vernier(J,T) :- ready_to_fire_conditions[vernier(*,*)](J,T). }}} | '''Example 2-9.''' We consider another example and the following template: ||<#FFFF99> #template sum{emp(2)}(1)[[BR]]{[[BR]][[BR]]precedes(X,Y) :- emp(X,_), emp(Y,_), X<Y.[[BR]]succ(X,Y) :- precedes(X,Y), not elementInMiddle(X,Y).[[BR]]elementInMiddle(X,Y) :- precedes(X,Z), precedes(Z,Y).[[BR]]first(X) :- emp(X,_), not hasPredecessor(X).[[BR]]last(X) :- emp(X,_), not hasSuccessor(X).[[BR]]hasPredecessor(X) :- succ(Y,X).[[BR]]hasSuccessor(Y) :- succ(Y,X).[[BR]]partialSum(X,Sx) :- first(X), emp(X,Sx).[[BR]]partialSum(Y,S) :- succ(X,Y), partialSum(X,PSx), emp(Y,Sy),S=PSx+Sy.[[BR]]sum(S) :- last(L), partialSum(L,S).[[BR]][[BR]]}|| It allows to sum all the values of the second attribute of the argument ''emp'' in an iterative manner. The first attribute must be an ID which identifies unambiguously the instance. The template first orders the instances of ''emp'' by their ID (first rule), then using the next rules it defines the successor of each instance. It introduces a new atom, ''succ(I,II)'', thus finding the first and the last of the list (''first(X)'',''first(X)''). The predicate ''partialSum'' scans the entire ordered list and addes the sum of the preceding values to the following value. The sum predicate receives the value associated with the last instance of ''emp''. This is possible due to the predicate ''last''. Now we consider the following EDB: ||< #FFFF99>employee(1,gisella,direction,80).[[BR]]employee(2,carlo,production,36).[[BR]]employee(3,giuseppe,administration,40).[[BR]]employee(4,franco,distribution,45).[[BR]]employee(5,carla,administration,45).[[BR]]employee(6,lorenzo,direction,73).[[BR]]employee(7,paola,production,46).[[BR]]employee(8,marco,administration,40).[[BR]]employee(9,gianni,production,32).|| it represents a set of persons of which it is known ID, name and salary. We use the template sum and we consider the following: result(DIP,S):- sum{impiegato(*,$,DIP,*)}(S). this is a template atom which calculates the sum of the values of the 4th attribute of employee (salary) also taking into consideration the first (ID), ommiting the second(name) and grouping the result by the third attribute(department). Launching dlt from command line, as it follows: {{{>}}} '''{{{./ dlt name_file -silent -filter=result }}}''' we obtain the model {{{{result(direction,153), result(production,114), result(administration,125), result(distribution,45)} }}} The result is obtained in a simple manner. First, the template is executed as if the ''DIP'' parameter, which is later used to group the results, is only used for the selection. Therefore the instances of employee are ordered by their ID. Then the ''DIP'' and salary of each are selected. A table with a row for each employee and two columns, the first for departiment and the second for salary is obtained. The table is then analysed dividing the rows in groups defined by their ''DIP'' values. After this, the sum is separately calculated for each group. The output (filtering only the ''result'' predicate) contains the sum of the salaries for each departiment. ---- | 
| Line 216: | Line 318: | 
| '''u,,1,,:'''{{{  head(Y) :- }}}'''{{{not}}}'''{{{ template_name[p(M,$,*)](const).}}} {{{ }}} '''u,,2,,:'''{{{ :- }}}'''{{{not}}}'''{{{ template_name[p($,M,*)](X).}}} '''u,,1,,''' is unsafe because the variable ‘Y’ that appear in the head of the rule does not appear in at least a positive atom of the body of the same rule. '''u,,2,,''' is unsafe because the variable ‘X’ in the output list of the given invocation does not appear in at least a positive atom of the body of the same rule. | ||< #FFFF99>'''u,,1,,:'''  head(Y) :- '''not''' template_name{p(M,$,*)}(const).|| ||< #FFFF99>'''u,,2,,:''' :- '''not''' template_name{p($,M,*)}(X).|| '''u,,1,,''' is unsafe because the variable ''Y'' that appear in the head of the rule does not appear in at least a positive atom of the body of the same rule. '''u,,2,,''' is unsafe because the variable ''X'' in the output list of the given invocation does not appear in at least a positive atom of the body of the same rule. | 
| Line 228: | Line 328: | 
| '''s,,1,,:'''{{{  head(Y) :- }}}'''{{{not}}}'''{{{ template_name[p(M,$,*)](const),}}} {{{ predicate(_,_,Y).}}} {{{ }}} '''s,,2,,:'''{{{ :- }}}'''{{{not}}}'''{{{ template_name[p($,M,*)](X), predicate(_,_,X).}}} {{{ }}} '''s,,3,,:'''{{{ head(X) :- }}}'''{{{not}}}'''{{{ template_name[p(M,$,*)](Y),}}} {{{ predicate(_,X,Y).}}} {{{ }}} '''s,,4,,:'''{{{ :- template_name[p(M,$,*)](X).}}} {{{ }}} '''s,,5,,:'''{{{ :- }}}'''{{{not}}}'''{{{ template_name[p(M,$,*)](const).}}} == Chapter 4. Using DLP^T Language == '''How to write a DLP^T^ program?''' For a correct writing and usage of the DLP^T^ language here are a few general conditions that must be satisfied. ==== Section 4.1. Writing Template Definition ==== We consider a generic definition of template: '''{{{#template}}}''' ''{{{template_name}}}''{{{[}}}p,,1,,{{{(}}}a,,1,,{{{),..., }}}p,,n,,{{{(}}}a,,n,,{{{)](arity)}}} '''{{{ { }}} {{{ < Body of Template > }}} {{{ } }}}''' All the following conditions must be respected: 1. In the body of a template definition at least a rule must have an atom in its head with the same name of the template in which it is defined; 2. The formal parameters (p,,1,,(a,,1,,),..., p,,n,,(a,,n,,)) cannot appear in a head; 3. All the formal parameters should appear in some rules; 4. In the program cannot exist two or more template definitions having the same name; 5. The formal parameters cannot be preceded from the true negation symbol and the negation as failure simbol in the template header in which they are uses; 6. The global predicates defined using the GLOBAL directive cannot appear in the head of a rule defined in the body of the template; 7. Definitions of template cannot appear in the body of a template; '''Example 4.1. Wrong Template Definitions''' '''t,,1,,''': '''{{{ #template}}}'''{{{ wrong_definition1[p(1)](1)}}} '''{{{ { }}}''' {{{ exceeded(X) :- p(X), p(Y), Y > X.}}} {{{ p(a).}}} '''{{{ } }}}''' '''t,,2,,''': '''{{{ #template}}}'''{{{ wrong_definition2[-p(1), not c(1)](1)}}} '''{{{ { }}}''' {{{ < Body of Template Definition > }}} '''{{{ } }}}''' '''t,,1,,''' is a wrong template definition because the formal parameter ‘p’ cannot appear in the head of a rule. '''t,,2,,''' is a wrong template definition because the formal parameter ‘p’ is preceded from the true negation symbol, while the formal parameter ‘c’ is preceded from the negation as failure symbol. '''Example 4.2 Correct Template Definitions''' '''c,,1,,''': '''{{{ #template}}}'''{{{ max[p(1)](1)}}} '''{{{ { }}}''' {{{ max(X) :- p(X), }}}'''{{{not}}}'''{{{ exceeded(X). }}} {{{ exceeded(X) :- p(X), p(Y), Y > X. }}} '''{{{ } }}}''' '''c,,2,,''': '''{{{ #template}}}'''{{{ max[p(1)](1) }}}'''{{{GLOBAL}}}'''{{{ node}}} '''{{{ { }}}''' {{{ max(X) :- p(X), }}}'''{{{not}}}'''{{{ exceeded(X).}}} {{{ exceeded(X) :- p(X), p(Y), Y > X.}}} '''{{{ } }}}''' ==== Section 4.2. Writing Template Atoms ==== We consider a generic definition of template: ''{{{template_name}}}''{{{[}}}p,,1,,{{{(}}}X,,1,,{{{),..., }}}p,,n,,(X,,n,,{{{)](Args)}}} All the following conditions must be respected: 1. For every template atom, defined in some rule of the general program, there must exist a definition having the same name of ''template_name''; 2. The number of actual predicates (p,,1,,(X,,1,,),..., p,,n,,(X,,n,,)) must be equal to the number of formal predicates of the correspondent definition; 3. The arity of the output terms must be equal to the output arity of the correspondent definition; 4. The number of asterisks of every actual parameter must be equal to the arity of the correspondent formal parameter; 5. The actual parameters cannot appear negated in the atom template that uses them(note that true negation is allowed instead). '''Example 4.3 Wrong Template Atoms''' We consider the template definition of the Example 2-1(definition of template ‘max’). The following are wrong template atoms: '''a,,1,,''': ''{{{ max}}}''{{{[person(Name,*,*)](X,Y,Z).}}} {{{ }}} '''a,,2,,''': ''{{{ max}}}''{{{[not person(Name,*,*)](X).}}} {{{ }}} '''a,,3,,''': ''{{{ max}}}''{{{[person(Name,}}}'''*''','''*'''{{{), person(}}}'''*''','''*'''{{{,$)](X).}}} '''a,,1,,''' is a wrong template atom because the arity of the output terms is not equal to the output arity of the correspondent definition. '''a,,2,,''' is a wrong template atom because the actual parameter ‘person’ is preceded by negation as failure. '''a,,3,,''' is a wrong template atom because the number of actual predicates is not equal to the number of formal predicates of the correspondent definition. '''Example 4.4 Correct Template Atoms''' c,,1,,: {{{ template_name[-person($,*,*)](X).}}} {{{ }}} c,,2,,: {{{ template_name[person($,*,*)](X).}}} {{{ }}} c,,3,,: {{{ template_name[person(Name,*,*)](X).}}} {{{ }}} c,,4,,: {{{ template_name[-person(Name,*,*)](X).}}} == Chapter 5. System Architecture == The DLP^T^ language has been implemented on top of the DLV system. The overall architecture of the system is shown in Figure 1. A DLP^T^ program is sent to a DLP^T^ pre-parser, which performs syntactic checks (included nonrecursivity checks), and builds an internal representation of the DLP^T^ program. The DLP^T^ Inflater produces an equivalent DLV program P’; P’ is piped towards the DLV system. The models M(P’) of P’, computed by DLV, are then converted in a readable format through the Post-parser module; the Post-parser filters out from M(P’) informations about internally generated predicates and rules. == Chapter 6. Knowledge Representation by DLP^T == | ||< #FFFF99>'''s,,1,,:'''  head(Y) :- '''not''' template_name{p(M,$,*)(const), predicate(_,_,Y).|| ||< #FFFF99>'''s,,2,,:''' :- '''not''' template_name{p($,M,*)}(X), predicate(_,_,X).|| ||< #FFFF99>'''s,,3,,:''' head(X) :- '''not''' template_name{p(M,$,*)}(Y),predicate(_,X,Y).|| ||< #FFFF99>'''s,,4,,:''' :- template_name{p(M,$,*)}(X).|| ||< #FFFF99>'''s,,5,,:''' :- '''not''' template_name{p(M,$,*)}(const).|| ---- == Chapter 4. Knowledge Representation by DLP^T == | 
| Line 402: | Line 347: | 
| ==== Section 6.1. Aggregates ==== | ==== Section 4.1. Aggregates ==== | 
| Line 407: | Line 352: | 
| '''{{{ #template}}}'''{{{ count[p(1),succ(2)](1)}}} {{{ { }}} {{{ partialCount(0,0). }}} {{{ partialCount(I,V) :- not p(Y), I=Y+1,partialCount(Y,V). }}} {{{ partialCount(I,V2) :- p(Y), I=Y+1,partialCount(Y,V), }}} {{{ succ(V,V2). }}} {{{ partialCount(I,V2) :- p(Y),I=Y+1,partialCount(Y,V), }}} {{{ max[succ(*,$)](V2). }}} {{{ count(M) :- max[partialCount($,*)](M). }}} {{{ } }}} The above template definition is conceived in order to count, in a iterative-like way, values of the '''p''' predicate through the ''partialCount'' predicate. A ground atom '''partialCount(i,a)''' means that at the stage ''i'', the constant '''a''' has been counted up. The predicate ''count'' takes the value which has been counted at the highest (i.e. the last) stage value. It is worth noting how ''max'' is employed over the binary predicate partialCount, instead of an unary one. Indeed, the ‘$’ and ‘*’ symbols are employed to project out the first argument of partialCount. The last rule is equivalent to the piece of code: {{{ partialCount'(X) :- partialCount(_,X). }}} {{{ count(M) :- max[partialCount'(*)](M). }}} | ||< #FFFF99>''' #template''' count{p(1),succ(2)}(1)[[BR]]    { [[BR]][[BR]]    partialCount(0,0). [[BR]]    partialCount(I,V)  :- not p(Y), I=Y+1,partialCount(Y,V). [[BR]]    partialCount(I,V2) :- p(Y), I=Y+1,partialCount(Y,V), [[BR]]                          succ(V,V2).  [[BR]]    partialCount(I,V2) :- p(Y),I=Y+1,partialCount(Y,V), [[BR]]                          max[succ(*,$)](V2). [[BR]]    count(M) :- max{partialCount($,*)}(M). [[BR]][[BR]]    } || The above template definition is conceived in order to count, in a iterative-like way, values of the '''p''' predicate through the ''partialCount'' predicate. A ground atom '''partialCount(i,a)''' means that at the stage ''i'', the constant '''a''' has been counted up. The predicate ''count'' takes the value which has been counted at the highest (i.e. the last) stage value. It is worth noting how ''max'' is employed over the binary predicate partialCount, instead of an unary one. Indeed, the '''$''' and '''*''' symbols are employed to project out the first argument of partialCount. The last rule is equivalent to the piece of code: ||< #FFFF99> partialCount'(X) :- partialCount(_,X). || ||< #FFFF99> count(M) :- max{partialCount'(*)}(M). || | 
| Line 435: | Line 362: | 
| '''{{{ #template}}}'''{{{ subset[p(1)](1) }}} {{{ { }}} {{{ subset(X) v -subset(X) :- p(X). }}} {{{ } }}} '''{{{ #template}}}'''{{{ permutation[p(1)](2). }}} {{{ { }}} {{{ permutation(X,N)v npermutation(X,N) :- p(X),#int(N), N <= N1, }}} {{{ count[p(*),>(*,*)](N1). }}} {{{ :- permutation(X,A),permutation(Z,A), Z <> X. }}} {{{ :- permutation(X,A),permutation(X,B), A <> B. }}} {{{ covered(X) :- permutation(X,A). }}} {{{ :- p(X), not covered(X). }}} {{{ } }}} | ||< #FFFF99>''' #template''' subset{p(1)}(1) [[BR]]  { [[BR]][[BR]]     subset(X) v -subset(X) :- p(X). [[BR]][[BR]]     } || ||< #FFFF99>''' #template''' permutation{p(1)}(2). [[BR]] { [[BR]][[BR]] permutation(X,N)v npermutation(X,N) :- p(X),#int(N), N <= N1, [[BR]] count{p(*),>(*,*)}(N1). [[BR]] :- permutation(X,A),permutation(Z,A), Z <> X. [[BR]] :- permutation(X,A),permutation(X,B), A <> B. [[BR]] covered(X) :- permutation(X,A). [[BR]] :- p(X), not covered(X). [[BR]][[BR]] } || | 
| Line 465: | Line 370: | 
| {{{ in_clique(X) :- subset[node(*)](X). }}} {{{ :- count[in_clique(*),>(*,*)](K),K < 5. }}} {{{ :- in_clique(X),in_clique(Y), X <> Y, not edge(X,Y). }}} | ||< #FFFF99> in_clique(X) :- subset{node(*)}(X). [[BR]] :- count{in_clique(*),>(*,*)}(K),K < 5. [[BR]] :- in_clique(X),in_clique(Y), X <> Y, not edge(X,Y). || | 
| Line 473: | Line 374: | 
| {{{ path(X,N) :- permutation[node(*)](X,N). }}} {{{ :- path(X,M), path(Y,N), not edge(X,Y), M = N+1. }}} | ||< #FFFF99> path(X,N) :- permutation{node(*)}(X,N). [[BR]] :- path(X,M), path(Y,N), not edge(X,Y), M = N+1. || | 
| Line 479: | Line 378: | 
| '''{{{ #template}}}'''{{{ intersection[a(1),b(1)](1). }}} {{{ { }}} {{{ intersection (X) :- a(X),b(X). }}} {{{ } }}} '''{{{ #template}}}'''{{{ union[a(1),b(1)](1). }}} {{{ { }}} {{{ union(X) :- a(X). }}} {{{ union(X) :- b(X). }}} {{{ } }}} '''{{{ #template}}}'''{{{ symmetricdifference[a(1),b(1)](1) }}} {{{ { }}} {{{ symmetricdifference(X) :- union[a(*),b(*)](X), }}} '''{{{ not}}}'''{{{ intersection[a(*),b(*)](X). }}} {{{ } }}} | ||< #FFFF99>''' #template''' intersection{a(1),b(1)}(1). [[BR]]    { [[BR]][[BR]]     intersection (X) :- a(X),b(X). [[BR]][[BR]]    } || ||< #FFFF99>''' #template''' union{a(1),b(1)}(1). [[BR]] { [[BR]][[BR]] union(X) :- a(X). [[BR]] union(X) :- b(X). [[BR]][[BR]] } || ||< #FFFF99>''' #template''' symmetricdifference{a(1),b(1)}(1) [[BR]] { [[BR]][[BR]] symmetricdifference(X) :- union{a(*),b(*)}(X), [[BR]]''' not''' intersection{a(*),b(*)}(X). [[BR]][[BR]] } || | 
| Line 508: | Line 386: | 
| '''{{{ #template}}}''' before[date1(3),date2(3)](6) }}} {{{ { }}} {{{ before(D,M,Y,D1,M1,Y1) :- date1(D,M,Y), }}} {{{ date2(D1,M1,Y1), Y < Y1. }}} {{{ before(D,M,Y,D1,M1,Y1) :- date1(D,M,Y), }}} {{{ date2(D1,M1,Y1), Y == Y1, M < M1. }}} {{{ before(D,M,Y,D1,M1,Y1) :- date1(D,M,Y), date2(D,M1,Y1), }}} {{{ Y == Y1, M = M1, D < D1. }}} {{{ } }}} | ||< #FFFF99>''' #template''' before{date1(3),date2(3)}(6) [[BR]]     { [[BR]][[BR]]     before(D,M,Y,D1,M1,Y1) :- date1(D,M,Y),  [[BR]]                               date2(D1,M1,Y1), Y < Y1. [[BR]]     before(D,M,Y,D1,M1,Y1) :- date1(D,M,Y),  [[BR]]                               date2(D1,M1,Y1), Y == Y1, M < M1. [[BR]]     before(D,M,Y,D1,M1,Y1) :- date1(D,M,Y), date2(D,M1,Y1),  [[BR]]                               Y == Y1, M = M1, D < D1. [[BR]][[BR]]     } || ---- [[Anchor(HIGH)]] == Chapter 5. Higher order predicates == With the use of higher order predicates it is possibile to consider the names of the predicates as variables. The user inserts atoms as if they where of first order, but if the option ''-higherorder'' is used it is possible to execute higher order queries. Moreover they are fundamental for the [#HIG1 Frame Logic] and for the use of predicates that may appear with [#HIG2 different arities]. '''Example 5-1.''' We consider the following EDB: ||<#FFFF99>sister(gisella,maria).[[BR]]mother(maria,giovanni).[[BR]]husband(gino,carmela).[[BR]]friend(gisella,rosina).[[BR]]aunt(gisella,sara).|| it represents a set of persons; the names of the predicates define the relations between the persons. The use higher order allows the user to use the names of the predicates as though they where attributes of other predicates. The following rule simply rewrites the facts, but also allows the user to understand how to perform higher order queries. ||<#FFFF99>relationship(X,R,Y):- R(X,Y).|| Launching dlt from command line with option ''-higherorder'', as it follows: {{{ > }}} '''{{{./dlt name_file -higherorder -solverparams={-nofact}}}}''' we obtain the model {{{ {relationship(gisella,sister,maria), relationship(gisella,friend,rosina), relationship(gisella,aunt,sara), relationship(maria,mother,giovanni), relationship(gino,husband,carmela)} }}} Please note that now the predicate names appear also as attributes of the relations (see for instance '' relationship(gisella,sister,maria)''). ---- == Chapter 6. Frame Logic == F-logic is really useful, since it allows to model reality in a very simple way, exploiting its object-oriented programming features. Moreover, using frames helps to keep the code compact and enhances readability with respect to having a big set of single rules. Indeed, in principle, everything which can be expressed with frames can be also done using predicates and standard atoms; however frames make programs simpler and more intuitive. Like in OO programming languages, classes and objects are both supported. When declaring an object, it can in fact belong to a class. A frame object has usually a list of attributes, which can eventually be empty. It is possible to create a Frame Molecule and place it either in the head or in the body of a rule. A simple frame molecule is: ||<#FFFF99>nameframe[nameattribute->valueattribute].|| The above contains a single attribute expression {{{nameattribute->valueattribute}}}. '''Example 6-1.''' ||<#FFFF99>brown:employee[ surname -> "Mr. Brown",skill ->> {java, asp}, salary -> 800 , married].|| Here we see interesting features. The object "brown" has been defined as belonging to the class "employee" by means of the operator ":". Then a list of attributes is associated to the object. Here we see two types of attributes, single-valued (characterized by the "->" arrow), and multi-valued (characterized by the "->>" arrow). Multi-valued attributes are very useful, as they permit the use of sets of values. In this case they are well suited to represent the set of skills of an employee. Please note that in case of multi-valued attributes, the curly brackets are used. They are usually omitted when there is only one attribute in the set. Attributes can also be boolean-valued. The last clause is a boolean attribute which indicates that ''married'' is true for the object ''brown''. With frames, as we have seen so far, it is possible to describe objects with ease, but they are very useful also to describe classes and taxonomies of classes, with many expressive constructs. To describe a taxonomy, we basically can make use of the binary operator "::", which means that the first operand is subclass of the second operand. '''Example 6-2.''' ||<#FFFF99>webDesigner::javaProgrammer. javaProgrammer::Programmer.|| In this example we can see that webDesigner is defined as a subclass of javaProgrammer, which in turn is defined as a subclass of Programmer. The subclassOf operator is transitive, and so webDesigner is a subclass of Programmer too. When describing classes it is possible, like in java or c++, to list a certain number of attributes, specifying either their value or their type. To do this, a wide number of "arrow" operators is available to the user, such as {{{->, ->>, *->, *->>, =>, =>>}}}. These operator on their own do not have a strong semantic meaning. Actually, they take the classical F-logic meaning when coupled with specific axioms, which we will discuss later. '''Example 6-3.''' ||<#FFFF99>programmer[salary => integer].|| The "=>" and "=>>" operators are used to describe the type of an attribute. Both support set of types for each attribute. If you want to use this features, keep in mind that the different values must be comma-separated and enclosed in curly brackets. '''Example 6-4.''' ||<#FFFF99>employee[skill =>>{string, int}].|| The main difference between the two operators is that they "map" on different operator applied to instances. In particular, the operator "=>" maps on the instance operator "->", while "=>>" maps on "->>". Except this, they are pretty similar. In a similar way it is possible to define the values associated to an attribute. Being in the context of classes, these values can be thought about as "default values" for the class. To associate values to a class attribute, different operators are available. As for the type operators, they have a particular meaning only when coupled to a F-logic semantics. The available operators are: {{{->,->>,*->,*->>}}}. By using the operators with "*", usually the will is to specify that the value assigned to the attribute is inherited automatically by the subclasses. This is not forced by default, actually. In general, "->" and "->>" operators work on single values, while the others work on sets. '''Example 6-4.''' ||<#FFFF99>employee[skill ->> {java, asp}, salary *->1000].|| While some attributes may be described as facts, other may be defined using inference rules. For example, suppose that ''john'' is married to ''mary'' and ''mary'' has three children. It can be assumed that ''john'' is the father of ''alice'', ''nancy'', and ''jack'' (the children of mary). Therefore the following fact: ||<#FFFF99>john[ children->>{ alice,nancy,jack } ].|| may be automatically derived if we add the following rule: ||<#FFFF99>X[ children -> C ] :- Y[ spouse->X,children->C ].|| In addition it is possible to specify the class an object belongs to using the following rules: ||<#FFFF99>john: man.[[BR]]mary: woman.|| Once the class of the object is identified, a specific list of attributes can be assigned at the same time: ||<#FFFF99>mary: woman[ children->>{ alice,nancy,jack } ].|| [[Anchor(ES63)]] '''Example 6-5.''' We consider an other definition frame: ||<#FFFF99>carlo:man[ married->camilla: woman[ children->>{ tom: man[ engaged->sara ],[[BR]]{{{ }}}laura: woman[ engaged->harryT ] },[[BR]]{{{ }}}father->bruce ],[[BR]]{{{ }}}children->>{ william: man[ mother->diana ],harry: man[ mother->diana ] },[[BR]]{{{ }}}mother->elisabeth: woman,[[BR]]{{{ }}}father->philip: man ].[[BR]]|| The frame describes the relationships existing between some member of the Britannic royal family. ''carlo'' is a ''man'' and he’s married with ''camilla''. He has two ''children'': ''william'' and ''harry'', and their ''mother'' was ''diana''. ''carlo'' ‘s mother is ''elisabeth'' and his father is ''philip''. ''camilla'' is a ''woman'' and she has a daughter, ''laura'', and a son, ''tom''. ''laura'' is betroth with ''harryT'' , and ''tom'' ’s girlfriend is ''sara''. Launching dlt from command line: {{{ > }}} '''{{{./dlt name_file}}}''' we obtain the model {{{ {carlo:man, camilla:woman, tom:man, laura:woman, william:man, harry:man, elisabeth:woman, philip:man, man::man, woman::woman, camilla[children->>{tom,laura},father->bruce],carlo[married->camilla,children->>{william,harry},father->philip,mother->elisabeth],harry[mother->diana],laura[engaged->harryT],tom[engaged->sara],william[mother->diana]} }}} The higher order atoms are fundamental for the frames. Indeed, the output consists of predicates whose names were attributes in the frames. For example look at the first part of the definition of the frame: ||< #FFFF99> carlo: man[ married->camilla ]. || ''carlo'' is a man, and he is married to ''camilla''; 'married' is an attribute of the object ''carlo'', but in the output it's presented as the name of a predicate: ||< #FFFF99> married(carlo, camilla). || '''Example 6-6.''' Let's show another example: ||<#FFFF99>gianni:student[[[BR]]pc->mistero:notebook os->debian:os[kind->gnu,[[BR]]kernel->linux]]].|| [[Anchor(HIG2)]] The frame says that Gianni is a student and has a pc named mistery. The pc is a notebook which uses debian as operating system. In addition information on the OS is given: kind and kernel. Launching this program without the option higherorder results with an error: {{{ > }}} '''{{{./dlt nome_file}}}''' {{{ os, first used with arity 2, now seen with arity 1. }}} The cause of the error is clear once we look at the opuput with the option -higherorder {{{ > }}} '''{{{./dlt -higherorder nome_file}}}''' {{{ {pc(gianni,mistero), os(mistero,debian), kind(debian,gnu), kernel(debian,linux), os(debian), notebook(mistero), student(gianni)} }}} The atoms os(mistero,debian) e os(debian) appears in the output. The same predicate is present with different arity. In the case of first order atoms this is not permitted. It is only possible using higher order atoms. ==== Section 6.1. Semantic Modeling ==== One very interesting feature is that it is possible to specify, in addiction to the file containing the frame program, other files containing the specification of a semantic behavior. Various pre-written modules are available to the user. The modules have been designed in order to give a semantic meaning to the various operators described previously. We esemplify the meaning of these modules with the following table: ovverridden(D,M,C) :- E[M -> V1], C::E, E::D, C!=E, E!=D. inheritable(C,M,D) :- C::D, D[M *-> V], not ovverridden(D,M,C). C[M *-> V] v -C[M *-> V] :- inheritable(C,M,D), D[M *->V]. exists(C,M) :- C[M *-> V]. :-inheritable(C,M,D), not exists(C,M). A[M -> V] :- A:C, C[M -> V]. ||< : 15% #99CCCC> '''OPERATOR''' ||< : 15% #99CCCC> '''AXIOM''' ||< : 15% #99CCCC> '''AXIOM DESCRIPTION'''||< : 15% #99CCCC> '''AXIOMATIC MODULE'''|| ||< : #ADD8E6> ''::'' ||< : #ADD8E6> ''A::B :- A::C, C::B'' ||< : #ADD8E6> '''' ||< : #ADD8E6> ''C'' || ||< : #ADD8E6> ''::'' ||< : #ADD8E6> ''A::A:-X:A'' ||< : #ADD8E6> '''' ||< : #ADD8E6> ''C'' || ||< : #ADD8E6> ''::'' ||< : #ADD8E6> '':- A::C, C::A, A!=C.'' ||< : #ADD8E6> '''' ||< : #ADD8E6> ''C'' || ||< : #ADD8E6> '':'' ||< : #ADD8E6> ''X:C :- X:D, D::C.'' ||< : #ADD8E6> '''' ||< : #ADD8E6> ''C'' || ||< : #ADD8E6> ''->'' ||< : #ADD8E6> ''- A[M -> V], A[M -> W], V!=W.'' ||< : #ADD8E6> '''' ||< : #ADD8E6> ''F'' || ||< : #ADD8E6> ''*->'' ||< : #ADD8E6> '':- A[M *-> V], A[M *-> W], V!=W.'' ||< : #ADD8E6> '''' ||< : #ADD8E6> ''F'' || ||< : #ADD8E6> ''=>'' ||< : #ADD8E6> ''D[A => T] :- D::C, C[A => T].'' ||< : #ADD8E6> '''' ||< : #ADD8E6> ''S'' || ||< : #ADD8E6> ''=>>'' ||< : #ADD8E6> ''D[A =>> T] :- D::C, C[A =>> T].'' ||< : #ADD8E6> '''' ||< : #ADD8E6> ''S'' || ||< : #ADD8E6> '''' ||< : #ADD8E6> '''' ||< : #ADD8E6> '''' ||< : #ADD8E6> '''' || ||< : #ADD8E6> '''' ||< : #ADD8E6> '''' ||< : #ADD8E6> '''' ||< : #ADD8E6> '''' || ||< : #ADD8E6> '''' ||< : #ADD8E6> '''' ||< : #ADD8E6> '''' ||< : #ADD8E6> '''' || ||< : #ADD8E6> '''' ||< : #ADD8E6> '''' ||< : #ADD8E6> '''' ||< : #ADD8E6> '''' || ||< : #ADD8E6> '''' ||< : #ADD8E6> '''' ||< : #ADD8E6> '''' ||< : #ADD8E6> '''' || ||< : #ADD8E6> '''' ||< : #ADD8E6> '''' ||< : #ADD8E6> '''' ||< : #ADD8E6> '''' || ||< : #ADD8E6> '''' ||< : #ADD8E6> '''' ||< : #ADD8E6> '''' ||< : #ADD8E6> '''' || ||< : #ADD8E6> '''' ||< : #ADD8E6> '''' ||< : #ADD8E6> '''' ||< : #ADD8E6> '''' || {{{ > }}} '''{{{./dlt -higherorder constrained_semantics filename}}}''' where constrained_semantics is the name of the file containing the stable models semantics, while filename is the file containing a dlt program. As you can see, it is necessary to give the option "-higherorder" to force the application of higher order predicates. Forgetting this option will result in an error, because the semantics file given is heavily based on the use of higher order predicates. '''Example 6-7.''' Let us consider the following example ||<#FFFF99>white : colour.[[BR]]black : colour. [[BR]]yellow : colour., [[BR]]carnivore[ canines *-> "sharp", color => colour ]. [[BR]]feline :: carnivore.[[BR]]feline[ genre *-> mammal ].[[BR]]lion :: feline.[[BR]]tiger:: feline.[[BR]]cat :: feline.[[BR]]tiger [color *-> yellow].[[BR]]whiteTiger :: tiger.[[BR]]whiteTiger [ color *-> white ].[[BR]]sherekhan : whiteTiger. || It describes the family of carnivores, and the relationship with felines. Here we can see that a carnivore is characterized by its sharp canines and its colour, while a feline is a carnivore whose genre is mammal. Then it gives some races of felines, like lion and tiger, each one characterized by something different. Let us concentrate on the tiger. The default color of a tiger is yellow, whith white stripes. A particular case of a tiger is a white tiger, which is white coloured. Sherekhan is an instance of a white tiger, and so we must suppose it is white coloured. Here a problem arises, in that sherekhan, being a white tiger, is also a tiger, and so it should be yellow, because color is inherited. Here non-monotonic inheritance is used,and so sherekhan takes its color from the most specific which it belongs to, in this case whiteTiger. This is realized thanks to the constrained semantics file, which imposes this behavior. if we launch the evaluation of the program with the command: {{{ > }}} '''{{{./dlt -higherorder constrained_semantics whiteTiger.dlf}}}''' the result will be the following: {{{ {white:colour, black:colour, yellow:colour,sherekhan:carnivore,sherekhan:feline,sherekhan:tiger,sherekhan:whiteTiger,colour::colour, carnivore::carnivore, feline::carnivore, feline::feline, lion::carnivore, lion::feline, tiger::carnivore,tiger::feline, tiger::tiger, cat::carnivore,cat::feline, whiteTiger::carnivore, whiteTiger::feline, whiteTiger::tiger,whiteTiger::whiteTiger,carnivore[canines*->"sharp",color=>colour],cat[color=>colour,genre*->mammal,canines*>"sharp"],feline[color=>colour,genre*->mammal,canines*->"sharp"],lion[color=>colour,genre*>mammal,canines*>"sharp"],sherekhan[color->white,genre->mammal,canines->"sharp"],tiger[color=>colour,color*->yellow,genre*->mammal,canines*->"sharp"],whiteTiger[color*->white,color=>colour,genre*->mammal,canines*>"sharp"]} }}} It is possible to see that dlt entails that sherekhan is carnivore, feline and a tiger, and so it has all the attributes inherited from these classes, but as it belongs to the class whiteTiger, its color is white,as we wanted to be. ==== Section 6.2. Frame Space directive ==== The Frame Space is a new directive introduced in DLT: ''@name.'' From the point after the directive each frame is interpreted as belonging to the frame space 'name'. A frame X[f->Y] is rewritten as f(X,Y,name). ''@.'' Cancels out the default Frame Space. Goes back to the normal rewriting X[f->Y] = f(X,Y). ---- == Chapter 7. System Architecture == A DLP^T^ program P, is sent to a DLP^T^ pre-parser, which performs syntactic checks (included nonrecursivity checks), and builds an internal representation of the DLP^T^ program. The DLP^T^ Inflater produces an equivalent SOLVER program P'; P' is piped towards the SOLVER system. The models M(P') of P', computed by SOLVER, are then converted in a readable format through the Post-parser module; the Post-parser filters out from M(P') informations about internally generated predicates and rules. '''Figure 1.''' attachment:flow.gif ==== Section 7.1 DLT Work-Flow ==== We show here the work-flow of DLT, exploiting [#ES63 example 6-3]. We launch the program with ''-preparsing'' option and divert the output in the file (''tmp''):[[BR]] [[Anchor(HIG1)]] {{{ > }}} '''{{{./dlt -silent -higherorder -preparser nome_file >tmp}}}''' The file ''tmp'' will appear as following: {{{ % LEGENDA OF TEMPLATE INVOCATIONS % END OF LEGENDA a_2(married,carlo,camilla). a_2(children,camilla,tom). a_2(engaged,tom,sara). a_1(man,tom). a_2(children,camilla,laura). a_2(engaged,laura,harryT). a_1(woman,laura). a_2(father,camilla,bruce). a_1(woman,camilla). a_2(children,carlo,william). a_2(mother,william,diana). a_1(man,william). a_2(children,carlo,harry). a_2(mother,harry,diana). a_1(man,harry). a_2(mother,carlo,elisabeth). a_1(woman,elisabeth). a_2(father,carlo,philip). a_1(man,philip). a_1(man,carlo). }}} Passing the content of ''tmp'' to the solver (i.e., DLV in this case) and diverting the output in another file (''tmp2''): {{{ > }}} '''{{{./dl -silent tmp > tmp2}}}''' the file ''tmp2'' will look as following: {{{ {a_2(married,carlo,camilla), a_2(children,carlo,william), a_2(children,carlo,harry), a_2(children,camilla,tom), a_2(children,camilla,laura), a_2(engaged,tom,sara), a_2(engaged,laura,harryT), a_2(father,carlo,philip), a_2(father,camilla,bruce), a_2(mother,carlo,elisabeth), a_2(mother,william,diana), a_2(mother,harry,diana), a_1(man,carlo), a_1(man,tom), a_1(man,william), a_1(man,harry), a_1(man,philip), a_1(woman,camilla), a_1(woman,laura), a_1(woman,elisabeth)} }}} [[Anchor(POST)]] Finally, we send the content of ''tmp2'' to DLT postparser module as following: {{{ > }}} '''{{{./dlt -postparsing <tmp2 }}}''' and we obtain the following: {{{ {married(carlo,camilla), children(carlo,william), children(carlo,harry), children(camilla,tom), children(camilla,laura), engaged(tom,sara), engaged(laura,harryT), father(carlo,philip), father(camilla,bruce), mother(carlo,elisabeth), mother(william,diana), mother(harry,diana), man(carlo), man(tom), man(william), man(harry), man(philip), woman(camilla), woman(laura), woman(elisabeth)} }}} the same output that we would have obtained launching the command: {{{ > }}} '''{{{./dlt family}}}''' Indeed, performing the whole process launching directly DLT as above, the only model is: {{{ {married(carlo,camilla), children(carlo,william), children(carlo,harry), children(camilla,tom), children(camilla,laura), engaged(tom,sara), engaged(laura,harryT), man(carlo), man(tom), man(william), man(harry), man(philip), woman(camilla),woman(laura), woman(elisabeth), father(carlo,philip), father(camilla,bruce), mother(carlo,elisabeth), mother(william,diana), mother(harry,diana)} }}} exactly the same of what obtained performing explicitly all the operations constituting the flow. ==== Section 7.2. Compatibility with other solvers ==== [[Anchor(SOLV)]] DLT can virtually support any solver. At the moment, we can state the compatibility with the syntax of some of them. The following table shows the instructions of the solvers which dlt natively recognizes. ||< : 15% #99CCCC> '''SOLVER''' ||< : 15% #99CCCC> '''WHAT'S SUPPORTED''' ||< : 15% #99CCCC> '''SYNTAX'''|| ||< : #ADD8E6> ''any solver'' ||<: #ADD8E6> ASITIS ||<: #ADD8E6> ASITIS%* Body *% [[BR]] ''ASITIS's Body is sent verbatim to solver''|| ||< : #ADD8E6> ''dlv'' ||<: #ADD8E6> Aggregates and weak constraints ||<: #ADD8E6> see [http://www.dbai.tuwien.ac.at/proj/dlv/man/ DLV - User Manual ]|| ||< : #ADD8E6> ''dlv+odbc'' ||<: #ADD8E6> Not tested yet ||<: #ADD8E6> see [http://www.dbai.tuwien.ac.at/proj/dlv/man/ DLV - User Manual ]|| ||< : #ADD8E6> ''dlvex'' ||<: #ADD8E6>external built-ins ||<: #ADD8E6> see [http://www.mat.unical.it/ianni/wiki/dlvex?action=AttachFile&do=get&target=usermanual.html dlvex - User Manual ] || ||< : #ADD8E6> ''dlvhex'' ||<: #ADD8E6>external atoms, namespace directive||<: #ADD8E6> see [http://con.fusion.at/dlvhex/ dlvhex - User Manual ] || ||< : #ADD8E6> ''smodels'' ||<: #ADD8E6> Supports normal programs with function symbols ||<: #ADD8E6> see [http://www.tcs.hut.fi/Software/smodels/ smodels - User Manual ] || '''Note:''' All the SOLVER options must be passed using the option ''-solverparams={}''. ==== Section 7.3. Notes on using DLV-HEX with DLT ==== DLV-HEX has higher order reasoning built-in, so it is not necessary to invoke DLT with the option {{{-higherorder}}} switched on. Also, DLV-HEX supports natively DLT. DLT can be used with DLV-HEX with different command line combinations such as {{{ dlt -silent -preparsing program.dlt | dlvhex --silent -- | dlt -postparsing }}} or {{{ dlvhex --dlt program.dlt }}} makes dlvhex to invoke dlt as a preparser | 
DLT USER MANUAL
Overview
DLT is a front-end for disjunctive datalog, extending the DLV system with Template predicates, Frame Logic and higher order predicate names. Template predicates can be seen as a way to define intensional predicates by means of a subprogram, where the subprogram is generic and reusable as many times is necessary. Frame Logic extends classical logic and enables frame-based or object-oriented syntax. Higher order predicates allow to treat predicate names as variables (although this semantics has quite nothing to deal with second order logics). BR
More information (including an online version of this manual and an online tutorial) and the executable program for Linux platform are available at the DLT homepage (http://dlt.gibbi.com).[[BR]]
Chapter 1. Getting Started
We can invoke the DLT system directly from the command-line, after the prompt of the system, as well the DLV system. Note that DLT relies on an external solver ES, which takes an input generated by DLT. Models generated by ES are then parsed back to the DLT syntax. ES can be chosen from a variety of solvers (see the Compatibility Table below)
If you do not specify any options or files, DLT will just print some informational output:
> ./dlt BR {{{ DLT. Disjunctive Logic Framework with Templates and more [V1.5.1 - Build DEV/Jul 22 2008 gcc 4.0.1 (Apple Inc. build 5465)]
usage: ./dlt {FRONTEND} {OPTIONS} [--higherorder] [--solver=<path>] [--preparsing] [--postparsing] [--framespace] [--uri] [--filter=<predicate>] [--pfilter=<predicate>] [--silent] [--aritycheck] [--safetycheck] [--solverparams={options_solver}] [filename [filename [...]]]
Specify -help for more detailed usage information.}}} BR The first line shows, first of all, the name of the executable program (DLT), then it gives information about the version of the executable program, the date when the binary was built and the identifier of the compiler. The last three lines give a brief explanation on how to use dlt, the last line informing the user of the -help option for futher instructions.BR Anchor(HELP)The -help or -h optionBRBR
> ./dlt -help or > ./dlt -h BRBR gives the following output: BR
DLT. Disjunctive Logic Framework with Templates and more
[V1.5.1 - Build DEV/Jul 22 2008   gcc 4.0.1 (Apple Inc. build 5465)]
usage:  ./dlt {FRONTEND} {OPTIONS} [--higherorder] [--solver=<path>] [--preparsing] [--postparsing] [--framespace] [--uri] [--filter=<predicate>] [--pfilter=<predicate>] [--silent] [--aritycheck] [--safetycheck] [--solverparams={options_solver}] [filename [filename [...]]]
DLT Specific options:
--higherorder               Enables higher order predicate names
--solver=[path_name]        The path of the solver executable file 
                            For example: --solver=/dlv/dl
--preparsing                Print out re-written code and exit
--postparsing               Maps higher-order atoms into first-order atom
--framespace                Activates displaying for framespace directives
--uri                       Enables more verbose output for URI constants
--filter=name1,..,nameN     Output only instances of the specified predicates
--pfilter=name1,..,nameN    Output only positive of the  instances 
                            specified predicates
--solverparams={options}    The specific solver options 
                            For example: -solverparams={-v -nofacts}
--aritycheck                Forces to check same arity for the same
                            predicate name
--safetycheck               Forces to check safety of rule's head variablesAnchor(SIL) If you do not want to see this status line, use the -silent option, which suppresses various informational output and blank lines.
- > ./dlt -silent 
{{{ usage: ./dlt {FRONTEND} {OPTIONS} [--higherorder] [--solver=<path>] [--preparsing] [--postparsing] [--framespace] [--uri] [--filter=<predicate>] [--pfilter=<predicate>] [--silent] [--aritycheck] [--safetycheck] [--solverparams={options_solver}] [filename [filename [...]]]
Specify -help for more detailed usage information.
}}}
Section 1.1. DLT Specific options
The following table shows the specific options of DLT.BRBR note: The –filter e –pfilter options are inherited by DLV, but they must to be considered as specific options of DLT. Indeed, DLT can change the predicate names before passing them to DLV ( DLT set them up again it in the post-parsing phase ). Thus, it’s necessary that DLT performs filtering operations directly, before printing the results.
| OPTION | EXPLANATION | 
| [#HIGH -higherorder] | Considers the names of the predicates as variables | 
| -solver=[path_name] | Allows to specify the path of the solver | 
| [#PREP -preparsing] | Prints to standard output the program rewritten by DLT ans stops without invoking the solver | 
| [#POST -postparsing] | Receives the output of the solver from standard input and maps data to the original DLT format (before rewriting) | 
| -filter=pred | Turns on filtering. Predicates pred (or true negated -pred ) will be the only included in the output. This option may be used multiple times, and pred may also be a comma-separated list of predicate names. | 
| -pfilter=pred | Turns on filtering. Positive instances of predicate pred will be included in the output. This option may be used multiple times, and pred may also be a comma-separated list of predicate names. | 
| [#SOLV -solverparams={options}] | In the curly brackets the user may pass a space separated list of options to be passed to the solver. | 
| [#SIL -silent] | Do not output information about the version of the executable program and the compiler. | 
| [#ARC -aritycheck] | Forces to check same arity for the same predicate name | 
| [#SAC -safetycheck] | Forces to check safety of rule's head variables | 
| [#HELP -help] or [#HELP -h] | Gives a detailed explanation on how to use DLT. | 
Chapter 2. DLP^T Language
DLT's native language is [http://www.dbai.tuwien.ac.at/proj/dlv/ DLV] extended with Template predicates and other constructs. We remember that DLV's native language is Disjunctive Datalog extended with constraints, true negation, aggregates and queries.
How to write a DLPT program
For a correct writing and usage of the DLPT language here are a few general conditions that must be satisfied.
Section 2.1. Writing Template Definition
A template definition D consists of a template header and an associated DLPT subprogram closed into curly braces. We consider a generic definition of template:
All the following conditions must be respected:
- In the body of a template definition at least a rule must have an atom in its head with the same name of the template in which it is defined;
- The formal parameters (p1(a1),..., pn(an)) cannot appear in a head; 
- All the formal parameters should appear in some rules;
- In the program cannot exist two or more template definitions having the same name;
- The formal parameters cannot be preceded from the true negation symbol and the negation as failure simbol in the template header in which they are uses;
- The global predicates defined using the GLOBAL directive cannot appear in the head of a rule defined in the body of the template;
- Definitions of template cannot appear in the body of a template;
Example 2-1. Correct Template Definitions
| c1: #template max{p(1)}(1)BR { BRBR max(X) :- p(X), not exceeded(X). BR exceeded(X) :- p(X), p(Y), Y > X. BRBR } | 
| c2: #templatemax{p(1)}(1)GLOBAL nodeBR { BRBR max(X) :- p(X), not exceeded(X).BR exceeded(X) :- p(X), p(Y), Y > X.BRBR } | 
Example 2-2 Wrong Template Definitions
t1 is a wrong template definition because the formal parameter pcannot appear in the head of a rule.
t2 is a wrong template definition because the formal parameter p is preceded from the true negation symbol, while the formal parameter c is preceded from the negation as failure symbol.
Anchor(ES23) Example 2-3. The following is a template definition:
| #template max{p(1)}(1)BR { BRBRmax(X) :- p(X), not exceeded(X).BRexceeded(X) :- p(X), p(Y), Y > X.BRBR { | 
This template introduces a generic template program, defining the predicate max, intended to compute the maximum value over the domain of a generic unary predicate p. In this example we have that
| #template max{p(1)}(1) | 
is the template header, and
is a DLPT subprogram. max is the template name; p is a predicate name, called formal predicate. The arity of formal predicate p is 1 and the output arity of template definition is 1.
Section 2.2. Writing Template Atoms
We consider a generic definition of template:
| template_name{ p1(X1),..., pn(Xn)}(Args) | 
All the following conditions must be respected:
- For every template atom, defined in some rule of the general program, there must exist a definition having the same name of template_name; 
- The number of actual predicates (p1(X1),..., pn(Xn)) must be equal to the number of formal predicates of the correspondent definition; 
- The arity of the output terms must be equal to the output arity of the correspondent definition;
- The number of asterisks of every actual parameter must be equal to the arity of the correspondent formal parameter;
- The actual parameters cannot appear negated in the atom template that uses them(note that true negation is allowed instead).
Example 2-4 Correct Template Atoms
| c1: template_name{-person($,*,*)}(X). | 
| c2: template_name{person($,*,*)}(X). | 
| c3: template_name{person(Name,*,*)}(X). | 
| c4: template_name{-person(Name,*,*)}(X). | 
Example 2-5 Wrong Template Atoms
We consider the template definition of the Example 2-3(definition of template max). The following are wrong template atoms:
| a1: maxperson(Name,*,*)}(X,Y,Z). | 
| a2: max{not person(Name,*,*)}(X). | 
| a3: max{person(Name,*,*), person(*,*,$)}(X). | 
a1 is a wrong template atom because the arity of the output terms is not equal to the output arity of the correspondent definition.
a2 is a wrong template atom because the actual parameter person is preceded by negation as failure.
a3 is a wrong template atom because the number of actual predicates is not equal to the number of formal predicates of the correspondent definition.
Note: The DLPT subprogram may contain disjunctive rules, constraints, weak constraints, aggregates, built-in predicates, negative literals, true negation, template atoms (or invocations), and it do not contains template definitions.
Section 2.3. Negative Rules with Template Atoms and True Negation
Template atoms can be used also as negated and negative atoms in the generic rules of the program. Then we can use a template atom after the negation as failure symbol (not) as it happens for the usual atoms.
Example 2-6. The following is an example of negated and negative template atom, respectively.
| not max{student(Sex,$,*)}(Age) | 
Negation for template atoms is treated as well negation for DLV atom. For more information about negation as failure and true negation we send back at the DLV user manual available at the DLV homepage (http://www.dbai.tuwien.ac.at/proj/dlv/).
Section 2.4. Template Definition with Global Predicates
Template programming can become a very useful feature in applications where it is necessary to compact repetitive pieces of code. We consider a planning program for the Space Shuttle Reaction Control System. Here is a sketch from the program:
| template_name ready_to_fire_conditions{propulsor_type(2)}(2)BR GLOBAL time_slot, tank, is_pressurized_by, is_damaged BR { BRBRready_to_fire_conditions(J,T) :-propulsor_type(J,R),BRtime_slot(T),tank(TK1,R),BRtank(TK2,R), TK1 !=TK2,BRis_pressurized_by(J,TK1,T),BRis_pressurized_by(J,TK2,T),BRnot is_damaged(J).BRBR } | 
the above template is employed to remove a set of repeated rules, which are changed this way
| fire_jet(J,T) :- ready_to_fire_conditions{jet(*,*)}(J,T). | 
| fire_vernier(J,T) :- ready_to_fire_conditions{vernier(*,*)}(J,T). | 
Section 2.5. Group-by
A template definition may be instantiated as many times as necessary, trough template atoms. The template atom or invocation consists of a template name, a list of atom and a list of variables and/or constants (literal constants and numerical constants).
Example 2-7. The followings are template atoms:
| max{weight(*)}(M) | 
| max{student($,Sex,*)}(Age) | 
max is the template name; weight and student are predicate names, called actual predicates; variables and constants are called standard terms (Sex is a standard term); the dollar $ symbol is called projection term and the symbol * is called parameter term. M and Age are a list of usual terms (i.e. either variables or constants), and are called output lists.
We note that the template definitions may be unified with a template atom in many ways. The above example contains a plain invocation ( max{weight(*)}(M) ), and a compound invocation ( max{ student($,Sex,*) }(Age) ). Intuitively, projection terms ($ symbols) are intended in order to indicate attributes of an actual predicate which have to be ignored. A standard term (a constant or a variable) within an actual atom indicates a group-by attribute, whereas parameter terms (* symbols) indicate attributes to be considered as parameter.
For example, the template atom max{student($,Sex,*)}(Age) allows to employ the definition of the template predicate max on a ternary predicate student, discarding the first attribute of student, and grouping by values of the second attribute. The intuitive meaning of this invocation is to define a predicate computing the student with maximum value of the Age attribute (the third attribute of the student predicate), grouped by the Sex attribute (the second one), ignoring the first attribute. The computed values of Age are returned through the output list.
Example 2-8. We consider the following EDB:
| person(riccy,f,26). BR person(kali,f,26). BR person(peppe,m,26). BR person(gibbi,m,29). BR person(paddy,f,27). | 
it represents a set of persons of which it is known name, sex and age. We consider the template definition of max given in [#ES23 Example 2-3]. The following:
| older_sex(Name,Sex,Age):- max{person($,Sex,*)}(Age),person(Name,Sex,Age). | 
is a template atom that compute the oldest person with grouping on the sex attribute, that is the person of age greater than female sex and that one of male sex. Launching dlt from command line, as it follows:
- > ./dlt -silent -solverparams={-nofacts} programFileName 
we obtain the model
{{{ {older_sex(paddy,f,27),older_sex(gibbi,m,29)} }}}
Therefore paddy is the greatest person of female sex, while gibbi is the greateest person of male sex.
Example 2-9. We consider another example and the following template:
| #template sum{emp(2)}(1)BR{BRBRprecedes(X,Y) :- emp(X,_), emp(Y,_), X<Y.BRsucc(X,Y) :- precedes(X,Y), not elementInMiddle(X,Y).BRelementInMiddle(X,Y) :- precedes(X,Z), precedes(Z,Y).BRfirst(X) :- emp(X,_), not hasPredecessor(X).BRlast(X) :- emp(X,_), not hasSuccessor(X).BRhasPredecessor(X) :- succ(Y,X).BRhasSuccessor(Y) :- succ(Y,X).BRpartialSum(X,Sx) :- first(X), emp(X,Sx).BRpartialSum(Y,S) :- succ(X,Y), partialSum(X,PSx), emp(Y,Sy),S=PSx+Sy.BRsum(S) :- last(L), partialSum(L,S).BRBR} | 
It allows to sum all the values of the second attribute of the argument emp in an iterative manner. The first attribute must be an ID which identifies unambiguously the instance. The template first orders the instances of emp by their ID (first rule), then using the next rules it defines the successor of each instance. It introduces a new atom, succ(I,II), thus finding the first and the last of the list (first(X),first(X)). The predicate partialSum scans the entire ordered list and addes the sum of the preceding values to the following value. The sum predicate receives the value associated with the last instance of emp. This is possible due to the predicate last.
Now we consider the following EDB:
| employee(1,gisella,direction,80).BRemployee(2,carlo,production,36).BRemployee(3,giuseppe,administration,40).BRemployee(4,franco,distribution,45).BRemployee(5,carla,administration,45).BRemployee(6,lorenzo,direction,73).BRemployee(7,paola,production,46).BRemployee(8,marco,administration,40).BRemployee(9,gianni,production,32). | 
it represents a set of persons of which it is known ID, name and salary. We use the template sum and we consider the following:
result(DIP,S):- sum{impiegato(*,$,DIP,*)}(S).
this is a template atom which calculates the sum of the values of the 4th attribute of employee (salary) also taking into consideration the first (ID), ommiting the second(name) and grouping the result by the third attribute(department).
Launching dlt from command line, as it follows:
> ./ dlt name_file -silent -filter=result
we obtain the model
{{{{result(direction,153), result(production,114), result(administration,125), result(distribution,45)} }}}
The result is obtained in a simple manner. First, the template is executed as if the DIP parameter, which is later used to group the results, is only used for the selection. Therefore the instances of employee are ordered by their ID. Then the DIP and salary of each are selected. A table with a row for each employee and two columns, the first for departiment and the second for salary is obtained. The table is then analysed dividing the rows in groups defined by their DIP values. After this, the sum is separately calculated for each group. The output (filtering only the result predicate) contains the sum of the salaries for each departiment.
Chapter 3. Safety
Safety of variables in template atom follows the usual criterion. We suppose to have in the body of a generic rule a negated invocation; then must be valid the following condition must be valid:
| Safety for the Negated Template Atom | 
| Each variable occurring in the output list of the given invocation also occurs in at least one non-comparative positive literal in the body of the same rule (resp. the same constraint). | 
Example 3-1. Unsafe Rules and Constraints
| u1: head(Y) :- not template_name{p(M,$,*)}(const). | 
| u2: :- not template_name{p($,M,*)}(X). | 
u1 is unsafe because the variable Y that appear in the head of the rule does not appear in at least a positive atom of the body of the same rule.
u2 is unsafe because the variable X in the output list of the given invocation does not appear in at least a positive atom of the body of the same rule.
Example 3-2. Safe Rules and Constraints
| s1: head(Y) :- not template_name{p(M,$,*)(const), predicate(_,_,Y). | 
| s2: :- not template_name{p($,M,*)}(X), predicate(_,_,X). | 
| s3: head(X) :- not template_name{p(M,$,*)}(Y),predicate(_,X,Y). | 
| s4: :- template_name{p(M,$,*)}(X). | 
| s5: :- not template_name{p(M,$,*)}(const). | 
Chapter 4. Knowledge Representation by DLP^T
In this section we show by example the main advantages of template programming. Example put in evidence the easiness of providing a succinct and elegant way for quickly introducing new constructs using the DLPT language.
Section 4.1. Aggregates
Aggregate predicates allow to represent properties over sets of elements: the next example shows how to fast prototype aggregate semantics without taking into account of the efficiency of a built-in implementation.Here we take advantage of the template predicate max, defined in Example 2.1. The next template predicate defines a general program to count distinct values of a predicate p, given an order relation succ defined on the domain of p.
| #template count{p(1),succ(2)}(1)BR { BRBR partialCount(0,0). BR partialCount(I,V) :- not p(Y), I=Y+1,partialCount(Y,V). BR partialCount(I,V2) :- p(Y), I=Y+1,partialCount(Y,V), BR succ(V,V2). BR partialCount(I,V2) :- p(Y),I=Y+1,partialCount(Y,V), BR max[succ(*,$)](V2). BR count(M) :- max{partialCount($,*)}(M). BRBR } | 
The above template definition is conceived in order to count, in a iterative-like way, values of the p predicate through the partialCount predicate. A ground atom partialCount(i,a) means that at the stage i, the constant a has been counted up. The predicate count takes the value which has been counted at the highest (i.e. the last) stage value. It is worth noting how max is employed over the binary predicate partialCount, instead of an unary one. Indeed, the $ and * symbols are employed to project out the first argument of partialCount. The last rule is equivalent to the piece of code:
| partialCount'(X) :- partialCount(_,X). | 
| count(M) :- max{partialCount'(*)}(M). | 
Template definitions can be employed to introduce and reuse constructs defining the most common search spaces. This improves declarativity of ASP programs to a larger extent. The next two examples show how to define a predicate subset and a predicate permutation, ranging, respectively, over subsets and permutations of the domain of a given predicate p.
| #template permutation{p(1)}(2). BR { BRBR permutation(X,N)v npermutation(X,N) :- p(X),#int(N), N <= N1, BR count{p(*),>(*,*)}(N1). BR :- permutation(X,A),permutation(Z,A), Z <> X. BR :- permutation(X,A),permutation(X,B), A <> B. BR covered(X) :- permutation(X,A). BR :- p(X), not covered(X). BRBR } | 
The explanation of the subset template predicate is quite straightforward. As for the permutation definition, a ground atom permutation(x,i) tells that the element x (taken from the domain of p), is in position i within the currently guessed permutation. The rest of the template subprogram forces permutations properties to be met.
Next we show how count and subset can be exploited to succinctly encode the k-clique problem, i.e., given a graph G (represented by predicates node and edge), find if there exists a complete subgraph containing at least k nodes (we consider here the 5-clique problem):
| in_clique(X) :- subset{node(*)}(X). BR :- count{in_clique(*),>(*,*)}(K),K < 5. BR :- in_clique(X),in_clique(Y), X <> Y, not edge(X,Y). | 
The first rule of this example guesses a clique from a subset of nodes. The first constraint forces a candidate clique to be at least of nodes, while the last forces a candidate clique to be strongly connected. The permutation template can be employed, for instance, to encode the Hamiltonian Path problem: given a graph G, find a path visiting each node of G exactly once:
| path(X,N) :- permutation{node(*)}(X,N). BR :- path(X,M), path(Y,N), not edge(X,Y), M = N+1. | 
Sets: Extending Datalog with Set programming is another matter of interest for the ASP field. It is fairly quick to introduce set primitives using DLPT; a set S is modeled through the domain of a given unary predicate s. Intuitive constructs like intersection, union, or symmetricdifference, may be modeled as follows.
| #template symmetricdifference{a(1),b(1)}(1) BR { BRBR symmetricdifference(X) :- union{a(*),b(*)}(X), BR not intersection{a(*),b(*)}(X). BRBR } | 
Dates: managing time and date data types is another important issue in engineering applications of DLP. The following template shows how to compare dates represented through a ternary relation day, month, year.
| #template before{date1(3),date2(3)}(6) BR { BRBR before(D,M,Y,D1,M1,Y1) :- date1(D,M,Y), BR date2(D1,M1,Y1), Y < Y1. BR before(D,M,Y,D1,M1,Y1) :- date1(D,M,Y), BR date2(D1,M1,Y1), Y == Y1, M < M1. BR before(D,M,Y,D1,M1,Y1) :- date1(D,M,Y), date2(D,M1,Y1), BR Y == Y1, M = M1, D < D1. BRBR } | 
Chapter 5. Higher order predicates
With the use of higher order predicates it is possibile to consider the names of the predicates as variables. The user inserts atoms as if they where of first order, but if the option -higherorder is used it is possible to execute higher order queries. Moreover they are fundamental for the [#HIG1 Frame Logic] and for the use of predicates that may appear with [#HIG2 different arities].
Example 5-1. We consider the following EDB:
| sister(gisella,maria).BRmother(maria,giovanni).BRhusband(gino,carmela).BRfriend(gisella,rosina).BRaunt(gisella,sara). | 
it represents a set of persons; the names of the predicates define the relations between the persons. The use higher order allows the user to use the names of the predicates as though they where attributes of other predicates. The following rule simply rewrites the facts, but also allows the user to understand how to perform higher order queries.
| relationship(X,R,Y):- R(X,Y). | 
Launching dlt from command line with option -higherorder, as it follows:
> ./dlt name_file -higherorder -solverparams={-nofact}
we obtain the model
{relationship(gisella,sister,maria), 
relationship(gisella,friend,rosina), 
relationship(gisella,aunt,sara), 
relationship(maria,mother,giovanni), 
relationship(gino,husband,carmela)}Please note that now the predicate names appear also as attributes of the relations (see for instance relationship(gisella,sister,maria)).
Chapter 6. Frame Logic
F-logic is really useful, since it allows to model reality in a very simple way, exploiting its object-oriented programming features. Moreover, using frames helps to keep the code compact and enhances readability with respect to having a big set of single rules. Indeed, in principle, everything which can be expressed with frames can be also done using predicates and standard atoms; however frames make programs simpler and more intuitive. Like in OO programming languages, classes and objects are both supported. When declaring an object, it can in fact belong to a class. A frame object has usually a list of attributes, which can eventually be empty. It is possible to create a Frame Molecule and place it either in the head or in the body of a rule. A simple frame molecule is:
| nameframe[nameattribute->valueattribute]. | 
The above contains a single attribute expression nameattribute->valueattribute.
Example 6-1.
| brown:employee[ surname -> "Mr. Brown",skill ->> {java, asp}, salary -> 800 , married]. | 
Here we see interesting features. The object "brown" has been defined as belonging to the class "employee" by means of the operator ":". Then a list of attributes is associated to the object. Here we see two types of attributes, single-valued (characterized by the "->" arrow), and multi-valued (characterized by the "->>" arrow). Multi-valued attributes are very useful, as they permit the use of sets of values. In this case they are well suited to represent the set of skills of an employee. Please note that in case of multi-valued attributes, the curly brackets are used. They are usually omitted when there is only one attribute in the set. Attributes can also be boolean-valued.
- The last clause is a boolean attribute which indicates that married is true for the object brown. 
With frames, as we have seen so far, it is possible to describe objects with ease, but they are very useful also to describe classes and taxonomies of classes, with many expressive constructs. To describe a taxonomy, we basically can make use of the binary operator "::", which means that the first operand is subclass of the second operand.
Example 6-2.
| webDesigner::javaProgrammer. javaProgrammer::Programmer. | 
In this example we can see that webDesigner is defined as a subclass of javaProgrammer, which in turn is defined as a subclass of Programmer. The subclassOf operator is transitive, and so webDesigner is a subclass of Programmer too.
When describing classes it is possible, like in java or c++, to list a certain number of attributes, specifying either their value or their type. To do this, a wide number of "arrow" operators is available to the user, such as ->, ->>, *->, *->>, =>, =>>. These operator on their own do not have a strong semantic meaning. Actually, they take the classical F-logic meaning when coupled with specific axioms, which we will discuss later.
Example 6-3.
| programmer[salary => integer]. | 
The "=>" and "=>>" operators are used to describe the type of an attribute. Both support set of types for each attribute. If you want to use this features, keep in mind that the different values must be comma-separated and enclosed in curly brackets.
Example 6-4.
| employee[skill =>>{string, int}]. | 
The main difference between the two operators is that they "map" on different operator applied to instances. In particular, the operator "=>" maps on the instance operator "->", while "=>>" maps on "->>". Except this, they are pretty similar.
In a similar way it is possible to define the values associated to an attribute. Being in the context of classes, these values can be thought about as "default values" for the class. To associate values to a class attribute, different operators are available. As for the type operators, they have a particular meaning only when coupled to a F-logic semantics. The available operators are: ->,->>,*->,*->>. By using the operators with "*", usually the will is to specify that the value assigned to the attribute is inherited automatically by the subclasses. This is not forced by default, actually. In general, "->" and "->>" operators work on single values, while the others work on sets.
Example 6-4.
| employee[skill ->> {java, asp}, salary *->1000]. | 
While some attributes may be described as facts, other may be defined using inference rules. For example, suppose that john is married to mary and mary has three children. It can be assumed that john is the father of alice, nancy, and jack (the children of mary). Therefore the following fact:
| john[ children->>{ alice,nancy,jack } ]. | 
may be automatically derived if we add the following rule:
| X[ children -> C ] :- Y[ spouse->X,children->C ]. | 
In addition it is possible to specify the class an object belongs to using the following rules:
| john: man.BRmary: woman. | 
Once the class of the object is identified, a specific list of attributes can be assigned at the same time:
| mary: woman[ children->>{ alice,nancy,jack } ]. | 
Anchor(ES63) Example 6-5. We consider an other definition frame:
| carlo:man[ married->camilla: woman[ children->>{ tom: man[ engaged->sara ],BR laura: woman[ engaged->harryT ] },BR father->bruce ],BR children->>{ william: man[ mother->diana ],harry: man[ mother->diana ] },BR mother->elisabeth: woman,BR father->philip: man ].BR | 
The frame describes the relationships existing between some member of the Britannic royal family. carlo is a man and he’s married with camilla. He has two children: william and harry, and their mother was diana. carlo ‘s mother is elisabeth and his father is philip. camilla is a woman and she has a daughter, laura, and a son, tom. laura is betroth with harryT , and tom ’s girlfriend is sara.
Launching dlt from command line:
> ./dlt name_file
we obtain the model
{carlo:man, camilla:woman, tom:man, laura:woman, william:man, harry:man, elisabeth:woman, philip:man, man::man, woman::woman, camilla[children->>{tom,laura},father->bruce],carlo[married->camilla,children->>{william,harry},father->philip,mother->elisabeth],harry[mother->diana],laura[engaged->harryT],tom[engaged->sara],william[mother->diana]}The higher order atoms are fundamental for the frames. Indeed, the output consists of predicates whose names were attributes in the frames. For example look at the first part of the definition of the frame:
| carlo: man[ married->camilla ]. | 
carlo is a man, and he is married to camilla; 'married' is an attribute of the object carlo, but in the output it's presented as the name of a predicate:
| married(carlo, camilla). | 
Example 6-6. Let's show another example:
Anchor(HIG2) The frame says that Gianni is a student and has a pc named mistery. The pc is a notebook which uses debian as operating system. In addition information on the OS is given: kind and kernel. Launching this program without the option higherorder results with an error:
> ./dlt nome_file
os, first used with arity 2, now seen with arity 1.
The cause of the error is clear once we look at the opuput with the option -higherorder
> ./dlt -higherorder nome_file
{pc(gianni,mistero), os(mistero,debian), kind(debian,gnu), kernel(debian,linux), os(debian), notebook(mistero), student(gianni)}The atoms os(mistero,debian) e os(debian) appears in the output. The same predicate is present with different arity. In the case of first order atoms this is not permitted. It is only possible using higher order atoms.
Section 6.1. Semantic Modeling
One very interesting feature is that it is possible to specify, in addiction to the file containing the frame program, other files containing the specification of a semantic behavior. Various pre-written modules are available to the user. The modules have been designed in order to give a semantic meaning to the various operators described previously. We esemplify the meaning of these modules with the following table:
ovverridden(D,M,C) :- E[M -> V1], C::E, E::D, C!=E, E!=D. inheritable(C,M,D) :- C::D, D[M *-> V], not ovverridden(D,M,C). C[M *-> V] v -C[M *-> V] :- inheritable(C,M,D), D[M *->V]. exists(C,M) :- C[M *-> V]. :-inheritable(C,M,D), not exists(C,M). A[M -> V] :- A:C, C[M -> V].
| OPERATOR | AXIOM | AXIOM DESCRIPTION | AXIOMATIC MODULE | 
| :: | A::B :- A::C, C::B | ' | C | 
| :: | A::A:-X:A | ' | C | 
| :: | :- A::C, C::A, A!=C. | ' | C | 
| : | X:C :- X:D, D::C. | ' | C | 
| -> | - A[M -> V], A[M -> W], V!=W. | ' | F | 
| *-> | :- A[M *-> V], A[M *-> W], V!=W. | ' | F | 
| => | D[A => T] :- D::C, C[A => T]. | ' | S | 
| =>> | D[A =>> T] :- D::C, C[A =>> T]. | ' | S | 
| ' | ' | ' | ' | 
| ' | ' | ' | ' | 
| ' | ' | ' | ' | 
| ' | ' | ' | ' | 
| ' | ' | ' | ' | 
| ' | ' | ' | ' | 
| ' | ' | ' | ' | 
| ' | ' | ' | ' | 
> ./dlt -higherorder constrained_semantics filename
where constrained_semantics is the name of the file containing the stable models semantics, while filename is the file containing a dlt program. As you can see, it is necessary to give the option "-higherorder" to force the application of higher order predicates. Forgetting this option will result in an error, because the semantics file given is heavily based on the use of higher order predicates.
Example 6-7. Let us consider the following example
| white : colour.BRblack : colour. BRyellow : colour., BRcarnivore[ canines *-> "sharp", color => colour ]. BRfeline :: carnivore.BRfeline[ genre *-> mammal ].BRlion :: feline.BRtiger:: feline.BRcat :: feline.BRtiger [color *-> yellow].BRwhiteTiger :: tiger.BRwhiteTiger [ color *-> white ].BRsherekhan : whiteTiger. | 
It describes the family of carnivores, and the relationship with felines. Here we can see that a carnivore is characterized by its sharp canines and its colour, while a feline is a carnivore whose genre is mammal. Then it gives some races of felines, like lion and tiger, each one characterized by something different. Let us concentrate on the tiger. The default color of a tiger is yellow, whith white stripes. A particular case of a tiger is a white tiger, which is white coloured. Sherekhan is an instance of a white tiger, and so we must suppose it is white coloured. Here a problem arises, in that sherekhan, being a white tiger, is also a tiger, and so it should be yellow, because color is inherited. Here non-monotonic inheritance is used,and so sherekhan takes its color from the most specific which it belongs to, in this case whiteTiger. This is realized thanks to the constrained semantics file, which imposes this behavior.
if we launch the evaluation of the program with the command:
> ./dlt -higherorder constrained_semantics whiteTiger.dlf
the result will be the following:
{white:colour, black:colour, yellow:colour,sherekhan:carnivore,sherekhan:feline,sherekhan:tiger,sherekhan:whiteTiger,colour::colour, carnivore::carnivore, feline::carnivore, feline::feline, lion::carnivore, lion::feline, tiger::carnivore,tiger::feline, tiger::tiger, cat::carnivore,cat::feline, whiteTiger::carnivore, whiteTiger::feline, whiteTiger::tiger,whiteTiger::whiteTiger,carnivore[canines*->"sharp",color=>colour],cat[color=>colour,genre*->mammal,canines*>"sharp"],feline[color=>colour,genre*->mammal,canines*->"sharp"],lion[color=>colour,genre*>mammal,canines*>"sharp"],sherekhan[color->white,genre->mammal,canines->"sharp"],tiger[color=>colour,color*->yellow,genre*->mammal,canines*->"sharp"],whiteTiger[color*->white,color=>colour,genre*->mammal,canines*>"sharp"]}It is possible to see that dlt entails that sherekhan is carnivore, feline and a tiger, and so it has all the attributes inherited from these classes, but as it belongs to the class whiteTiger, its color is white,as we wanted to be.
Section 6.2. Frame Space directive
The Frame Space is a new directive introduced in DLT:
@name.
- From the point after the directive each frame is interpreted as  belonging to the frame space 'name'. A frame X[f->Y] is rewritten as f(X,Y,name). 
@.
- Cancels out the default Frame Space. Goes back to the normal  rewriting X[f->Y] = f(X,Y). 
Chapter 7. System Architecture
A DLPT program P, is sent to a DLPT pre-parser, which performs syntactic checks (included nonrecursivity checks), and builds an internal representation of the DLPT program. The DLPT Inflater produces an equivalent SOLVER program P'; P' is piped towards the SOLVER system. The models M(P') of P', computed by SOLVER, are then converted in a readable format through the Post-parser module; the Post-parser filters out from M(P') informations about internally generated predicates and rules.
Figure 1.
attachment:flow.gif
Section 7.1 DLT Work-Flow
We show here the work-flow of DLT, exploiting [#ES63 example 6-3]. We launch the program with -preparsing option and divert the output in the file (tmp):BR Anchor(HIG1) > ./dlt -silent -higherorder -preparser nome_file >tmp
The file tmp will appear as following:
% LEGENDA OF TEMPLATE INVOCATIONS % END OF LEGENDA a_2(married,carlo,camilla). a_2(children,camilla,tom). a_2(engaged,tom,sara). a_1(man,tom). a_2(children,camilla,laura). a_2(engaged,laura,harryT). a_1(woman,laura). a_2(father,camilla,bruce). a_1(woman,camilla). a_2(children,carlo,william). a_2(mother,william,diana). a_1(man,william). a_2(children,carlo,harry). a_2(mother,harry,diana). a_1(man,harry). a_2(mother,carlo,elisabeth). a_1(woman,elisabeth). a_2(father,carlo,philip). a_1(man,philip). a_1(man,carlo).
Passing the content of tmp to the solver (i.e., DLV in this case) and diverting the output in another file (tmp2):
> ./dl -silent tmp > tmp2
the file tmp2 will look as following:
{a_2(married,carlo,camilla), a_2(children,carlo,william), a_2(children,carlo,harry), a_2(children,camilla,tom), a_2(children,camilla,laura), a_2(engaged,tom,sara), a_2(engaged,laura,harryT), a_2(father,carlo,philip), a_2(father,camilla,bruce), a_2(mother,carlo,elisabeth), a_2(mother,william,diana), a_2(mother,harry,diana), a_1(man,carlo), a_1(man,tom), a_1(man,william), a_1(man,harry), a_1(man,philip), a_1(woman,camilla), a_1(woman,laura), a_1(woman,elisabeth)}Anchor(POST) Finally, we send the content of tmp2 to DLT postparser module as following:
> ./dlt -postparsing <tmp2
and we obtain the following:
{married(carlo,camilla), children(carlo,william), children(carlo,harry), children(camilla,tom), children(camilla,laura), engaged(tom,sara), engaged(laura,harryT), father(carlo,philip), father(camilla,bruce), mother(carlo,elisabeth), mother(william,diana), mother(harry,diana), man(carlo), man(tom), man(william), man(harry), man(philip), woman(camilla), woman(laura), woman(elisabeth)}the same output that we would have obtained launching the command:
> ./dlt family
Indeed, performing the whole process launching directly DLT as above, the only model is:
{married(carlo,camilla), children(carlo,william), children(carlo,harry), children(camilla,tom), children(camilla,laura), engaged(tom,sara), engaged(laura,harryT), man(carlo), man(tom), man(william), man(harry), man(philip), woman(camilla),woman(laura), woman(elisabeth), father(carlo,philip), father(camilla,bruce), mother(carlo,elisabeth), mother(william,diana), mother(harry,diana)}exactly the same of what obtained performing explicitly all the operations constituting the flow.
Section 7.2. Compatibility with other solvers
Anchor(SOLV) DLT can virtually support any solver. At the moment, we can state the compatibility with the syntax of some of them. The following table shows the instructions of the solvers which dlt natively recognizes.
| SOLVER | WHAT'S SUPPORTED | SYNTAX | 
| any solver | ASITIS | ASITIS%* Body *% BR ASITIS's Body is sent verbatim to solver | 
| dlv | Aggregates and weak constraints | see [http://www.dbai.tuwien.ac.at/proj/dlv/man/ DLV - User Manual ] | 
| dlv+odbc | Not tested yet | see [http://www.dbai.tuwien.ac.at/proj/dlv/man/ DLV - User Manual ] | 
| dlvex | external built-ins | see [http://www.mat.unical.it/ianni/wiki/dlvex?action=AttachFile&do=get&target=usermanual.html dlvex - User Manual ] | 
| dlvhex | external atoms, namespace directive | see [http://con.fusion.at/dlvhex/ dlvhex - User Manual ] | 
| smodels | Supports normal programs with function symbols | see [http://www.tcs.hut.fi/Software/smodels/ smodels - User Manual ] | 
Note: All the SOLVER options must be passed using the option -solverparams={}.
Section 7.3. Notes on using DLV-HEX with DLT
DLV-HEX has higher order reasoning built-in, so it is not necessary to invoke DLT with the option -higherorder switched on. Also, DLV-HEX supports natively DLT. DLT can be used with DLV-HEX with different command line combinations such as
dlt -silent -preparsing program.dlt | dlvhex --silent -- | dlt -postparsing
or
dlvhex --dlt program.dlt
makes dlvhex to invoke dlt as a preparser
