transcendental-lisp/src/function/UserDefinedFunction.java

92 lines
3.0 KiB
Java
Raw Normal View History

2016-12-19 13:05:53 -05:00
package function;
2016-12-07 16:38:26 -05:00
import java.util.ArrayList;
2016-12-19 13:05:53 -05:00
import function.builtin.*;
import sexpression.*;
2016-12-19 13:05:53 -05:00
import table.SymbolTable;
public class UserDefinedFunction extends LispFunction {
2016-12-07 16:38:26 -05:00
private ArgumentValidator argumentValidator;
2016-12-07 16:38:26 -05:00
private String name;
private Cons body;
private Cons lambdaExpression;
2016-12-07 16:38:26 -05:00
private SymbolTable environment;
private ArrayList<String> formalParameters;
2016-12-07 16:38:26 -05:00
/**
* Create a new user-defined function with the specified name, lambda list and body.
2016-12-07 16:38:26 -05:00
*
* @param name
* the name of this user-defined function
2016-12-07 16:38:26 -05:00
* @param lambdaList
* a list of the formal parameters of this user-defined function (MUST BE A PROPER
* LIST)
2016-12-07 16:38:26 -05:00
* @param body
* the body of this user-defined function (MUST BE A PROPER LIST)
2016-12-07 16:38:26 -05:00
*/
public UserDefinedFunction(String name, Cons lambdaList, Cons body) {
2016-12-07 16:38:26 -05:00
this.name = name;
this.body = body;
this.lambdaExpression = new Cons(new Symbol(name), new Cons(lambdaList, body));
2016-12-07 16:38:26 -05:00
this.environment = SETF.getEnvironment();
this.formalParameters = new ArrayList<String>();
2016-12-07 16:38:26 -05:00
for (; lambdaList.consp(); lambdaList = (Cons) lambdaList.getCdr())
this.formalParameters.add(lambdaList.getCar().toString());
2016-12-07 16:38:26 -05:00
this.argumentValidator = new ArgumentValidator(this.name);
this.argumentValidator.setExactNumberOfArguments(this.formalParameters.size());
2016-12-07 16:38:26 -05:00
}
public SExpression call(Cons argList) {
argumentValidator.validate(argList);
2016-12-07 16:38:26 -05:00
// push a new symbol table onto this function's environment (for its
// parameters)
environment = new SymbolTable(environment);
// bind the values of the arguments to the formal parameter names
for (String param : formalParameters) {
2016-12-07 16:38:26 -05:00
SExpression currentArg = argList.getCar();
environment.put(param, currentArg);
argList = (Cons) argList.getCdr();
}
// store the environment of the S-expression that called this function
// (the current environment)
SymbolTable currentEnvironment = SETF.getEnvironment();
// replace the current environment with the environment of this
// function
SETF.setEnvironment(environment);
Cons currentSExpression = body;
SExpression retval = null;
// evaluate all the S-expressions making up this function's body
while (currentSExpression.consp()) {
retval = EVAL.eval(currentSExpression.getCar());
currentSExpression = (Cons) currentSExpression.getCdr();
}
// replace the environment of the S-expression that called this
// function
SETF.setEnvironment(currentEnvironment);
// remove the bindings of the arguments to the formal parameter names
// in the environment of this function
environment = new SymbolTable(environment.getParent());
return retval;
}
public Cons getLambdaExpression() {
return lambdaExpression;
2016-12-07 16:38:26 -05:00
}
}