summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--eval.c1
-rw-r--r--lib.c63
-rw-r--r--lib.h1
-rw-r--r--stdlib/doc-syms.tl5
-rw-r--r--txr.119
5 files changed, 86 insertions, 3 deletions
diff --git a/eval.c b/eval.c
index f32608e2..48045bec 100644
--- a/eval.c
+++ b/eval.c
@@ -7150,6 +7150,7 @@ void eval_init(void)
reg_fun(intern(lit("rfind-if"), user_package), func_n3o(rfind_if, 2));
reg_fun(intern(lit("find-max"), user_package), func_n3o(find_max, 1));
reg_fun(intern(lit("find-min"), user_package), func_n3o(find_min, 1));
+ reg_fun(intern(lit("find-true"), user_package), func_n3o(find_true, 2));
reg_fun(intern(lit("multi-sort"), user_package), func_n3o(multi_sort, 2));
reg_fun(intern(lit("set-diff"), user_package), func_n4o(set_diff, 2));
reg_fun(intern(lit("diff"), user_package), func_n4o(diff, 2));
diff --git a/lib.c b/lib.c
index 987a6b1d..56a548ec 100644
--- a/lib.c
+++ b/lib.c
@@ -10411,6 +10411,69 @@ val find_min(val seq, val testfun, val keyfun)
return find_max(seq, default_arg(testfun, less_f), keyfun);
}
+val find_true(val pred, val seq, val key)
+{
+ val self = lit("find-true");
+ val keyfun = default_arg(key, identity_f);
+ seq_info_t si = seq_info(seq);
+
+ switch (si.kind) {
+ case SEQ_NIL:
+ break;
+ case SEQ_HASHLIKE:
+ {
+ struct hash_iter hi;
+ val cell;
+
+ hash_iter_init(&hi, si.obj, self);
+
+ while ((cell = hash_iter_next(&hi))) {
+ val key = funcall1(keyfun, cell);
+ val res = funcall1(pred, key);
+ if (res)
+ return res;
+ }
+
+ break;
+ }
+ case SEQ_LISTLIKE:
+ {
+ gc_hint(seq);
+
+ for (seq = z(si.obj); seq; seq = cdr(seq)) {
+ val elt = car(seq);
+ val key = funcall1(keyfun, elt);
+ val res = funcall1(pred, key);
+ if (res)
+ return res;
+ }
+
+ break;
+ }
+ case SEQ_VECLIKE:
+ {
+ val vec = si.obj;
+ cnum len = c_fixnum(length(vec), self);
+ cnum i;
+
+ for (i = 0; i < len; i++) {
+ val elt = ref(vec, num_fast(i));
+ val key = funcall1(keyfun, elt);
+ val res = funcall1(pred, key);
+ if (res)
+ return res;
+ }
+
+ break;
+ }
+ case SEQ_NOTSEQ:
+ default:
+ unsup_obj(self, seq);
+ }
+
+ return nil;
+}
+
val find_if(val pred, val seq, val key)
{
val self = lit("find-if");
diff --git a/lib.h b/lib.h
index 512c3462..41181f6c 100644
--- a/lib.h
+++ b/lib.h
@@ -1150,6 +1150,7 @@ val find_if(val pred, val list, val key);
val rfind_if(val pred, val list, val key);
val find_max(val seq, val testfun, val keyfun);
val find_min(val seq, val testfun, val keyfun);
+val find_true(val pred, val list, val keyfun);
val posqual(val obj, val list);
val rposqual(val obj, val list);
val posql(val obj, val list);
diff --git a/stdlib/doc-syms.tl b/stdlib/doc-syms.tl
index 90d8bd07..13361afc 100644
--- a/stdlib/doc-syms.tl
+++ b/stdlib/doc-syms.tl
@@ -729,16 +729,17 @@
("filter-string-tree" "N-00C9EEB0")
("finalize" "N-01230613")
("finally" "D-0079")
- ("find" "N-005431FF")
+ ("find" "N-00C9DFF6")
("find-frame" "N-02B97226")
("find-frames" "N-02B97226")
- ("find-if" "N-005431FF")
+ ("find-if" "N-00C9DFF6")
("find-max" "N-02BB4231")
("find-min" "N-02BB4231")
("find-package" "N-0250826D")
("find-struct-type" "N-01E5EEA7")
("find-symbol" "N-01EA8B50")
("find-symbol-fb" "N-01EA8B50")
+ ("find-true" "N-00C9DFF6")
("first" "N-02D60463")
("fixnum-max" "N-02A6CE24")
("fixnum-min" "N-02A6CE24")
diff --git a/txr.1 b/txr.1
index ce6c7d64..75234425 100644
--- a/txr.1
+++ b/txr.1
@@ -33403,10 +33403,11 @@ is returned.
(partition* "abcd" '(0 1 2 3)) -> nil
.brev
-.coNP Functions @ find and @ find-if
+.coNP Functions @, find @ find-if and @ find-true
.synb
.mets (find < key < sequence >> [ testfun <> [ keyfun ]])
.mets (find-if < predfun >> { sequence | << hash } <> [ keyfun ])
+.mets (find-true < predfun >> { sequence | << hash } <> [ keyfun ])
.syne
.desc
The
@@ -33415,6 +33416,12 @@ and
.code find-if
functions search through a sequence for an item which
matches a key, or satisfies a predicate function, respectively.
+The
+.code find-true
+function is a variant of
+.code find-if
+which returns the value of the predicate function, instead
+of the item.
The
.meta keyfun
@@ -33475,6 +33482,16 @@ of which are the hash values. If the caller doesn't specify a
.meta keyfun
then these cells are taken as their keys.
+The
+.code find-true
+funtion's argument conventions and search semantics are identical to those of
+.codn find-if ,
+but the return value is different. Instead of returning the found item,
+.code find-true
+returns the value which
+.meta predfun
+returned for the found item's key.
+
.coNP Functions @ rfind and @ rfind-if
.synb
.mets (rfind < key < sequence >> [ testfun <> [ keyfun ]])