newLISP Fan Club

Forum => Anything else we might add? => Topic started by: ssqq on July 16, 2016, 03:21:41 AM

Title: primitive function empty? could not accept array
Post by: ssqq on July 16, 2016, 03:21:41 AM

> (empty? (array 2 '(1 2)))
ERR: list or string expected : (1 2)


I think any check function could accept any type of data.
Title: Re: primitive function empty? could not accept array
Post by: rrq on July 18, 2016, 03:20:15 PM
I'd agree that empty? should be defined for arrays. I don't use arrays much in newlisp, but in most other languages the empty array test (i.e., size==0) pops up very often.
Title: Re: primitive function empty? could not accept array
Post by: Lutz on July 19, 2016, 08:39:46 PM
There are no empty arrays ;-)
Title: Re: primitive function empty? could not accept array
Post by: ssqq on July 20, 2016, 09:40:04 PM
I think array should have empty array, Just like other language.



I think "Not empty array" is a design error.
Title: Re: primitive function empty? could not accept array
Post by: TedWalther on July 20, 2016, 09:43:59 PM
By default, we perceive an array as a special case of a list; so we expect it to act like a like in pretty much every way.  So, even if an array is never empty, it makes sense for every function that acts on a list, to also act on an array and do the sensible thing.  Even though under the hood, an array and list are very different.  Principle of Least Surprise.
Title: Re: primitive function empty? could not accept array
Post by: newBert on July 21, 2016, 02:07:09 AM
Quote from: "TedWalther"By default, we perceive an array as a special case of a list; so we expect it to act like a like in pretty much every way.


Yes, but we can't effectively create an empty array :
newLISP v.10.7.1 32-bit on Linux IPv4/6 UTF-8 libffi, options: newlisp -h

> (array 0)

ERR: wrong dimensions in function array : 0
>

If we'd like to consider an array as a special case of a list, we must think it's above all a non-empty list.

So, the use of empty? with an array seems to be  an incoherence, or even a nonsense... and it's not so bad that an error message reminds us of it :)
Title: Re: primitive function empty? could not accept array
Post by: rrq on July 21, 2016, 04:34:55 AM
Again, I haven't really used arrays in newlisp, so I don't mind the lack of empty arrays once told :-)



But upon review it seems that ideally the functions like find and ref (with friends) should handle arrays as if lists. Intuitively that's needed in order to take some advantage of the (as advertised) faster element access by index.



And (while loitering at the wishing well), maybe there could also be a binary search lookup for sorted array, to take even more advantage of the speedier indexing.
Title: Re: primitive function empty? could not accept array
Post by: rickyboy on July 21, 2016, 05:38:54 AM
Quote from: "Lutz"There are no empty arrays ;-)

+1 for pithy, direct answer.  :)



... that unfortunately seemed to be ignored for a bit. :(
Title: Re: primitive function empty? could not accept array
Post by: TedWalther on July 21, 2016, 11:30:58 AM
Since you can convert an array into a list, it is easy to overlook that there is no "list to array" function.  But array creation does let you use a list as an "initializer".  '() is a list... intuitively, until you try and fail, you wouldn't guess that the empty list isn't allowed as an array initializer.



The array-list function sets up a correspondance in the mind.  This creates the expectation that arrays are just another type or format of list; so all the list functions should apply to it.  Contexts are different enough from lists that you learn to treat them as their own data type from the beginning.  Arrays are similar enough to lists, that it is jarring when they don't act like lists.
Title: Re: primitive function empty? could not accept array
Post by: ssqq on July 22, 2016, 08:52:41 PM
I need make array, hash, list use newLISP. SO i do like follows:

(define (is-blank-array x) (= x '()))

(define (is-list x) (list? x) (and (= (first x) "list") (list? (last x))))

(define (is-blank-hash x) (= x '(())))

(define (is-hash @x)
  (and (list? @x) (for-all is-pair @x)))

(define (is-pair @x)
  (and (list? @x) (= 2 (length @x)) (is-key (@x 0))))

(define (is-key @x) (or (string? @x) (number? @x)))
Title: Re: primitive function empty? could not accept array
Post by: rickyboy on July 23, 2016, 11:17:37 AM
Hi ssqq,



Let's take this one by one.


Quote from: "ssqq"I need make array, hash, list use newLISP. SO i do like follows:

(define (is-blank-array x) (= x '()))

The name `is-blank-array` belies what the procedure is actually doing which is checking that its input is an empty list.  In other words, it has nothing to do with arrays.  Also, recall that Lutz said that there are no empty arrays in newLISP.


Quote from: "ssqq"
(define (is-list x) (list? x) (and (= (first x) "list") (list? (last x))))

The `(list? x)` expression is superfluous (i.e., does nothing).  Moreover, this function doesn't do what its name seems to intend.  Witness:


> (is-list (list 1 2 3))
nil

This is due to the fact that lists in newLISP are not implemented as "sequences" where the first item is the string "list".  That's just wrong.



This doesn't even work.


> (is-list (list "list" 1 2 3))
nil

because `(list? (last x))` fails.


> (last (list "list" 1 2 3))
3

Lists like this will work:


> (is-list (list "list" 1 2 (list 3)))
true

but I think you intended a more general test.


Quote from: "ssqq"
(define (is-blank-hash x) (= x '(())))

(define (is-hash @x)
  (and (list? @x) (for-all is-pair @x)))

This doesn't "describe" a "hash" (i.e., hash table/map), but an association list.  Lispers have called this "alist" for short. So, your function names might be better expressed as `empty-alist?` and `alist?`.


Quote from: "ssqq"
(define (is-pair @x)
  (and (list? @x) (= 2 (length @x)) (is-key (@x 0))))

(define (is-key @x) (or (string? @x) (number? @x)))

In an alist, the keys can usually be more than just strings or numbers -- they can be whatever data in the language is comparable under certain equality operators; these operators will be the ones used in your alist lookup functions, by the way.  In hash tables/maps, keys are anything that is "hashable" (i.e., any data for which the hash function can produced a hash).
Title: Re: primitive function empty? could not accept array
Post by: rickyboy on July 23, 2016, 11:31:40 AM
Quote from: "ssqq"I think array should have empty array, Just like other language.



I think "Not empty array" is a design error.

I disagree.  Why?  Do you have a good use case?
Title: Re: primitive function empty? could not accept array
Post by: rickyboy on July 23, 2016, 11:38:40 AM
Hello Ted!


Quote from: "TedWalther"The array-list function sets up a correspondance in the mind.  This creates the expectation that arrays are just another type or format of list; so all the list functions should apply to it.

The subject of the action is incorrect here IMO.  It's not the `array-list` function that "sets up a correspondence"; it's your own mind that is doing this. My mind is not making the same correspondence. :D


Quote from: "TedWalther"Contexts are different enough from lists that you learn to treat them as their own data type from the beginning.  Arrays are similar enough to lists, that it is jarring when they don't act like lists.

In my mind, arrays are *different* enough from lists ... :)
Title: Re: primitive function empty? could not accept array
Post by: ssqq on July 24, 2016, 03:15:22 AM
in newLISP, no "true" hash, is just assoc-list. I just want simulate Hash, Array use newlisp.



Just like no Hash, List and symbol in C, but Lautz could write newLISP use C.



I write Spp language implement with newLISP, it have Array, Hash, List, Rule:


> (my @array [])
[]
> (@array << :a)
[:a]
> (my %hash {1 : 2, 3 : 4})
{1 ; 2, 3 : 4}
> (%hash 1)
2
> (my @lst :(1 2 3))
(1 2 3)