From f31b2ea0327d26f92e5b4460a3f8c5d9103af4b6 Mon Sep 17 00:00:00 2001 From: Paulo Moura Date: Fri, 16 Sep 2016 16:44:34 +0100 Subject: [logtalk/en] Initial version of the Logtalk language tutorial (#2374) * Initial version of the Logtalk language tutorial * Add example of error when sending an unknown message to an object and improve class comment --- logtalk.html.markdown | 520 ++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 520 insertions(+) create mode 100644 logtalk.html.markdown (limited to 'logtalk.html.markdown') diff --git a/logtalk.html.markdown b/logtalk.html.markdown new file mode 100644 index 00000000..2defef61 --- /dev/null +++ b/logtalk.html.markdown @@ -0,0 +1,520 @@ +--- +language: Logtalk +contributors: + - ["Paulo Moura", "http://github.com/pmoura"] +filename: learnlogtalk.lgt +--- + +Logtalk is an object-oriented logic programming language that extends and leverages Prolog with modern code encapsulation and code reuse mechanisms without compromising its declarative programming features. Logtalk is implemented in highly portable code and can use most modern and standards compliant Prolog implementations as a back-end compiler. + +To keep its size reasonable, this tutorial necessarily assumes that the reader have a working knowledge of Prolog and is biased towards describing Logtalk object-oriented features. + +# Syntax + +Logtalk uses standard Prolog syntax with the addition of a few operators and directives for a smooth learning curve and wide portability. One important consequence is that Prolog code can be easily encapsulated in objects with little or no changes. Moreover, Logtalk can transparently interpret most Prolog modules as Logtalk objects. + +Some of the most important operators are: + +* `::/2` - sending a message to an object +* `::/1` - sending a message to _self_ (i.e. to the object that received the message being processed) +* `^^/1` - _super_ call (of an inherited or imported predicate) + +* `<>)/2` infix operator connecting them to the lambda. Some simple examples using library meta-predicates: + +```logtalk +?- {library(metapredicates_loader)}. +yes + +?- meta::map([X,Y]>>(Y is 2*X), [1,2,3], Ys). +Ys = [2,4,6] +yes +``` + +Currying is also supported: + +```logtalk +?- meta::map([X]>>([Y]>>(Y is 2*X)), [1,2,3], Ys). +Ys = [2,4,6] +yes +``` + + +# Macros + +Terms and goals in source files can be _expanded_ at compile time by specifying a _hook object_ that defines term-expansion and goal-expansion rules. For example, consider the following simple object, saved in a `source.lgt` file: + +```logtalk +:- object(source). + + :- public(bar/1). + bar(X) :- foo(X). + + foo(a). foo(b). foo(c). + +:- end_object. +``` + +Let's define an hook object, saved in a `my_macros.lgt` file, that changes clauses and calls to the `foo/1` local predicate: + +```logtalk +:- object(my_macros, + implements(expanding)). % built-in protocol for expanding predicates + + term_expansion(foo(Char), baz(Code)) :- + char_code(Char, Code). % standard built-in predicate + + goal_expansion(foo(X), baz(X)). + +:- end_object. +``` + +After loading the macros file, we can then expand our source file with it: + +```logtalk +?- logtalk_load(my_macros), logtalk_load(source, [hook(my_macros)]). +yes + +?- source::bar(X). +X = 97 ; +X = 98 ; +X = 99 +true +``` + +The Logtalk library provides support for combining hook objects using different workflows (for example, defining a pipeline of expansions). + +# Further information + +Visit the [Logtalk website](http://logtalk.org) for more information. -- cgit v1.2.3 From 13205b60606c1145dcb485c606ebb2c3e461782c Mon Sep 17 00:00:00 2001 From: Adam Bard Date: Mon, 19 Sep 2016 17:46:02 -0700 Subject: Update logtalk.html.markdown Remove logtalk highlighting for now (was crashing middleman) --- logtalk.html.markdown | 62 +++++++++++++++++++++++++-------------------------- 1 file changed, 31 insertions(+), 31 deletions(-) (limited to 'logtalk.html.markdown') diff --git a/logtalk.html.markdown b/logtalk.html.markdown index 2defef61..4f7a207b 100644 --- a/logtalk.html.markdown +++ b/logtalk.html.markdown @@ -31,7 +31,7 @@ Logtalk provides _objects_, _protocols_, and _categories_ as first-class entitie An object encapsulates predicate declarations and definitions. Objects can be created dynamically but are usually static and defined in source files. A single source file can contain any number of entity definitions. A simple object, defining a list member public predicate: -```logtalk +``` :- object(list). :- public(member/2). @@ -46,7 +46,7 @@ An object encapsulates predicate declarations and definitions. Objects can be cr Assuming that the code above for the `list` object is saved in a `list.lgt` file, it can be compiled and loaded using the `logtalk_load/1` built-in predicates or its abbreviation, `{}/1`, with the file path as argument (the extension can be omitted): -```logtalk +``` ?- {list}. yes ``` @@ -55,7 +55,7 @@ yes The `::/2` infix operator is used to send a message to an object. As in Prolog, we can backtrack for alternative solutions: -```logtalk +``` ?- list::member(X, [1,2,3]). X = 1 ; X = 2 ; @@ -65,7 +65,7 @@ yes Encapsulation is enforced. A predicate can be declared _public_, _protected_, or _private_. It can also be _local_ when there is no scope directive for it. For example: -```logtalk +``` :- object(scopes). :- private(bar/0). @@ -78,7 +78,7 @@ Encapsulation is enforced. A predicate can be declared _public_, _protected_, or Assuming the object is saved in a `scopes.lgt` file: -```logtalk +``` ?- {scopes}. yes @@ -91,7 +91,7 @@ error(existence_error(predicate_declaration,local/0),logtalk(scopes::local,user) When the predicate in a message is unknown for the object (the role it plays determines the lookup procedures), we also get an error. For example: -```logtalk +``` ?- scopes::unknown. error(existence_error(predicate_declaration,unknown/0),logtalk(scopes::unknown,user)) ``` @@ -100,7 +100,7 @@ error(existence_error(predicate_declaration,unknown/0),logtalk(scopes::unknown,u Protocols contain predicate declarations that can be implemented by objects and categories: -```logtalk +``` :- protocol(listp). :- public(member/2). @@ -119,7 +119,7 @@ Protocols contain predicate declarations that can be implemented by objects and The scope of the protocol predicates can be restricted using protected or private implementation. For example: -```logtalk +``` :- object(stack, implements(private::listp)). @@ -132,7 +132,7 @@ In fact, all entity relations can be qualified as public (the default), protecte An object without an _instantiation_ or _specialization_ relation with another object plays the role of a prototype. A prototype can _extend_ another object, its parent prototype. -```logtalk +``` % clyde, our prototypical elephant :- object(clyde). @@ -155,7 +155,7 @@ An object without an _instantiation_ or _specialization_ relation with another When answering a message sent to an object playing the role of a prototype, we validate the message and look for an answer first in the prototype itself and, if not found, we delegate to the prototype parents if any: -```logtalk +``` ?- fred::number_of_legs(N). N = 4 yes @@ -167,7 +167,7 @@ yes A message is valid if the corresponding predicate is declared (and the sender is within scope) but it will fail, rather then throwing an error, if the predicate is not defined. This is called the _closed-world assumption_. For example, consider the following object, saved in a `foo.lgt` file: -```logtalk +``` :- object(foo). :- public(bar). @@ -177,7 +177,7 @@ A message is valid if the corresponding predicate is declared (and the sender is Loading the file and trying to call the `bar/0` predicate fails as expected. Note that this is different from calling an _unknown_ predicate, which results in an error: -```logtalk +``` ?- {foo}. yes @@ -194,7 +194,7 @@ Error = error( In order to define objects playing the role of classes and/or instances, an object must have at least an instantiation or a specialization relation with another object. Objects playing the role of meta-classes can be used when we need to see a class also as an instance. We use the following example to also illustrate how to dynamically create new objects at runtime: -```logtalk +``` % a simple, generic, meta-class defining a new/2 predicate for its instances :- object(class, instantiates(class)). @@ -231,7 +231,7 @@ In order to define objects playing the role of classes and/or instances, an obje When answering a message sent to an object playing the role of an instance, we validate the message by starting in its class and going up to its class superclasses if necessary. Assuming that the message is valid, then we look for an answer starting in the instance itself: -```logtalk +``` ?- person::new(Instance, [name(paulo)]). Instance = o1 yes @@ -253,7 +253,7 @@ yes A category is a fine grained unit of code reuse, used to encapsulate a _cohesive_ set of predicate declarations and definitions, implementing a _single_ functionality, that can be imported into any object. A category can thus be seen as the dual concept of a protocol. In the following example, we define categories representing car engines and then import them into car objects: -```logtalk +``` % a protocol describing engine characteristics :- protocol(carenginep). @@ -307,7 +307,7 @@ A category is a fine grained unit of code reuse, used to encapsulate a _cohesive Categories are independently compiled and thus allow importing objects to be updated by simple updating the imported categories without requiring object recompilation. Categories also provide _runtime transparency_: -```logtalk +``` ?- sedan::current_predicate(Predicate). Predicate = reference/1 ; Predicate = capacity/1 ; @@ -322,7 +322,7 @@ yes Categories can be also be used for hot-patching objects. A category can add new predicates to an object and/or replace object predicate definitions. For example, consider the following object: -```logtalk +``` :- object(buggy). :- public(p/0). @@ -333,7 +333,7 @@ Categories can be also be used for hot-patching objects. A category can add new Assume that the object prints the wrong string when sent the message `p/0`: -```logtalk +``` ?- {buggy}. yes @@ -344,7 +344,7 @@ yes If the object source code is not available and we need to fix an application running the object code, we can simply define a category that fixes the buggy predicate: -```logtalk +``` :- category(patch, complements(buggy)). @@ -356,7 +356,7 @@ If the object source code is not available and we need to fix an application run After compiling and loading the category into the running application we will now get: -```logtalk +``` ?- {patch}. yes @@ -369,7 +369,7 @@ yes Objects and categories can be parameterized by using as identifier a compound term instead of an atom. Object and category parameters are _logical variables_ shared with all encapsulated predicates. An example with geometric circles: -```logtalk +``` :- object(circle(_Radius, _Color)). :- public([ @@ -389,7 +389,7 @@ Objects and categories can be parameterized by using as identifier a compound te Parametric objects are used just as any other object, usually providing values for the parameters when sending a message: -```logtalk +``` ?- circle(1.23, blue)::area(Area). Area = 4.75291 yes @@ -397,7 +397,7 @@ yes Parametric objects also provide a simple way of associating a set of predicates with a Prolog predicate. Prolog facts can be interpreted as _parametric object proxies_ when they have the same functor and arity as the identifiers of parametric objects. Handy syntax is provided to for working with proxies. For example, assuming the following clauses for a `circle/2` predicate: -```logtalk +``` circle(1.23, blue). circle(3.71, yellow). circle(0.39, green). @@ -407,7 +407,7 @@ circle(8.32, cyan). With these clauses loaded, we can easily compute for example a list with the areas of all the circles: -```logtalk +``` ?- findall(Area, {circle(_, _)}::area(Area), Areas). Areas = [4.75291, 43.2412, 0.477836, 103.508, 217.468] yes @@ -417,7 +417,7 @@ yes Logtalk supports _event-driven programming_ by allowing defining events and monitors for those events. An event is simply the sending of a message to an object. Interpreting message sending as an atomic activity, a _before_ event and an _after_ event are recognized. Event monitors define event handler predicates, `before/3` and `after/3`, and can query, register, and delete a system-wide event registry that associates events with monitors. For example, a simple tracer for any message being sent using the `::/2` control construct can be defined as: -```logtalk +``` :- object(tracer, implements(monitoring)). % built-in protocol for event handlers @@ -436,7 +436,7 @@ Logtalk supports _event-driven programming_ by allowing defining events and moni Assuming that the `tracer` object and the `list` object defined earlier are compiled and loaded, we can observe the event handlers in action by sending a message: -```logtalk +``` ?- list::member(X, [1,2,3]). call: list <-- member(X, [1,2,3]) from user @@ -453,7 +453,7 @@ yes Logtalk supports lambda expressions. Lambda parameters are represented using a list with the `(>>)/2` infix operator connecting them to the lambda. Some simple examples using library meta-predicates: -```logtalk +``` ?- {library(metapredicates_loader)}. yes @@ -464,7 +464,7 @@ yes Currying is also supported: -```logtalk +``` ?- meta::map([X]>>([Y]>>(Y is 2*X)), [1,2,3], Ys). Ys = [2,4,6] yes @@ -475,7 +475,7 @@ yes Terms and goals in source files can be _expanded_ at compile time by specifying a _hook object_ that defines term-expansion and goal-expansion rules. For example, consider the following simple object, saved in a `source.lgt` file: -```logtalk +``` :- object(source). :- public(bar/1). @@ -488,7 +488,7 @@ Terms and goals in source files can be _expanded_ at compile time by specifying Let's define an hook object, saved in a `my_macros.lgt` file, that changes clauses and calls to the `foo/1` local predicate: -```logtalk +``` :- object(my_macros, implements(expanding)). % built-in protocol for expanding predicates @@ -502,7 +502,7 @@ Let's define an hook object, saved in a `my_macros.lgt` file, that changes claus After loading the macros file, we can then expand our source file with it: -```logtalk +``` ?- logtalk_load(my_macros), logtalk_load(source, [hook(my_macros)]). yes -- cgit v1.2.3 From c08ca5b9c94abb576b56c00206315f3074b9b89d Mon Sep 17 00:00:00 2001 From: Paulo Moura Date: Tue, 18 Oct 2016 16:15:43 +0100 Subject: Fixes and improvements to the Logtalk tutorial (#2470) --- logtalk.html.markdown | 77 +++++++++++++++++++++++++++++++++------------------ 1 file changed, 50 insertions(+), 27 deletions(-) (limited to 'logtalk.html.markdown') diff --git a/logtalk.html.markdown b/logtalk.html.markdown index 4f7a207b..385e38f9 100644 --- a/logtalk.html.markdown +++ b/logtalk.html.markdown @@ -13,14 +13,12 @@ To keep its size reasonable, this tutorial necessarily assumes that the reader h Logtalk uses standard Prolog syntax with the addition of a few operators and directives for a smooth learning curve and wide portability. One important consequence is that Prolog code can be easily encapsulated in objects with little or no changes. Moreover, Logtalk can transparently interpret most Prolog modules as Logtalk objects. -Some of the most important operators are: +The main operators are: * `::/2` - sending a message to an object * `::/1` - sending a message to _self_ (i.e. to the object that received the message being processed) * `^^/1` - _super_ call (of an inherited or imported predicate) -* `<>)/2` infix operator connecting them to the lambda. Some simple examples using library meta-predicates: @@ -470,6 +492,7 @@ Ys = [2,4,6] yes ``` +Lambda free variables can be expressed using the extended syntax `{Free1, ...}/[Parameter1, ...]>>Lambda`. # Macros @@ -486,21 +509,21 @@ Terms and goals in source files can be _expanded_ at compile time by specifying :- end_object. ``` -Let's define an hook object, saved in a `my_macros.lgt` file, that changes clauses and calls to the `foo/1` local predicate: +Assume the following hook object, saved in a `my_macros.lgt` file, that expands clauses and calls to the `foo/1` local predicate: ``` :- object(my_macros, - implements(expanding)). % built-in protocol for expanding predicates + implements(expanding)). % built-in protocol for expanding predicates term_expansion(foo(Char), baz(Code)) :- - char_code(Char, Code). % standard built-in predicate + char_code(Char, Code). % standard built-in predicate goal_expansion(foo(X), baz(X)). :- end_object. ``` -After loading the macros file, we can then expand our source file with it: +After loading the macros file, we can then expand our source file with it using the `hook` compiler flag: ``` ?- logtalk_load(my_macros), logtalk_load(source, [hook(my_macros)]). -- cgit v1.2.3 From eb6b74a568facc1c20e3c347b196fa143a1fa976 Mon Sep 17 00:00:00 2001 From: Samantha McVey Date: Sat, 31 Dec 2016 20:20:26 -0800 Subject: Add highlighting for fortran and logtalk --- logtalk.html.markdown | 62 +++++++++++++++++++++++++-------------------------- 1 file changed, 31 insertions(+), 31 deletions(-) (limited to 'logtalk.html.markdown') diff --git a/logtalk.html.markdown b/logtalk.html.markdown index 385e38f9..d4010f8b 100644 --- a/logtalk.html.markdown +++ b/logtalk.html.markdown @@ -29,7 +29,7 @@ Logtalk provides _objects_, _protocols_, and _categories_ as first-class entitie An object encapsulates predicate declarations and definitions. Objects can be created dynamically but are usually static and defined in source files. A single source file can contain any number of entity definitions. A simple object, defining a list member public predicate: -``` +```logtalk :- object(list). :- public(member/2). @@ -44,7 +44,7 @@ An object encapsulates predicate declarations and definitions. Objects can be cr Assuming that the code above for the `list` object is saved in a `list.lgt` file, it can be compiled and loaded using the `logtalk_load/1` built-in predicate or its abbreviation, `{}/1`, with the file path as argument (the extension can be omitted): -``` +```logtalk ?- {list}. yes ``` @@ -53,7 +53,7 @@ yes The `::/2` infix operator is used to send a message to an object. As in Prolog, we can backtrack for alternative solutions: -``` +```logtalk ?- list::member(X, [1,2,3]). X = 1 ; X = 2 ; @@ -63,7 +63,7 @@ yes Encapsulation is enforced. A predicate can be declared _public_, _protected_, or _private_. It can also be _local_ when there is no scope directive for it. For example: -``` +```logtalk :- object(scopes). :- private(bar/0). @@ -76,7 +76,7 @@ Encapsulation is enforced. A predicate can be declared _public_, _protected_, or Assuming the object is saved in a `scopes.lgt` file: -``` +```logtalk ?- {scopes}. yes @@ -97,7 +97,7 @@ yes When the predicate in a message is unknown for the object (the role it plays determines the lookup procedures), we also get an error. For example: -``` +```logtalk ?- catch(scopes::unknown, Error, true). Error = error( existence_error(predicate_declaration, unknown/0), @@ -112,7 +112,7 @@ A subtle point is that predicate scope directives specify predicate _calling_ se Protocols contain predicate declarations that can be implemented by any number of objects and categories: -``` +```logtalk :- protocol(listp). :- public(member/2). @@ -131,7 +131,7 @@ Protocols contain predicate declarations that can be implemented by any number o The scope of the protocol predicates can be restricted using protected or private implementation. For example: -``` +```logtalk :- object(stack, implements(private::listp)). @@ -144,7 +144,7 @@ In fact, all entity relations (in an entity opening directive) can be qualified An object without an _instantiation_ or _specialization_ relation with another object plays the role of a prototype. A prototype can _extend_ another object, its parent prototype. -``` +```logtalk % clyde, our prototypical elephant :- object(clyde). @@ -167,7 +167,7 @@ An object without an _instantiation_ or _specialization_ relation with another o When answering a message sent to an object playing the role of a prototype, we validate the message and look for an answer first in the prototype itself and, if not found, we delegate to the prototype parents if any: -``` +```logtalk ?- fred::number_of_legs(N). N = 4 yes @@ -179,7 +179,7 @@ yes A message is valid if the corresponding predicate is declared (and the sender is within scope) but it will fail, rather then throwing an error, if the predicate is not defined. This is called the _closed-world assumption_. For example, consider the following object, saved in a `foo.lgt` file: -``` +```logtalk :- object(foo). :- public(bar). @@ -189,7 +189,7 @@ A message is valid if the corresponding predicate is declared (and the sender is Loading the file and trying to call the `bar/0` predicate fails as expected. Note that this is different from calling an _unknown_ predicate, which results in an error: -``` +```logtalk ?- {foo}. yes @@ -208,7 +208,7 @@ yes In order to define objects playing the role of classes and/or instances, an object must have at least an instantiation or a specialization relation with another object. Objects playing the role of meta-classes can be used when we need to see a class also as an instance. We use the following example to also illustrate how to dynamically create new objects at runtime: -``` +```logtalk % a simple, generic, metaclass defining a new/2 predicate for its instances :- object(metaclass, instantiates(metaclass)). @@ -245,7 +245,7 @@ In order to define objects playing the role of classes and/or instances, an obje When answering a message sent to an object playing the role of an instance, we validate the message by starting in its class and going up to its class superclasses if necessary. Assuming that the message is valid, then we look for an answer starting in the instance itself: -``` +```logtalk ?- person::new(Instance, [name(paulo)]). Instance = o1 yes @@ -267,7 +267,7 @@ yes A category is a fine grained unit of code reuse, used to encapsulate a _cohesive_ set of predicate declarations and definitions, implementing a _single_ functionality, that can be imported into any object. A category can thus be seen as the dual concept of a protocol. In the following example, we define categories representing car engines and then import them into car objects: -``` +```logtalk % a protocol describing engine characteristics :- protocol(carenginep). @@ -321,7 +321,7 @@ A category is a fine grained unit of code reuse, used to encapsulate a _cohesive Categories are independently compiled and thus allow importing objects to be updated by simple updating the imported categories without requiring object recompilation. Categories also provide _runtime transparency_. I.e. the category protocol adds to the protocol of the objects importing the category: -``` +```logtalk ?- sedan::current_predicate(Predicate). Predicate = reference/1 ; Predicate = capacity/1 ; @@ -336,7 +336,7 @@ yes Categories can be also be used for hot-patching objects. A category can add new predicates to an object and/or replace object predicate definitions. For example, consider the following object: -``` +```logtalk :- object(buggy). :- public(p/0). @@ -347,7 +347,7 @@ Categories can be also be used for hot-patching objects. A category can add new Assume that the object prints the wrong string when sent the message `p/0`: -``` +```logtalk ?- {buggy}. yes @@ -358,7 +358,7 @@ yes If the object source code is not available and we need to fix an application running the object code, we can simply define a category that fixes the buggy predicate: -``` +```logtalk :- category(patch, complements(buggy)). @@ -370,7 +370,7 @@ If the object source code is not available and we need to fix an application run After compiling and loading the category into the running application we will now get: -``` +```logtalk ?- {patch}. yes @@ -385,7 +385,7 @@ As hot-patching forcefully breaks encapsulation, there is a `complements` compil Objects and categories can be parameterized by using as identifier a compound term instead of an atom. Object and category parameters are _logical variables_ shared with all encapsulated predicates. An example with geometric circles: -``` +```logtalk :- object(circle(_Radius, _Color)). :- public([ @@ -405,7 +405,7 @@ Objects and categories can be parameterized by using as identifier a compound te Parametric objects are used just as any other object, usually providing values for the parameters when sending a message: -``` +```logtalk ?- circle(1.23, blue)::area(Area). Area = 4.75291 yes @@ -413,7 +413,7 @@ yes Parametric objects also provide a simple way of associating a set of predicates with a plain Prolog predicate. Prolog facts can be interpreted as _parametric object proxies_ when they have the same functor and arity as the identifiers of parametric objects. Handy syntax is provided to for working with proxies. For example, assuming the following clauses for a `circle/2` predicate: -``` +```logtalk circle(1.23, blue). circle(3.71, yellow). circle(0.39, green). @@ -423,7 +423,7 @@ circle(8.32, cyan). With these clauses loaded, we can easily compute for example a list with the areas of all the circles: -``` +```logtalk ?- findall(Area, {circle(_, _)}::area(Area), Areas). Areas = [4.75291, 43.2412, 0.477836, 103.508, 217.468] yes @@ -435,7 +435,7 @@ The `{Goal}::Message` construct proves `Goal`, possibly instantiating any variab Logtalk supports _event-driven programming_ by allowing defining events and monitors for those events. An event is simply the sending of a message to an object. Interpreting message sending as an atomic activity, a _before_ event and an _after_ event are recognized. Event monitors define event handler predicates, `before/3` and `after/3`, and can query, register, and delete a system-wide event registry that associates events with monitors. For example, a simple tracer for any message being sent using the `::/2` control construct can be defined as: -``` +```logtalk :- object(tracer, implements(monitoring)). % built-in protocol for event handlers @@ -454,7 +454,7 @@ Logtalk supports _event-driven programming_ by allowing defining events and moni Assuming that the `tracer` object and the `list` object defined earlier are compiled and loaded, we can observe the event handlers in action by sending a message: -``` +```logtalk ?- list::member(X, [1,2,3]). call: list <-- member(X, [1,2,3]) from user @@ -475,7 +475,7 @@ Event-driven programming can be seen as a form of _computational reflection_. Bu Logtalk supports lambda expressions. Lambda parameters are represented using a list with the `(>>)/2` infix operator connecting them to the lambda. Some simple examples using library meta-predicates: -``` +```logtalk ?- {library(metapredicates_loader)}. yes @@ -486,7 +486,7 @@ yes Currying is also supported: -``` +```logtalk ?- meta::map([X]>>([Y]>>(Y is 2*X)), [1,2,3], Ys). Ys = [2,4,6] yes @@ -498,7 +498,7 @@ Lambda free variables can be expressed using the extended syntax `{Free1, ...}/[ Terms and goals in source files can be _expanded_ at compile time by specifying a _hook object_ that defines term-expansion and goal-expansion rules. For example, consider the following simple object, saved in a `source.lgt` file: -``` +```logtalk :- object(source). :- public(bar/1). @@ -511,7 +511,7 @@ Terms and goals in source files can be _expanded_ at compile time by specifying Assume the following hook object, saved in a `my_macros.lgt` file, that expands clauses and calls to the `foo/1` local predicate: -``` +```logtalk :- object(my_macros, implements(expanding)). % built-in protocol for expanding predicates @@ -525,7 +525,7 @@ Assume the following hook object, saved in a `my_macros.lgt` file, that expands After loading the macros file, we can then expand our source file with it using the `hook` compiler flag: -``` +```logtalk ?- logtalk_load(my_macros), logtalk_load(source, [hook(my_macros)]). yes -- cgit v1.2.3 From dbecb473cecfeb0914cb104ff4d680f9fef2b31d Mon Sep 17 00:00:00 2001 From: Paulo Moura Date: Sat, 11 Feb 2017 16:48:10 +0000 Subject: Fixes and improvements to the Logtalk tutorial (#2651) --- logtalk.html.markdown | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) (limited to 'logtalk.html.markdown') diff --git a/logtalk.html.markdown b/logtalk.html.markdown index d4010f8b..bff52cae 100644 --- a/logtalk.html.markdown +++ b/logtalk.html.markdown @@ -106,7 +106,7 @@ Error = error( yes ``` -A subtle point is that predicate scope directives specify predicate _calling_ semantics, not _definitions_ semantics. For example, if an object playing the role of a class declares a predicate private, the predicate can be defined in subclasses and instances *but* can only be called in its instances _from_ the class. +A subtle point is that predicate scope directives specify predicate _calling_ semantics, not _definition_ semantics. For example, if an object playing the role of a class declares a predicate private, the predicate can be defined in subclasses and instances *but* can only be called in its instances _from_ the class. # Defining and implementing a protocol -- cgit v1.2.3 From 5d55924bdcce28882eefb62b755d245f26dec38c Mon Sep 17 00:00:00 2001 From: Paulo Moura Date: Tue, 21 Feb 2017 12:00:24 +0000 Subject: [logtalk/en] fix typo in example code (#2669) --- logtalk.html.markdown | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) (limited to 'logtalk.html.markdown') diff --git a/logtalk.html.markdown b/logtalk.html.markdown index bff52cae..5a52bd3c 100644 --- a/logtalk.html.markdown +++ b/logtalk.html.markdown @@ -182,7 +182,7 @@ A message is valid if the corresponding predicate is declared (and the sender is ```logtalk :- object(foo). - :- public(bar). + :- public(bar/0). :- end_object. ``` -- cgit v1.2.3