Re: Functional programming
 
Jon Harrop <jon@ffconsultancy.com> writes:
Functional programming languages are divided into purely functional
(Miranda, Clean, Haskell etc.) and impure (Lisp, Scheme, Erlang, OCaml,
Standard ML, F#, Scala, Nemerle etc.).
let parse_command (state, para, doc as accu) line =
  I wrote a purely functional parser in Java once for fun.
  That is, all assignments are final, thus, there are no
  storage modifications. No side effects used anywhere
  (except in the test code and in the debug code, which 
  does the side-effect of printing.) Code is given below.
  The building scheme of the ?functional? methods 
  of the scanner and parser below is always like:
let x0 = ...
    x1 = ...
    x2 = ...
in ...
  which, in Java, is written as
{ final x0 = ...;
  final x1 = ...;
  final x2 = ...;
  return ...; }
  (When replying to my post, please do not quote my complete
  post, but only those parts you directly refer to.
  Especially, please do not quote the complete source code.)
public class List
{
  // implementation of a dotted pair 
  private final int  car; private final List cdr;
  List( final int car, final List cdr ){ this.car = car; this.cdr = cdr; }
  // That was the actual class ?List?
  // The rest are static methods.
  // support for lists
  private static final boolean DEBUG = false;
  static List cons( final int car, final List cdr )  
  { return new List( car, cdr ); }
  static int  fst(   final List l ){ return l == null ? INVALID : l.car; }
  static List rest(  final List l ){ return l == null ? null : l.cdr;    }
  static int  snd(   final List l ){ return fst( rest( l ));             }
  static int  caddr( final List l ){ return fst( rest( rest( l )));      }
  static List cddr(  final List l ){ return rest( rest( l ));            }
  static List cdddr( final List l ){ return rest( cddr( l ));            }
  static String prints( final List l )
  { return l == null ? "" : "" + name( fst( l ))+ ", " + prints( rest( l )); }
  static List print( final List l )
  { if( DEBUG )System.out.println( "( "+ prints( l ) + " )" ); return l; }
  static List print( final String s, final List l )
  { if( DEBUG )System.out.println( s + "( "+ prints( l ) + " )" ); return l; }
  // main with some test cases
  public static void main( String[] args )
  { System.out.println( "" + evaluate( "(3+4)*2" ) + " [14]." ); 
    System.out.println( "" + evaluate( "2*10" ) + " [20]." ); 
    System.out.println( "" + evaluate( "(9+9)*9+9" ) + " [171]." ); 
    System.out.println( "" + evaluate( "3*(4+5)" )   + " [27]."  ); 
    System.out.println( "" + evaluate( "3*(4*5)" )   + " [60]."  ); 
    System.out.println( "" + evaluate( "3*(4+5)*2" )   + " [54]."  ); 
    System.out.println( "" + evaluate( "3*(4*5)+2" )   + " [62]."  ); 
    System.out.println( "" + evaluate( "1+1+1+1+1+1" )   + " [6]."  ); 
    System.out.println( "" + evaluate( "2*2*2*2*2*2" )   + " [64]."  ); 
    System.out.println( "" + evaluate( "1+1+(2*2*2)+1+1" )   + " [12]."  ); 
    System.out.println( "" + evaluate( "1+1+(1+1+1)+1+1" )   + " [7]."  ); 
    System.out.println( "" + evaluate( "2*2*(2+2+2)*2*2" )   + " [96]."  ); 
    System.out.println( "" + evaluate( "2*2*(2*2*2)*2*2" )   + " [128]."  ); 
    System.out.println( "" + 
      evaluate( "2*2*((1+1)*2*2)*2*2" )   + " [128]."  ); 
    System.out.println( "" + 
      evaluate( "2*2*(2*(1+1)*2)*2*2" )   + " [128]."  ); 
    System.out.println( "" + 
      evaluate( "2*2*(2*2*(1+1))*2*2" )   + " [128]."  ); 
    System.out.println( "" + 
      evaluate( "2*2*(2*(1+1)*(1+1))*2*2" )   + " [128]."  ); 
    System.out.println( "" + 
      evaluate( "2*2*((1+1)*(1+1)*2)*2*2" )   + " [128]."  ); 
    System.out.println( "" + 
      evaluate( "2*2*((1+1)*(1+1)*(1+1))*2*2" )   + " [128]."  ); 
    System.out.println( "" + 
      evaluate( "0+2*2*((1+1)*(1+1+0)*(1+1))*2*2+0" )   + " [128]."  ); 
    System.out.println( "" + 
      evaluate( "10+2*2*((1+1)*(1+1+0)*(1+1))*2*2+10" )   + " [148]."  ); 
    System.out.println( "" + 
      evaluate( "10+2*2*((1+1)*(1+1+0)*(1+1))*2*2+100" )   + 
      " [238]."  ); }
  // the evaluator
  static int evaluate( final String string )
  { return fst( expression( tokenize( string ))); }
  // syntactical analyzer
  static List expression( final List arg )
  { final List tokens   = term( arg );
    final int  term     = fst( tokens );
    final int  operator = snd( tokens );
    final List tail     = cddr( tokens );
    final List tailval  = operator == PLUS ? expression( tail ) : null;
    return tail == null ? cons( term, null ) :
    operator == PLUS    ? cons( term + fst( tailval ), rest( tailval )) :
    operator == R_PAREN ? cons( term, tail ) : null; }
  static List term( final List arg )
  { final List pair    = factor( arg );
    final int  result  = fst( pair );
    final List tokens  = rest( pair );
    final List tail    = rest( tokens );
    final List tailval = factor( tail );
    return tail == null ? cons( result, null ) :
    fst( tokens ) == MUL ? 
    term( cons( result * fst( tailval ), rest( tailval ))) : 
    cons( result, tokens ); }
  static List factor( final List arg )
  { final int  result  = fst( arg );
    final List tokens  = rest( arg );
    return result >= NUMBER  ? cons( result, tokens ) : 
    result == L_PAREN ? expression( tokens ) : null; }
  // support for the lexical analyzer
  static int numlen( final String s )
  { final boolean done    = s.equals( "" );
    final boolean valid   = !done;
    final String  r       = valid ? s.substring( 1, s.length() ): "";
    final char    c       = valid ? s.charAt( 0 ): 0;
    final boolean isdigit = Character.isDigit( c );
    return done ? 0 : isdigit ? 1 + numlen( r ) : 0; }
  static int max( final int i, final int j )
  { return i > j ? i : j; }
  // The lexical analyzer
  static final int NUMBER   =  0;
  static final int INVALID  = -1;
  static final int PLUS     = -2;
  static final int MUL      = -3;
  static final int L_PAREN  = -4;
  static final int R_PAREN  = -5;
  static String name( final int l )
  { return l >= NUMBER ? "" + l  : l == INVALID ? "INVALID" : 
    l == PLUS ? "PLUS" : l == MUL ? "MUL" : l == L_PAREN ? "L_PAREN" : 
    l == R_PAREN ? "R_PAREN" : "(UNKNOWN)"; }
  static List tokenize( final String s )
  { final boolean done    = s.equals( "" );
    final boolean valid   = !done;
    final String  r       = 
    valid ? s.substring( max( 1, numlen( s )), s.length() ): "";
    final char    c       = valid ? s.charAt( 0 ): 0;
    final boolean isdigit = Character.isDigit( c );
    final int     d       = isdigit ? 
    Integer.valueOf( s.substring( 0, numlen( s ))).intValue() : 0;
    return done ? null : cons
    ( ( isdigit ? d : c == '+'  ? PLUS : c == '*' ? MUL : 
        c == '(' ? L_PAREN : c == ')' ? R_PAREN   : INVALID ), 
      tokenize( r )); }}
/*
171 [171].
27 [27].
60 [60].
54 [54].
62 [62].
6 [6].
64 [64].
12 [12].
7 [7].
96 [96].
128 [128].
128 [128].
128 [128].
128 [128].
128 [128].
128 [128].
128 [128].
128 [128].
148 [148].
238 [238].
*/
  When replying to my post, please do not quote my complete
  post, but only those parts you directly refer to.
  Especially, please do not quote the complete source code.