2013-12-25 20:18:09 +01:00
|
|
|
================
|
2013-11-30 01:59:20 +01:00
|
|
|
Anaphoric Macros
|
2013-12-25 20:18:09 +01:00
|
|
|
================
|
|
|
|
|
|
|
|
.. 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
|
|
|
|
2016-12-26 23:44:59 +01:00
|
|
|
To use these macros you need to require the ``hy.extra.anaphoric`` module like so:
|
2015-10-16 20:28:30 +02:00
|
|
|
|
2016-12-26 23:44:59 +01:00
|
|
|
``(require [hy.extra.anaphoric [*]])``
|
2013-12-26 02:28:32 +01:00
|
|
|
|
|
|
|
.. _ap-if:
|
|
|
|
|
|
|
|
ap-if
|
2015-08-12 02:11:33 +02:00
|
|
|
=====
|
2013-12-26 02:28:32 +01:00
|
|
|
|
|
|
|
Usage: ``(ap-if (foo) (print it))``
|
|
|
|
|
2014-12-07 07:05:52 +01:00
|
|
|
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
|
2015-08-12 02:11:33 +02:00
|
|
|
=======
|
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
|
2014-12-07 07:05:52 +01:00
|
|
|
``True``.
|
2013-11-30 01:59:20 +01:00
|
|
|
|
2014-03-14 14:01:47 +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
|
2014-12-06 21:32:11 +01:00
|
|
|
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.
|
|
|
|
|
2014-03-14 14:01:47 +01:00
|
|
|
.. 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.
|
|
|
|
|
2014-03-14 14:01:47 +01:00
|
|
|
.. 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.
|
|
|
|
|
2014-03-14 14:01:47 +01:00
|
|
|
.. code-block:: hy
|
2013-11-30 01:59:20 +01:00
|
|
|
|
|
|
|
=> (list (ap-filter (> (* it 2) 6) [1 2 3 4 5]))
|
|
|
|
[4, 5]
|
|
|
|
|
|
|
|
|
2013-12-25 20:18:09 +01:00
|
|
|
.. _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.
|
|
|
|
|
2014-03-14 14:01:47 +01:00
|
|
|
.. code-block:: hy
|
2013-12-25 20:18:09 +01:00
|
|
|
|
|
|
|
=> (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.
|
|
|
|
|
2014-03-14 14:01:47 +01:00
|
|
|
.. code-block:: hy
|
2013-12-25 20:18:09 +01:00
|
|
|
|
|
|
|
=> (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.
|
|
|
|
|
2014-03-14 14:01:47 +01:00
|
|
|
.. code-block:: hy
|
2013-12-25 20:18:09 +01:00
|
|
|
|
|
|
|
=>(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.
|
|
|
|
|
2014-03-14 14:01:47 +01:00
|
|
|
.. code-block:: hy
|
2013-12-25 20:18:09 +01:00
|
|
|
|
|
|
|
=>(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``.
|
|
|
|
|
2014-03-14 14:01:47 +01:00
|
|
|
.. code-block:: hy
|
2013-12-25 20:18:09 +01:00
|
|
|
|
|
|
|
=>(ap-reduce (+ it acc) (range 10))
|
|
|
|
45
|
2015-04-26 18:35:46 +02:00
|
|
|
|
|
|
|
|
2017-10-23 03:53:05 +02:00
|
|
|
.. _#%
|
2015-08-12 02:11:33 +02:00
|
|
|
|
2017-10-23 03:53:05 +02:00
|
|
|
#%
|
2015-08-12 02:11:33 +02:00
|
|
|
==
|
|
|
|
|
2017-10-26 04:46:38 +02:00
|
|
|
Usage ``#% expr``
|
2015-08-12 02:11:33 +02:00
|
|
|
|
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.
|
2015-08-12 02:11:33 +02:00
|
|
|
|
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--``#()``.
|
2015-08-12 02:11:33 +02:00
|
|
|
|
|
|
|
.. 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.
|
|
|
|
|