I'm trying to pick up Lisp as my new language, and I'm having some issues working out how to have parts of a function work on each element of the list passed to it.
For the purpose of learning how to get around this, I am trying to write a fairly basic form of division does not croak when one of the elements of the list are 0 (but instead just returns 0)
(defun divtest (elements)
(dolist (x elements)
(if (zerop x) 0 () )
(/ elements)))))
I try to run this as:
(divtest '(20 2 5))
Which yields:
*** - /: (20 2 5) is not a number
The point of failure seems to be rooted in the fact that i am not "extracting" the elements in the list before passing them to the function (in this case, neither / nor dolist works as intended, as x never evaluates to 0). If I'm right, can someone tell me how to perform this "extraction"?
Note: This question is related to one that I've asked earlier, but as I'm unclear about which part of the previous answer actually allowed it to work as intended with this specific problem i decided to go further into the basics
You can send any data types of argument to a function (string, number, list, dictionary etc.), and it will be treated as the same data type inside the function.
In Python, you can unpack list , tuple , dict (dictionary) and pass its elements to function as arguments by adding * to list or tuple and ** to dictionary when calling function.
What is * in Python function argument? Python has *args which allow us to pass the variable number of non keyword arguments to function. In the function, we should use an asterisk * before the parameter name to pass variable length arguments.
A tuple can also be passed as a single argument to the function. Individual tuples as arguments are just individual variables.
/
takes as arguments one or more numbers, but in your code you're passing it a list - clearly this will not work. The function apply
is your friend here - (apply #'foo a b (list c d e))
is equivalent to (foo a b c d e)
. Note the the arguments to apply
between the function to use and the final list are optional, so (apply #'/ '(20 2 5))
is equivalent to (/ 20 2 5)
.
Also, your attempt at removing zeros will not work. dolist
is evaluating its body for each item in the argument list elements
, but you're not actually doing anything to change the content of elements
(the result of evaluating dolist
s body is not reassigned to the source element as you seem to expect).
The functions remove-if
(and its destructive counterpart, delete-if
) are what you are looking for. The following shows how to use it (it takes lots of optional arguments, which you don't need to worry about for this purpose).
(defun divtest (elements)
(apply #'/ (remove-if #'zerop elements)))
Also note that this won't behave correctly if the elements
list has zero as its first element (assuming I understand what the function's meant to do). So you might instead want something like
(defun divtest (elements)
(apply #'/ (first elements) (remove-if #'zerop (rest elements))))
See the Hyperspec for more details.
Or you can write it like this
(defun divtest (elements)
(if (member 0 elements)
0
(apply #'/ elements)))
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With