bug-gnu-emacs
[Top][All Lists]
Advanced

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

bug#70155: 29.3; Several Emacs Lisp list functions accept non-list argum


From: Mattias Engdegård
Subject: bug#70155: 29.3; Several Emacs Lisp list functions accept non-list arguments
Date: Sat, 27 Apr 2024 11:48:52 +0200

27 apr. 2024 kl. 10.22 skrev Eli Zaretskii <eliz@gnu.org>:

> Mattias, Stefan: any comments on this?  Should we document this, or
> should we change the code?

Eli, thank you for bringing this to our attention. I personally don't see much 
that we need to change.
Our priorities in Lisp are, arguably:

1. Well-defined behaviour should work as documented.
2. Undefined behaviour (ie, usage errors) should not cause crashes or 
corruption. This is what we usually call 'safety', or protecting the system's 
own abstractions.
3. Useful error conditions (such as file-not-found) should work as documented 
or at least be handled in a predictable and useful way.
4. The implementation should be efficient.
5. Incorrect usage should be detected and signalled as far as pragmatically 
possible, to help programmers find mistakes in their code.

The low ranking of the last item means that we don't necessarily promise to 
signal an error for every mistake a programmer can make.

However, in this case there are also algebraic motivations for the behaviour:

> From: <tpeplt@gmail.com>
> Date: Tue, 02 Apr 2024 19:15:28 -0400
> 
>> The built-in Emacs Lisp functions ‘last’, ‘nthcdr’, ‘take’,
>> and ‘ntake’ are functions that accept lists as an argument.
>> However, they also accept non-list arguments without
>> signaling an error.

(nth 1 '(a b . c)) does not signal an error nor do we expect it to, so it's 
reasonable that

  (take 2 '(a b . c)) -> (a b)
  (take 1 '(a . b))   -> (a)
  (take 0 'a)         -> ()

do not signal errors either. nthcdr is the complement of [n]take and behaves 
the same way with respect to nonpositive arguments, and in fact has the 
definition

  (nthcdr N L) = (cdr^N L)

for all naturals N, which means that

  (nthcdr 0 'a)       -> a

is correct.

(Of course if you ask me, I'd prefer it if lists were guaranteed to be proper, 
immutable, with the empty list an object distinct from the symbol nil and the 
false boolean value. Maybe next year.)

`last` works in the same way:

  (last '(xN ... x1 . x0) M)
  -> (last '(x{min(N,M)} ... x1 . x0) M)

with the base case

  (last '(xN ... x1 . x0) 0) -> x0

ie,

  (last x0 0) -> x0

for any atom x0.
Finally, [n]butlast are just [n]take with a different numeric argument, but 
here the special case

  (butlast X 0) -> X

is motivated by performance considerations; it's more important to have it go 
in constant time.






reply via email to

[Prev in Thread] Current Thread [Next in Thread]