hy/docs/extra/anaphoric.rst

240 lines
5.0 KiB
ReStructuredText
Raw Normal View History

================
2013-11-30 01:59:20 +01:00
Anaphoric Macros
================
.. versionadded:: 0.9.12
2013-11-30 01:59:20 +01:00
The anaphoric macros module makes functional programming in Hy very
concise and easy to read.
An anaphoric macro is a type of programming macro that
deliberately captures some form supplied to the macro which may be
referred to by an anaphor (an expression referring to another).
2016-09-04 22:35:46 +02:00
-- Wikipedia (https://en.wikipedia.org/wiki/Anaphoric_macro)
2013-11-30 01:59:20 +01:00
To use these macros you need to require the ``hy.extra.anaphoric`` module like so:
``(require [hy.extra.anaphoric [*]])``
2013-12-26 02:28:32 +01:00
These macros are implemented by replacing any use of the designated
anaphoric symbols (``it``, in most cases) with a gensym. Consequently,
it's unwise to nest these macros, or to use an affected symbol as
something other than a variable name, as in ``(print "My favorite
Stephen King book is" 'it)``.
2013-12-26 02:28:32 +01:00
.. _ap-if:
ap-if
=====
2013-12-26 02:28:32 +01:00
Usage: ``(ap-if (foo) (print it))``
Evaluates the first form for truthiness, and bind it to ``it`` in both the
true and false branches.
2013-12-26 02:28:32 +01:00
2013-11-30 01:59:20 +01:00
.. _ap-each:
ap-each
=======
2013-11-30 01:59:20 +01:00
Usage: ``(ap-each [1 2 3 4 5] (print it))``
Evaluate the form for each element in the list for side-effects.
.. _ap-each-while:
ap-each-while
=============
Usage: ``(ap-each-while list pred body)``
Evaluate the form for each element where the predicate form returns
``True``.
2013-11-30 01:59:20 +01:00
.. code-block:: hy
2013-11-30 01:59:20 +01:00
=> (ap-each-while [1 2 3 4 5 6] (< it 4) (print it))
1
2
3
.. _ap-map:
ap-map
======
Usage: ``(ap-map form list)``
The anaphoric form of map works just like regular map except that
instead of a function object it takes a Hy form. The special name
2013-11-30 01:59:20 +01:00
``it`` is bound to the current object from the list in the iteration.
.. code-block:: hy
2013-11-30 01:59:20 +01:00
=> (list (ap-map (* it 2) [1 2 3]))
[2, 4, 6]
.. _ap-map-when:
ap-map-when
===========
Usage: ``(ap-map-when predfn rep list)``
Evaluate a mapping over the list using a predicate function to
determin when to apply the form.
.. code-block:: hy
2013-11-30 01:59:20 +01:00
=> (list (ap-map-when odd? (* it 2) [1 2 3 4]))
[2, 2, 6, 4]
=> (list (ap-map-when even? (* it 2) [1 2 3 4]))
[1, 4, 3, 8]
.. _ap-filter:
ap-filter
=========
Usage: ``(ap-filter form list)``
As with ``ap-map`` we take a special form instead of a function to
filter the elements of the list. The special name ``it`` is bound to
the current element in the iteration.
.. code-block:: hy
2013-11-30 01:59:20 +01:00
=> (list (ap-filter (> (* it 2) 6) [1 2 3 4 5]))
[4, 5]
.. _ap-reject:
ap-reject
=========
Usage: ``(ap-reject form list)``
This function does the opposite of ``ap-filter``, it rejects the
elements passing the predicate . The special name ``it`` is bound to
the current element in the iteration.
.. code-block:: hy
=> (list (ap-reject (> (* it 2) 6) [1 2 3 4 5]))
[1, 2, 3]
.. _ap-dotimes:
ap-dotimes
==========
Usage ``(ap-dotimes n body)``
This function evaluates the body *n* times, with the special
variable ``it`` bound from *0* to *1-n*. It is useful for side-effects.
.. code-block:: hy
=> (setv n [])
=> (ap-dotimes 3 (.append n it))
=> n
[0, 1, 2]
.. _ap-first:
ap-first
========
Usage ``(ap-first predfn list)``
This function returns the first element that passes the predicate or
``None``, with the special variable ``it`` bound to the current element in
iteration.
.. code-block:: hy
=>(ap-first (> it 5) (range 10))
6
.. _ap-last:
ap-last
========
Usage ``(ap-last predfn list)``
This function returns the last element that passes the predicate or
``None``, with the special variable ``it`` bound to the current element in
iteration.
.. code-block:: hy
=>(ap-last (> it 5) (range 10))
9
.. _ap-reduce:
ap-reduce
=========
Usage ``(ap-reduce form list &optional initial-value)``
This function returns the result of applying form to the first 2
elements in the body and applying the result and the 3rd element
etc. until the list is exhausted. Optionally an initial value can be
supplied so the function will be applied to initial value and the
first element instead. This exposes the element being iterated as
``it`` and the current accumulated value as ``acc``.
.. code-block:: hy
=>(ap-reduce (+ it acc) (range 10))
45
2017-10-23 03:53:05 +02:00
.. _#%
2017-10-23 03:53:05 +02:00
#%
==
2017-10-26 04:46:38 +02:00
Usage ``#% expr``
2017-10-26 04:46:38 +02:00
Makes an expression into a function with an implicit ``%`` parameter list.
2017-10-23 05:36:30 +02:00
2017-10-26 04:46:38 +02:00
A ``%i`` symbol designates the (1-based) *i* th parameter (such as ``%3``).
Only the maximum ``%i`` determines the number of ``%i`` parameters--the
others need not appear in the expression.
2017-10-23 05:36:30 +02:00
``%*`` and ``%**`` name the ``&rest`` and ``&kwargs`` parameters, respectively.
2017-10-26 04:46:38 +02:00
.. code-block:: hy
=> (#%[%1 %6 42 [%2 %3] %* %4] 1 2 3 4 555 6 7 8)
[1, 6, 42, [2, 3], (7, 8), 4]
=> (#% %** :foo 2)
{"foo": 2}
When used on an s-expression,
``#%`` is similar to Clojure's anonymous function literals--``#()``.
.. code-block:: hy
2017-10-26 04:46:38 +02:00
=> (setv add-10 #%(+ 10 %1))
=> (add-10 6)
16
``#%`` determines the parameter list by the presence of a ``%*`` or ``%**``
symbol and by the maximum ``%i`` symbol found *anywhere* in the expression,
so nesting of ``#%`` forms is not recommended.