| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | /** | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  |  * @file   library/lists.yap | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  |  * @author Bob Welham, Lawrence Byrd, and R. A. O'Keefe. Contributions from Vitor Santos Costa, Jan Wielemaker and others. | 
					
						
							|  |  |  |  * @date   1999 | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  |  * @brief  List Manipulation Predicates | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | % This file has been included as an YAP library by Vitor Santos Costa, 1999 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | :- module(lists, | 
					
						
							|  |  |  | 	  [ | 
					
						
							|  |  |  | 	   append/3, | 
					
						
							|  |  |  | 	   append/2, | 
					
						
							|  |  |  | 	   delete/3, | 
					
						
							|  |  |  | 	   intersection/3, | 
					
						
							|  |  |  | 	   flatten/2, | 
					
						
							|  |  |  | 	   last/2, | 
					
						
							|  |  |  | 	   list_concat/2, | 
					
						
							|  |  |  | 	   max_list/2, | 
					
						
							|  |  |  | 	   list_to_set/2, | 
					
						
							|  |  |  | 	   member/2, | 
					
						
							|  |  |  | 	   memberchk/2, | 
					
						
							|  |  |  | 	   min_list/2, | 
					
						
							|  |  |  | 	   nextto/3, | 
					
						
							|  |  |  | 	   nth/3, | 
					
						
							|  |  |  | 	   nth/4, | 
					
						
							|  |  |  | 	   nth0/3, | 
					
						
							|  |  |  | 	   nth0/4, | 
					
						
							|  |  |  | 	   nth1/3, | 
					
						
							|  |  |  | 	   nth1/4, | 
					
						
							|  |  |  | 	   numlist/3, | 
					
						
							|  |  |  | 	   permutation/2, | 
					
						
							|  |  |  | 	   prefix/2, | 
					
						
							|  |  |  | 	   remove_duplicates/2, | 
					
						
							|  |  |  | 	   reverse/2, | 
					
						
							|  |  |  | 	   same_length/2, | 
					
						
							|  |  |  | 	   select/3, | 
					
						
							|  |  |  | 	   selectchk/3, | 
					
						
							|  |  |  | 	   sublist/2, | 
					
						
							|  |  |  | 	   substitute/4, | 
					
						
							|  |  |  | 	   subtract/3, | 
					
						
							|  |  |  | 	   suffix/2, | 
					
						
							|  |  |  | 	   sum_list/2, | 
					
						
							|  |  |  | 	   sum_list/3, | 
					
						
							|  |  |  | 	   sumlist/2 | 
					
						
							|  |  |  | 	  ]). | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | /** @defgroup lists List Manipulation | 
					
						
							|  |  |  | @ingroup library | 
					
						
							|  |  |  | @{ | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | The following list manipulation routines are available once included | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | with the `use_module(library(lists))` command. | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | /** @pred list_concat(+ _Lists_,? _List_) | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | True when  _Lists_ is a list of lists and  _List_ is the | 
					
						
							|  |  |  | concatenation of  _Lists_. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | /** @pred max_list(? _Numbers_, ? _Max_) | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | True when  _Numbers_ is a list of numbers, and  _Max_ is the maximum. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | /** @pred min_list(? _Numbers_, ? _Min_) | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | True when  _Numbers_ is a list of numbers, and  _Min_ is the minimum. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | /** @pred nth(? _N_, ? _List_, ? _Elem_) | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The same as nth1/3. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | */ | 
					
						
							|  |  |  | /** @pred nth(? _N_, ? _List_, ? _Elem_, ? _Rest_) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Same as `nth1/4`. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | /** @pred nth0(? _N_, ? _List_, ? _Elem_) | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | True when  _Elem_ is the Nth member of  _List_, | 
					
						
							|  |  |  | counting the first as element 0.  (That is, throw away the first | 
					
						
							|  |  |  | N elements and unify  _Elem_ with the next.)  It can only be used to | 
					
						
							|  |  |  | select a particular element given the list and index.  For that | 
					
						
							|  |  |  | task it is more efficient than member/2 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | */ | 
					
						
							|  |  |  | /** @pred nth0(? _N_, ? _List_, ? _Elem_, ? _Rest_) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Unifies  _Elem_ with the Nth element of  _List_, | 
					
						
							|  |  |  | counting from 0, and  _Rest_ with the other elements.  It can be used | 
					
						
							|  |  |  | to select the Nth element of  _List_ (yielding  _Elem_ and  _Rest_), or to | 
					
						
							|  |  |  | insert  _Elem_ before the Nth (counting from 1) element of  _Rest_, when | 
					
						
							|  |  |  | it yields  _List_, e.g. `nth0(2, List, c, [a,b,d,e])` unifies List with | 
					
						
							|  |  |  | `[a,b,c,d,e]`.  `nth/4` is the same except that it counts from 1.  `nth0/4` | 
					
						
							|  |  |  | can be used to insert  _Elem_ after the Nth element of  _Rest_. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | /** @pred nth1(+ _Index_,? _List_,? _Elem_) | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Succeeds when the  _Index_-th element of  _List_ unifies with | 
					
						
							|  |  |  |  _Elem_. Counting starts at 1. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Set environment variable.   _Name_ and  _Value_ should be | 
					
						
							|  |  |  | instantiated to atoms or integers.  The environment variable will be | 
					
						
							|  |  |  | passed to `shell/[0-2]` and can be requested using `getenv/2`. | 
					
						
							|  |  |  | They also influence expand_file_name/2. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | /** @pred nth1(? _N_, ? _List_, ? _Elem_) | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The same as nth0/3, except that it counts from | 
					
						
							|  |  |  | 1, that is `nth(1, [H|_], H)`. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | */ | 
					
						
							|  |  |  | /** @pred nth1(? _N_, ? _List_, ? _Elem_, ? _Rest_) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Unifies  _Elem_ with the Nth element of  _List_, counting from 1, | 
					
						
							|  |  |  | and  _Rest_ with the other elements.  It can be used to select the | 
					
						
							|  |  |  | Nth element of  _List_ (yielding  _Elem_ and  _Rest_), or to | 
					
						
							|  |  |  | insert  _Elem_ before the Nth (counting from 1) element of | 
					
						
							|  |  |  |  _Rest_, when it yields  _List_, e.g. `nth(3, List, c, [a,b,d,e])` unifies List with `[a,b,c,d,e]`.  `nth/4` | 
					
						
							|  |  |  | can be used to insert  _Elem_ after the Nth element of  _Rest_. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | /** @pred numlist(+ _Low_, + _High_, + _List_) | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | If  _Low_ and  _High_ are integers with  _Low_ =< | 
					
						
							|  |  |  |  _High_, unify  _List_ to a list `[Low, Low+1, ...High]`. See | 
					
						
							|  |  |  | also between/3. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | /** @pred permutation(+ _List_,? _Perm_) | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | True when  _List_ and  _Perm_ are permutations of each other. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | /** @pred remove_duplicates(+ _List_, ? _Pruned_) | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Removes duplicated elements from  _List_.  Beware: if the  _List_ has | 
					
						
							|  |  |  | non-ground elements, the result may surprise you. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | /** @pred same_length(? _List1_, ? _List2_) | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | True when  _List1_ and  _List2_ are both lists and have the same number | 
					
						
							|  |  |  | of elements.  No relation between the values of their elements is | 
					
						
							|  |  |  | implied. | 
					
						
							|  |  |  | Modes `same_length(-,+)` and `same_length(+,-)` generate either list given | 
					
						
							|  |  |  | the other; mode `same_length(-,-)` generates two lists of the same length, | 
					
						
							|  |  |  | in which case the arguments will be bound to lists of length 0, 1, 2, ... | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2008-07-22 23:34:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | %%   @pred append(? _Lists_,? _Combined_) | 
					
						
							| 
									
										
										
										
											2008-02-12 17:03:59 +00:00
										 |  |  | % | 
					
						
							|  |  |  | %	Concatenate a list of lists.  Is  true   if  Lists  is a list of | 
					
						
							|  |  |  | %	lists, and List is the concatenation of these lists. | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | % | 
					
						
							| 
									
										
										
										
											2008-02-12 17:03:59 +00:00
										 |  |  | %	@param	ListOfLists must be a list of -possibly- partial lists | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | append(ListOfLists, List) :- | 
					
						
							|  |  |  | %	must_be(list, ListOfLists), | 
					
						
							|  |  |  | 	append_(ListOfLists, List). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | append_([], []). | 
					
						
							| 
									
										
										
										
											2014-04-09 12:39:52 +01:00
										 |  |  | append_([L], L). | 
					
						
							|  |  |  | append_([L1,L2], L) :- | 
					
						
							|  |  |  | 	append(L1,L2,L). | 
					
						
							|  |  |  | append_([L1,L2|[L3|LL]], L) :- | 
					
						
							|  |  |  | 	append(L1,L2,LI), | 
					
						
							|  |  |  | 	append_([LI|[L3|LL]],L). | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | /** @pred last(+ _List_,? _Last_) | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | True when  _List_ is a list and  _Last_ is identical to its last element. | 
					
						
							|  |  |  | d(_, [X], L). | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | last([H|List], Last) :- | 
					
						
							|  |  |  | 	last(List, H, Last). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | last([], Last, Last). | 
					
						
							|  |  |  | last([H|List], _, Last) :- | 
					
						
							|  |  |  | 	last(List, H, Last). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | %   nextto(X, Y, List) | 
					
						
							|  |  |  | %   is true when X and Y appear side-by-side in List.  It could be written as | 
					
						
							| 
									
										
										
										
											2007-02-12 10:33:01 +00:00
										 |  |  | %	nextto(X, Y, List) :- append(_, [X,Y,_], List). | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | %   It may be used to enumerate successive pairs from the list. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | nextto(X,Y, [X,Y|_]). | 
					
						
							|  |  |  | nextto(X,Y, [_|List]) :- | 
					
						
							|  |  |  | 	nextto(X,Y, List). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-08-27 15:27:29 +00:00
										 |  |  | %   nth0(?N, +List, ?Elem) is true when Elem is the Nth member of List, | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | %   counting the first as element 0.  (That is, throw away the first | 
					
						
							|  |  |  | %   N elements and unify Elem with the next.)  It can only be used to | 
					
						
							|  |  |  | %   select a particular element given the list and index.  For that | 
					
						
							|  |  |  | %   task it is more efficient than nmember. | 
					
						
							|  |  |  | %   nth(+N, +List, ?Elem) is the same as nth0, except that it counts from | 
					
						
							|  |  |  | %   1, that is nth(1, [H|_], H). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-08-27 15:27:29 +00:00
										 |  |  | nth0(V, In, Element) :- var(V), !, | 
					
						
							| 
									
										
										
										
											2005-07-05 18:29:25 +00:00
										 |  |  | 	generate_nth(0, V, In, Element). | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | nth0(0, [Head|_], Head) :- !. | 
					
						
							|  |  |  | nth0(N, [_|Tail], Elem) :- | 
					
						
							|  |  |  | 	M is N-1, | 
					
						
							| 
									
										
										
										
											2001-08-27 15:27:29 +00:00
										 |  |  | 	find_nth0(M, Tail, Elem). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | find_nth0(0, [Head|_], Head) :- !. | 
					
						
							|  |  |  | find_nth0(N, [_|Tail], Elem) :- | 
					
						
							|  |  |  | 	M is N-1, | 
					
						
							|  |  |  | 	find_nth0(M, Tail, Elem). | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-22 23:34:50 +00:00
										 |  |  | nth1(V, In, Element) :- var(V), !, | 
					
						
							|  |  |  | 	generate_nth(1, V, In, Element). | 
					
						
							|  |  |  | nth1(1, [Head|_], Head) :- !. | 
					
						
							|  |  |  | nth1(N, [_|Tail], Elem) :- | 
					
						
							|  |  |  | 	nonvar(N), !, | 
					
						
							|  |  |  | 	M is N-1,			% should be succ(M, N) | 
					
						
							|  |  |  | 	find_nth(M, Tail, Elem). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-08-27 15:27:29 +00:00
										 |  |  | nth(V, In, Element) :- var(V), !, | 
					
						
							| 
									
										
										
										
											2005-07-05 18:29:25 +00:00
										 |  |  | 	generate_nth(1, V, In, Element). | 
					
						
							| 
									
										
										
										
											2001-08-27 15:27:29 +00:00
										 |  |  | nth(1, [Head|_], Head) :- !. | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | nth(N, [_|Tail], Elem) :- | 
					
						
							| 
									
										
										
										
											2001-04-27 16:02:43 +00:00
										 |  |  | 	nonvar(N), !, | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	M is N-1,			% should be succ(M, N) | 
					
						
							| 
									
										
										
										
											2001-08-27 15:27:29 +00:00
										 |  |  | 	find_nth(M, Tail, Elem). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | find_nth(1, [Head|_], Head) :- !. | 
					
						
							|  |  |  | find_nth(N, [_|Tail], Elem) :- | 
					
						
							|  |  |  | 	M is N-1, | 
					
						
							|  |  |  | 	find_nth(M, Tail, Elem). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-07-05 18:29:25 +00:00
										 |  |  | generate_nth(I, I, [Head|_], Head). | 
					
						
							| 
									
										
										
										
											2009-10-23 23:54:00 +01:00
										 |  |  | generate_nth(I, IN, [_|List], El) :- | 
					
						
							| 
									
										
										
										
											2005-07-05 18:29:25 +00:00
										 |  |  | 	I1 is I+1, | 
					
						
							|  |  |  | 	generate_nth(I1, IN, List, El). | 
					
						
							| 
									
										
										
										
											2001-08-27 15:27:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | %   nth0(+N, ?List, ?Elem, ?Rest) unifies Elem with the Nth element of List, | 
					
						
							|  |  |  | %   counting from 0, and Rest with the other elements.  It can be used | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | %   to select the Nth element of List (yielding Elem and Rest), or to | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | %   insert Elem before the Nth (counting from 1) element of Rest, when | 
					
						
							|  |  |  | %   it yields List, e.g. nth0(2, List, c, [a,b,d,e]) unifies List with | 
					
						
							|  |  |  | %   [a,b,c,d,e].  nth is the same except that it counts from 1.  nth | 
					
						
							|  |  |  | %   can be used to insert Elem after the Nth element of Rest. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-08-27 15:27:29 +00:00
										 |  |  | nth0(V, In, Element, Tail) :- var(V), !, | 
					
						
							| 
									
										
										
										
											2005-07-05 18:29:25 +00:00
										 |  |  | 	generate_nth(0, V, In, Element, Tail). | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | nth0(0, [Head|Tail], Head, Tail) :- !. | 
					
						
							|  |  |  | nth0(N, [Head|Tail], Elem, [Head|Rest]) :- | 
					
						
							|  |  |  | 	M is N-1, | 
					
						
							|  |  |  | 	nth0(M, Tail, Elem, Rest). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-08-27 15:27:29 +00:00
										 |  |  | find_nth0(0, [Head|Tail], Head, Tail) :- !. | 
					
						
							|  |  |  | find_nth0(N, [Head|Tail], Elem, [Head|Rest]) :- | 
					
						
							|  |  |  | 	M is N-1, | 
					
						
							|  |  |  | 	find_nth0(M, Tail, Elem, Rest). | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-08-27 15:27:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-22 23:34:50 +00:00
										 |  |  | nth1(V, In, Element, Tail) :- var(V), !, | 
					
						
							|  |  |  | 	generate_nth(1, V, In, Element, Tail). | 
					
						
							|  |  |  | nth1(1, [Head|Tail], Head, Tail) :- !. | 
					
						
							|  |  |  | nth1(N, [Head|Tail], Elem, [Head|Rest]) :- | 
					
						
							|  |  |  | 	M is N-1, | 
					
						
							|  |  |  | 	nth1(M, Tail, Elem, Rest). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-08-27 15:27:29 +00:00
										 |  |  | nth(V, In, Element, Tail) :- var(V), !, | 
					
						
							| 
									
										
										
										
											2005-07-05 18:29:25 +00:00
										 |  |  | 	generate_nth(1, V, In, Element, Tail). | 
					
						
							| 
									
										
										
										
											2001-08-27 15:27:29 +00:00
										 |  |  | nth(1, [Head|Tail], Head, Tail) :- !. | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | nth(N, [Head|Tail], Elem, [Head|Rest]) :- | 
					
						
							|  |  |  | 	M is N-1, | 
					
						
							|  |  |  | 	nth(M, Tail, Elem, Rest). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-08-27 15:27:29 +00:00
										 |  |  | find_nth(1, [Head|Tail], Head, Tail) :- !. | 
					
						
							|  |  |  | find_nth(N, [Head|Tail], Elem, [Head|Rest]) :- | 
					
						
							|  |  |  | 	M is N-1, | 
					
						
							|  |  |  | 	find_nth(M, Tail, Elem, Rest). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-07-05 18:29:25 +00:00
										 |  |  | generate_nth(I, I, [Head|Tail], Head, Tail). | 
					
						
							| 
									
										
										
										
											2009-10-23 23:19:55 +01:00
										 |  |  | generate_nth(I, IN, [E|List], El, [E|Tail]) :- | 
					
						
							| 
									
										
										
										
											2005-07-05 18:29:25 +00:00
										 |  |  | 	I1 is I+1, | 
					
						
							|  |  |  | 	generate_nth(I1, IN, List, El, Tail). | 
					
						
							| 
									
										
										
										
											2001-08-27 15:27:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | %   permutation(List, Perm) | 
					
						
							|  |  |  | %   is true when List and Perm are permutations of each other.  Of course, | 
					
						
							|  |  |  | %   if you just want to test that, the best way is to keysort/2 the two | 
					
						
							|  |  |  | %   lists and see if the results are the same.  Or you could use list_to_bag | 
					
						
							|  |  |  | %   (from BagUtl.Pl) to see if they convert to the same bag.  The point of | 
					
						
							|  |  |  | %   perm is to generate permutations.  The arguments may be either way round, | 
					
						
							|  |  |  | %   the only effect will be the order in which the permutations are tried. | 
					
						
							|  |  |  | %   Be careful: this is quite efficient, but the number of permutations of an | 
					
						
							|  |  |  | %   N-element list is N!, even for a 7-element list that is 5040. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | permutation([], []). | 
					
						
							|  |  |  | permutation(List, [First|Perm]) :- | 
					
						
							|  |  |  | 	select(First, List, Rest),	%  tries each List element in turn | 
					
						
							|  |  |  | 	permutation(Rest, Perm). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | % prefix(Part, Whole) iff Part is a leading substring of Whole | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | prefix([], _). | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | prefix([Elem | Rest_of_part], [Elem | Rest_of_whole]) :- | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   prefix(Rest_of_part, Rest_of_whole). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-11-18 17:25:22 +00:00
										 |  |  | %   remove_duplicates(List, Pruned) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | %   removes duplicated elements from List.  Beware: if the List has | 
					
						
							|  |  |  | %   non-ground elements, the result may surprise you. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-19 17:53:19 +00:00
										 |  |  | remove_duplicates([], []). | 
					
						
							|  |  |  | remove_duplicates([Elem|L], [Elem|NL]) :- | 
					
						
							|  |  |  | 	delete(L, Elem, Temp), | 
					
						
							|  |  |  | 	remove_duplicates(Temp, NL). | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | %   reverse(List, Reversed) | 
					
						
							|  |  |  | %   is true when List and Reversed are lists with the same elements | 
					
						
							|  |  |  | %   but in opposite orders.  rev/2 is a synonym for reverse/2. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | reverse(List, Reversed) :- | 
					
						
							|  |  |  | 	reverse(List, [], Reversed). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | reverse([], Reversed, Reversed). | 
					
						
							|  |  |  | reverse([Head|Tail], Sofar, Reversed) :- | 
					
						
							|  |  |  | 	reverse(Tail, [Head|Sofar], Reversed). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | %   same_length(?List1, ?List2) | 
					
						
							|  |  |  | %   is true when List1 and List2 are both lists and have the same number | 
					
						
							|  |  |  | %   of elements.  No relation between the values of their elements is | 
					
						
							|  |  |  | %   implied. | 
					
						
							|  |  |  | %   Modes same_length(-,+) and same_length(+,-) generate either list given | 
					
						
							|  |  |  | %   the other; mode same_length(-,-) generates two lists of the same length, | 
					
						
							|  |  |  | %   in which case the arguments will be bound to lists of length 0, 1, 2, ... | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | same_length([], []). | 
					
						
							|  |  |  | same_length([_|List1], [_|List2]) :- | 
					
						
							|  |  |  | 	same_length(List1, List2). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-22 23:34:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | /** @pred selectchk(? _Element_, ? _List_, ? _Residue_) | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Semi-deterministic selection from a list. Steadfast: defines as | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ~~~~~{.prolog} | 
					
						
							|  |  |  | selectchk(Elem, List, Residue) :- | 
					
						
							|  |  |  |         select(Elem, List, Rest0), !, | 
					
						
							|  |  |  |         Rest = Rest0. | 
					
						
							|  |  |  | ~~~~~ | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2008-07-22 23:34:50 +00:00
										 |  |  | selectchk(Elem, List, Rest) :- | 
					
						
							|  |  |  |         select(Elem, List, Rest0), !, | 
					
						
							|  |  |  |         Rest = Rest0. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | /** @pred select(? _Element_, ? _List_, ? _Residue_) | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | True when  _Set_ is a list,  _Element_ occurs in  _List_, and | 
					
						
							|  |  |  |  _Residue_ is everything in  _List_ except  _Element_ (things | 
					
						
							|  |  |  | stay in the same order). | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | select(Element, [Element|Rest], Rest). | 
					
						
							|  |  |  | select(Element, [Head|Tail], [Head|Rest]) :- | 
					
						
							|  |  |  | 	select(Element, Tail, Rest). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-15 14:19:05 +00:00
										 |  |  | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | %%	sublist(?Sub, +List) is nondet. | 
					
						
							|  |  |  | % | 
					
						
							|  |  |  | %	True if all elements of Sub appear in List in the same order. | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | % | 
					
						
							|  |  |  | %   ALlo, both `append(_,Sublist,S)` and `append(S,_,List)` hold. | 
					
						
							| 
									
										
										
										
											2010-03-15 14:19:05 +00:00
										 |  |  | sublist(L, L). | 
					
						
							|  |  |  | sublist(Sub, [H|T]) :- | 
					
						
							|  |  |  | 	'$sublist1'(T, H, Sub). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | '$sublist1'(Sub, _, Sub). | 
					
						
							|  |  |  | '$sublist1'([H|T], _, Sub) :- | 
					
						
							|  |  |  | 	'$sublist1'(T, H, Sub). | 
					
						
							|  |  |  | '$sublist1'([H|T], X, [X|Sub]) :- | 
					
						
							|  |  |  | 	'$sublist1'(T, H, Sub). | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | %   substitute(X, XList, Y, YList) | 
					
						
							|  |  |  | %   is true when XList and YList only differ in that the elements X in XList | 
					
						
							|  |  |  | %   are replaced by elements Y in the YList. | 
					
						
							|  |  |  | substitute(X, XList, Y, YList) :- | 
					
						
							| 
									
										
										
										
											2008-06-03 22:43:14 +00:00
										 |  |  | 	substitute2(XList, X, Y, YList). | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-06-03 22:43:14 +00:00
										 |  |  | substitute2([], _, _, []). | 
					
						
							|  |  |  | substitute2([X0|XList], X, Y, [Y|YList]) :- | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 	X == X0, !, | 
					
						
							| 
									
										
										
										
											2008-06-03 22:43:14 +00:00
										 |  |  | 	substitute2(XList, X, Y, YList). | 
					
						
							|  |  |  | substitute2([X0|XList], X, Y, [X0|YList]) :- | 
					
						
							|  |  |  | 	substitute2(XList, X, Y, YList). | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | /** @pred suffix(? _Suffix_, ? _List_) | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | Holds when `append(_,Suffix,List)` holds. | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | suffix(Suffix, Suffix). | 
					
						
							|  |  |  | suffix(Suffix, [_|List]) :- | 
					
						
							|  |  |  | 	suffix(Suffix,List). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | /** @pred sumlist(? _Numbers_, ? _Total_) | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | True when  _Numbers_ is a list of integers, and  _Total_ is their | 
					
						
							|  |  |  | sum. The same as sum_list/2, please do use sum_list/2 | 
					
						
							|  |  |  | instead. | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | sumlist(Numbers, Total) :- | 
					
						
							|  |  |  | 	sumlist(Numbers, 0, Total). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | /** @pred sum_list(? _Numbers_, + _SoFar_, ? _Total_) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | True when  _Numbers_ is a list of numbers, and  _Total_ is the sum of their total plus  _SoFar_. | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2008-11-03 16:00:22 +00:00
										 |  |  | sum_list(Numbers, SoFar, Total) :- | 
					
						
							|  |  |  | 	sumlist(Numbers, SoFar, Total). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | /** @pred sum_list(? _Numbers_, ? _Total_) | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | True when  _Numbers_ is a list of numbers, and  _Total_ is their sum. | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2001-04-26 14:44:43 +00:00
										 |  |  | sum_list(Numbers, Total) :- | 
					
						
							|  |  |  | 	sumlist(Numbers, 0, Total). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | sumlist([], Total, Total). | 
					
						
							|  |  |  | sumlist([Head|Tail], Sofar, Total) :- | 
					
						
							|  |  |  | 	Next is Sofar+Head, | 
					
						
							|  |  |  | 	sumlist(Tail, Next, Total). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-17 21:07:41 +00:00
										 |  |  | %   list_concat(Lists, List) | 
					
						
							|  |  |  | %   is true when Lists is a list of lists, and List is the | 
					
						
							|  |  |  | %   concatenation of these lists. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | list_concat([], []). | 
					
						
							|  |  |  | list_concat([H|T], L) :- | 
					
						
							|  |  |  | 	list_concat(H, L, Li), | 
					
						
							|  |  |  | 	list_concat(T, Li). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | list_concat([], L, L). | 
					
						
							|  |  |  | list_concat([H|T], [H|Lf], Li) :- | 
					
						
							|  |  |  | 	list_concat(T, Lf, Li). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | /** @pred flatten(+ _List_, ? _FlattenedList_) | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Flatten a list of lists  _List_ into a single list | 
					
						
							|  |  |  |  _FlattenedList_. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ~~~~~{.prolog} | 
					
						
							|  |  |  | ?- flatten([[1],[2,3],[4,[5,6],7,8]],L). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | L = [1,2,3,4,5,6,7,8] ? ; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | no | 
					
						
							|  |  |  | ~~~~~ | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2002-09-17 16:43:00 +00:00
										 |  |  | flatten(X,Y) :- flatten_list(X,Y,[]). | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-08-25 20:41:09 +01:00
										 |  |  | flatten_list(V) --> {var(V)}, !, [V]. | 
					
						
							| 
									
										
										
										
											2002-09-17 16:43:00 +00:00
										 |  |  | flatten_list([]) --> !. | 
					
						
							|  |  |  | flatten_list([H|T]) --> !, flatten_list(H),flatten_list(T). | 
					
						
							|  |  |  | flatten_list(H) --> [H]. | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-01-29 04:43:14 +00:00
										 |  |  | max_list([H|L],Max) :- | 
					
						
							|  |  |  | 	max_list(L,H,Max). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | max_list([],Max,Max). | 
					
						
							|  |  |  | max_list([H|L],Max0,Max) :- | 
					
						
							|  |  |  | 	( | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | 	  H > Max0 | 
					
						
							| 
									
										
										
										
											2005-01-29 04:43:14 +00:00
										 |  |  | 	-> | 
					
						
							|  |  |  | 	  max_list(L,H,Max) | 
					
						
							|  |  |  | 	; | 
					
						
							|  |  |  | 	  max_list(L,Max0,Max) | 
					
						
							|  |  |  | 	). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | min_list([H|L],Max) :- | 
					
						
							| 
									
										
										
										
											2007-05-15 11:33:51 +00:00
										 |  |  | 	min_list(L,H,Max). | 
					
						
							| 
									
										
										
										
											2005-01-29 04:43:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | min_list([],Max,Max). | 
					
						
							|  |  |  | min_list([H|L],Max0,Max) :- | 
					
						
							|  |  |  | 	( | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | 	  H < Max0 | 
					
						
							| 
									
										
										
										
											2005-01-29 04:43:14 +00:00
										 |  |  | 	-> | 
					
						
							| 
									
										
										
										
											2006-07-16 01:44:17 +00:00
										 |  |  | 	  min_list(L, H, Max) | 
					
						
							| 
									
										
										
										
											2005-01-29 04:43:14 +00:00
										 |  |  | 	; | 
					
						
							| 
									
										
										
										
											2006-07-16 01:44:17 +00:00
										 |  |  | 	  min_list(L, Max0, Max) | 
					
						
							| 
									
										
										
										
											2005-01-29 04:43:14 +00:00
										 |  |  | 	). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-07-22 23:34:50 +00:00
										 |  |  | %%      numlist(+Low, +High, -List) is semidet. | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | % | 
					
						
							| 
									
										
										
										
											2008-07-22 23:34:50 +00:00
										 |  |  | %       List is a list [Low, Low+1, ... High].  Fails if High < Low.% | 
					
						
							|  |  |  | % | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | %       @error type_error(integer, Low) | 
					
						
							| 
									
										
										
										
											2008-07-22 23:34:50 +00:00
										 |  |  | %       @error type_error(integer, High) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | numlist(L, U, Ns) :- | 
					
						
							|  |  |  |         must_be(integer, L), | 
					
						
							|  |  |  |         must_be(integer, U), | 
					
						
							|  |  |  |         L =< U, | 
					
						
							|  |  |  |         numlist_(L, U, Ns). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-25 13:00:09 -05:00
										 |  |  | numlist_(U, U, OUT) :- !, OUT = [U]. | 
					
						
							| 
									
										
										
										
											2008-07-22 23:34:50 +00:00
										 |  |  | numlist_(L, U, [L|Ns]) :- | 
					
						
							|  |  |  |         succ(L, L2), | 
					
						
							|  |  |  |         numlist_(L2, U, Ns). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | /** @pred intersection(+ _Set1_, + _Set2_, + _Set3_) | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Succeeds if  _Set3_ unifies with the intersection of  _Set1_ and | 
					
						
							|  |  |  |  _Set2_.  _Set1_ and  _Set2_ are lists without duplicates. They | 
					
						
							|  |  |  | need not be ordered. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The code was copied from SWI-Prolog's list library. | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-21 00:15:11 +01:00
										 |  |  | % copied from SWI lists library. | 
					
						
							|  |  |  | intersection([], _, []) :- !. | 
					
						
							|  |  |  | intersection([X|T], L, Intersect) :- | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | 	memberchk(X, L), !, | 
					
						
							|  |  |  | 	Intersect = [X|R], | 
					
						
							| 
									
										
										
										
											2010-04-21 00:15:11 +01:00
										 |  |  | 	intersection(T, L, R). | 
					
						
							|  |  |  | intersection([_|T], L, R) :- | 
					
						
							|  |  |  | 	intersection(T, L, R). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-03 21:55:42 -06:00
										 |  |  | %%	subtract(+Set, +Delete, -Result) is det. | 
					
						
							|  |  |  | % | 
					
						
							|  |  |  | %	Delete all elements from `Set' that   occur  in `Delete' (a set) | 
					
						
							|  |  |  | %	and unify the  result  with  `Result'.   Deletion  is  based  on | 
					
						
							|  |  |  | %	unification using memberchk/2. The complexity is |Delete|*|Set|. | 
					
						
							|  |  |  | % | 
					
						
							|  |  |  | %	@see ord_subtract/3. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | subtract([], _, []) :- !. | 
					
						
							|  |  |  | subtract([E|T], D, R) :- | 
					
						
							|  |  |  | 	memberchk(E, D), !, | 
					
						
							|  |  |  | 	subtract(T, D, R). | 
					
						
							|  |  |  | subtract([H|T], D, [H|R]) :- | 
					
						
							|  |  |  | 	subtract(T, D, R). | 
					
						
							| 
									
										
										
										
											2011-06-14 09:01:48 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | %%	list_to_set(+List, ?Set) is det. | 
					
						
							|  |  |  | % | 
					
						
							|  |  |  | %	True when Set has the same element   as  List in the same order. | 
					
						
							|  |  |  | %	The left-most copy of the duplicate  is retained. The complexity | 
					
						
							|  |  |  | %	of this operation is |List|^2. | 
					
						
							|  |  |  | % | 
					
						
							|  |  |  | %	@see sort/2. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | list_to_set(List, Set) :- | 
					
						
							|  |  |  | 	list_to_set_(List, Set0), | 
					
						
							|  |  |  | 	Set = Set0. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | list_to_set_([], R) :- | 
					
						
							|  |  |  | 	close_list(R). | 
					
						
							|  |  |  | list_to_set_([H|T], R) :- | 
					
						
							|  |  |  | 	memberchk(H, R), !, | 
					
						
							|  |  |  | 	list_to_set_(T, R). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | close_list([]) :- !. | 
					
						
							|  |  |  | close_list([_|T]) :- | 
					
						
							|  |  |  | 	close_list(T). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-03 02:06:09 +00:00
										 |  |  | %% @} |