factor/basis/math/polynomials/polynomials.factor

102 lines
2.3 KiB
Factor
Raw Normal View History

2008-10-03 03:19:03 -04:00
! Copyright (C) 2008 Doug Coleman.
! See http://factorcode.org/license.txt for BSD license.
2010-04-30 09:42:29 -04:00
USING: arrays combinators fry kernel macros make math math.bits
math.order math.vectors sequences splitting vectors ;
2007-09-20 18:09:08 -04:00
IN: math.polynomials
<PRIVATE
: 2pad-head ( p q n -- p q ) [ 0 pad-head ] curry bi@ ;
: 2pad-tail ( p q n -- p q ) [ 0 pad-tail ] curry bi@ ;
: pextend ( p q -- p q ) 2dup [ length ] bi@ max 2pad-tail ;
: pextend-left ( p q -- p q ) 2dup [ length ] bi@ max 2pad-head ;
2008-09-06 18:15:25 -04:00
: unempty ( seq -- seq ) [ { 0 } ] when-empty ;
2008-03-29 21:36:58 -04:00
: 2unempty ( seq seq -- seq seq ) [ unempty ] bi@ ;
2007-09-20 18:09:08 -04:00
PRIVATE>
: powers ( n x -- seq )
<repetition> 1 [ * ] accumulate nip ;
: p= ( p q -- ? ) pextend = ;
2007-09-20 18:09:08 -04:00
: ptrim ( p -- q )
dup length 1 = [ [ zero? ] trim-tail ] unless ;
2007-09-20 18:09:08 -04:00
: 2ptrim ( p q -- p' q' ) [ ptrim ] bi@ ;
: p+ ( p q -- r ) pextend v+ ;
: p- ( p q -- r ) pextend v- ;
2010-04-30 09:42:29 -04:00
ALIAS: n*p n*v
2007-09-20 18:09:08 -04:00
: pextend-conv ( p q -- p' q' )
2010-04-30 09:42:29 -04:00
2dup [ length ] bi@ + 1 - 2pad-tail ;
2007-09-20 18:09:08 -04:00
: p* ( p q -- r )
2010-04-30 09:42:29 -04:00
2unempty pextend-conv
[ drop length [ iota ] keep ]
[ nip <reversed> ]
[ drop ] 2tri
'[ _ _ <slice> _ v* sum ] map reverse ;
2010-04-30 09:42:29 -04:00
: p-sq ( p -- p^2 ) dup p* ; inline
2007-09-20 18:09:08 -04:00
ERROR: negative-power-polynomial p n ;
: (p^) ( p n -- p^n )
make-bits { 1 } [ [ over p* ] when [ p-sq ] dip ] reduce nip ;
: p^ ( p n -- p^n )
dup 0 >=
[ (p^) ]
[ negative-power-polynomial ] if ;
2007-09-20 18:09:08 -04:00
<PRIVATE
: p/mod-setup ( p p -- p p n )
2ptrim
2008-03-29 21:36:58 -04:00
2dup [ length ] bi@ -
2007-09-20 18:09:08 -04:00
dup 1 < [ drop 1 ] when
2009-05-06 00:32:23 -04:00
[ over length + 0 pad-head pextend ] keep 1 + ;
2007-09-20 18:09:08 -04:00
2010-04-30 09:42:29 -04:00
: /-last ( seq1 seq2 -- x ) [ last ] bi@ / ;
2007-09-20 18:09:08 -04:00
: (p/mod) ( p p -- p p )
2007-09-20 18:09:08 -04:00
2dup /-last
2dup , n*p swapd
p- >vector
dup pop* swap rest-slice ;
2007-09-20 18:09:08 -04:00
PRIVATE>
: p/mod ( p q -- z w )
2007-09-20 18:09:08 -04:00
p/mod-setup [ [ (p/mod) ] times ] V{ } make
reverse nip swap 2ptrim pextend ;
<PRIVATE
2007-09-20 18:09:08 -04:00
: (pgcd) ( b a y x -- a d )
2010-04-30 09:42:29 -04:00
dup V{ 0 } p= [
2007-09-20 18:09:08 -04:00
drop nip
] [
2009-01-23 19:20:47 -05:00
[ nip ] [ p/mod ] 2bi
[ pick p* swap [ swapd p- ] dip ] dip (pgcd)
2007-09-20 18:09:08 -04:00
] if ;
PRIVATE>
: pgcd ( p q -- a d )
[ V{ 0 } clone V{ 1 } clone ] 2dip swap (pgcd) [ >array ] bi@ ;
2007-09-20 18:09:08 -04:00
: pdiff ( p -- p' )
dup length v* { 0 } ?head drop ;
2009-05-10 11:41:50 -04:00
: polyval ( x p -- p[x] )
[ length swap powers ] [ nip ] 2bi v. ;
MACRO: polyval* ( p -- )
reverse
[ 1 tail [ \ * swap \ + [ ] 3sequence ] map ]
[ first \ drop swap [ ] 2sequence ] bi
prefix \ cleave [ ] 2sequence ;
2007-09-20 18:09:08 -04:00