335 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
		
		
			
		
	
	
			335 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								:- protocol(symdiffp).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									:- info([
							 | 
						||
| 
								 | 
							
										author is 'Paulo Moura',
							 | 
						||
| 
								 | 
							
										version is 1.0,
							 | 
						||
| 
								 | 
							
										date is 1999/12/29,
							 | 
						||
| 
								 | 
							
										comment is 'Symbolic differentiation and simplification protocol.',
							 | 
						||
| 
								 | 
							
										source is 'Example based on the Clocksin and Mellish Prolog book.']).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									:- public(diff/1).
							 | 
						||
| 
								 | 
							
									:- mode(diff(-expression), one).
							 | 
						||
| 
								 | 
							
									:- info(diff/1, [
							 | 
						||
| 
								 | 
							
										comment is 'Returns the symbolic differentiation of self.',
							 | 
						||
| 
								 | 
							
										argnames is ['Expression']]).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									:- public(simplify/1).
							 | 
						||
| 
								 | 
							
									:- mode(simplify(-expression), one).
							 | 
						||
| 
								 | 
							
									:- info(simplify/1, [
							 | 
						||
| 
								 | 
							
										comment is 'Returns the symbolic simplification of self.',
							 | 
						||
| 
								 | 
							
										argnames is ['Expression']]).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								:- end_protocol.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								:- object(x,
							 | 
						||
| 
								 | 
							
									implements(symdiffp)).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									:- info([
							 | 
						||
| 
								 | 
							
										author is 'Paulo Moura',
							 | 
						||
| 
								 | 
							
										version is 1.0,
							 | 
						||
| 
								 | 
							
										date is 1999/12/29,
							 | 
						||
| 
								 | 
							
										comment is 'Symbolic differentiation and simplification of a variable.',
							 | 
						||
| 
								 | 
							
										source is 'Example based on the Clocksin and Mellish Prolog book.']).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									diff(1).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(x).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								:- end_object.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								:- object(_ + _,
							 | 
						||
| 
								 | 
							
									implements(symdiffp)).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									:- info([
							 | 
						||
| 
								 | 
							
										author is 'Paulo Moura',
							 | 
						||
| 
								 | 
							
										version is 1.0,
							 | 
						||
| 
								 | 
							
										date is 1999/12/29,
							 | 
						||
| 
								 | 
							
										parnames is ['Expression1', 'Expression2'],
							 | 
						||
| 
								 | 
							
										comment is 'Symbolic differentiation and simplification of +/2 expressions.',
							 | 
						||
| 
								 | 
							
										source is 'Example based on the Clocksin and Mellish Prolog book.']).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									diff(Diff) :-
							 | 
						||
| 
								 | 
							
										this(X + Y),
							 | 
						||
| 
								 | 
							
										once(diff(X, Y, Diff)).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									diff(I, J, 0) :-
							 | 
						||
| 
								 | 
							
										integer(I),
							 | 
						||
| 
								 | 
							
										integer(J).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									diff(X, J, DX) :-
							 | 
						||
| 
								 | 
							
										integer(J),
							 | 
						||
| 
								 | 
							
										X::diff(DX).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									diff(I, Y, DY) :-
							 | 
						||
| 
								 | 
							
										integer(I),
							 | 
						||
| 
								 | 
							
										Y::diff(DY).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									diff(X, Y, DX + DY) :-
							 | 
						||
| 
								 | 
							
										X::diff(DX),
							 | 
						||
| 
								 | 
							
										Y::diff(DY).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(S) :-
							 | 
						||
| 
								 | 
							
										this(X + Y),
							 | 
						||
| 
								 | 
							
										once(simplify(X, Y, S)).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(I, J, S) :-
							 | 
						||
| 
								 | 
							
										integer(I),
							 | 
						||
| 
								 | 
							
										integer(J),
							 | 
						||
| 
								 | 
							
										S is I + J.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(X, 0, S) :-
							 | 
						||
| 
								 | 
							
										X::simplify(S).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(0, Y, S) :-
							 | 
						||
| 
								 | 
							
										Y::simplify(S).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(X, J, S + J) :-
							 | 
						||
| 
								 | 
							
										integer(J),
							 | 
						||
| 
								 | 
							
										X::simplify(S).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(I, Y, I + S) :-
							 | 
						||
| 
								 | 
							
										integer(I),
							 | 
						||
| 
								 | 
							
										Y::simplify(S).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(X, Y, S) :-
							 | 
						||
| 
								 | 
							
										X::simplify(SX),
							 | 
						||
| 
								 | 
							
										Y::simplify(SY),
							 | 
						||
| 
								 | 
							
										(X + Y \= SX + SY ->
							 | 
						||
| 
								 | 
							
											(SX + SY)::simplify(S)
							 | 
						||
| 
								 | 
							
											;
							 | 
						||
| 
								 | 
							
											S = SX + SY).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								:- end_object.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								:- object(_ - _,
							 | 
						||
| 
								 | 
							
									implements(symdiffp)).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									:- info([
							 | 
						||
| 
								 | 
							
										author is 'Paulo Moura',
							 | 
						||
| 
								 | 
							
										version is 1.0,
							 | 
						||
| 
								 | 
							
										date is 1999/12/29,
							 | 
						||
| 
								 | 
							
										parnames is ['Expression1', 'Expression2'],
							 | 
						||
| 
								 | 
							
										comment is 'Symbolic differentiation and simplification of -/2 expressions.',
							 | 
						||
| 
								 | 
							
										source is 'Example based on the Clocksin and Mellish Prolog book.']).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									diff(Diff) :-
							 | 
						||
| 
								 | 
							
										this(X - Y),
							 | 
						||
| 
								 | 
							
										once(diff(X, Y, Diff)).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									diff(I, J, 0) :-
							 | 
						||
| 
								 | 
							
										integer(I),
							 | 
						||
| 
								 | 
							
										integer(J).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									diff(X, J, DX) :-
							 | 
						||
| 
								 | 
							
										integer(J),
							 | 
						||
| 
								 | 
							
										X::diff(DX).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									diff(I, Y, DY) :-
							 | 
						||
| 
								 | 
							
										integer(I),
							 | 
						||
| 
								 | 
							
										Y::diff(DY).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									diff(X, Y, DX - DY) :-
							 | 
						||
| 
								 | 
							
										X::diff(DX),
							 | 
						||
| 
								 | 
							
										Y::diff(DY).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(S) :-
							 | 
						||
| 
								 | 
							
										this(X - Y),
							 | 
						||
| 
								 | 
							
										once(simplify(X, Y, S)).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(X, X, 0).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(I, J, S) :-
							 | 
						||
| 
								 | 
							
										integer(I),
							 | 
						||
| 
								 | 
							
										integer(J),
							 | 
						||
| 
								 | 
							
										S is I - J.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(X, 0, S) :-
							 | 
						||
| 
								 | 
							
										X::simplify(S).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(0, Y, S) :-
							 | 
						||
| 
								 | 
							
										Y::simplify(S).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(X, J, S - J) :-
							 | 
						||
| 
								 | 
							
										integer(J),
							 | 
						||
| 
								 | 
							
										X::simplify(S).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(I, Y, I - S) :-
							 | 
						||
| 
								 | 
							
										integer(I),
							 | 
						||
| 
								 | 
							
										Y::simplify(S).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(X, Y, S) :-
							 | 
						||
| 
								 | 
							
										X::simplify(SX),
							 | 
						||
| 
								 | 
							
										Y::simplify(SY),
							 | 
						||
| 
								 | 
							
										(X - Y \= SX - SY ->
							 | 
						||
| 
								 | 
							
											(SX - SY)::simplify(S)
							 | 
						||
| 
								 | 
							
											;
							 | 
						||
| 
								 | 
							
											S = SX - SY).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								:- end_object.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								:- object(_ * _,
							 | 
						||
| 
								 | 
							
									implements(symdiffp)).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									:- info([
							 | 
						||
| 
								 | 
							
										author is 'Paulo Moura',
							 | 
						||
| 
								 | 
							
										version is 1.0,
							 | 
						||
| 
								 | 
							
										date is 1999/12/29,
							 | 
						||
| 
								 | 
							
										parnames is ['Expression1', 'Expression2'],
							 | 
						||
| 
								 | 
							
										comment is 'Symbolic differentiation and simplification of */2 expressions.',
							 | 
						||
| 
								 | 
							
										source is 'Example based on the Clocksin and Mellish Prolog book.']).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									diff(Diff) :-
							 | 
						||
| 
								 | 
							
										this(X * Y),
							 | 
						||
| 
								 | 
							
										once(diff(X, Y, Diff)).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									diff(I, J, 0) :-
							 | 
						||
| 
								 | 
							
										integer(I),
							 | 
						||
| 
								 | 
							
										integer(J).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									diff(0, _, 0).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									diff(_, 0, 0).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									diff(X, J, J * DX) :-
							 | 
						||
| 
								 | 
							
										integer(J),
							 | 
						||
| 
								 | 
							
										X::diff(DX).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									diff(I, Y, I * DY) :-
							 | 
						||
| 
								 | 
							
										integer(I),
							 | 
						||
| 
								 | 
							
										Y::diff(DY).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									diff(X, Y, X * DY + DX * Y) :-
							 | 
						||
| 
								 | 
							
										X::diff(DX),
							 | 
						||
| 
								 | 
							
										Y::diff(DY).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(S) :-
							 | 
						||
| 
								 | 
							
										this(X * Y),
							 | 
						||
| 
								 | 
							
										once(simplify(X, Y, S)).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(I, J, S) :-
							 | 
						||
| 
								 | 
							
										integer(I),
							 | 
						||
| 
								 | 
							
										integer(J),
							 | 
						||
| 
								 | 
							
										S is I * J.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(0, _, 0).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(_, 0, 0).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(1, Y, SY) :-
							 | 
						||
| 
								 | 
							
										Y::simplify(SY).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(X, 1, SX) :-
							 | 
						||
| 
								 | 
							
										X::simplify(SX).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(I, Y, I * SY) :-
							 | 
						||
| 
								 | 
							
										integer(I),
							 | 
						||
| 
								 | 
							
										Y::simplify(SY).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(X, J, J * SX) :-
							 | 
						||
| 
								 | 
							
										integer(J),
							 | 
						||
| 
								 | 
							
										X::simplify(SX).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(X, Y, SX * SY) :-
							 | 
						||
| 
								 | 
							
										X::simplify(SX),
							 | 
						||
| 
								 | 
							
										Y::simplify(SY).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								:- end_object.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								:- object(_ ** _,
							 | 
						||
| 
								 | 
							
									implements(symdiffp)).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									:- info([
							 | 
						||
| 
								 | 
							
										author is 'Paulo Moura',
							 | 
						||
| 
								 | 
							
										version is 1.0,
							 | 
						||
| 
								 | 
							
										date is 1999/12/29,
							 | 
						||
| 
								 | 
							
										parnames is ['Expression', 'Power'],
							 | 
						||
| 
								 | 
							
										comment is 'Symbolic differentiation and simplification of **/2 expressions.',
							 | 
						||
| 
								 | 
							
										source is 'Example based on the Clocksin and Mellish Prolog book.']).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									diff(Diff) :-
							 | 
						||
| 
								 | 
							
										this(X ** Y),
							 | 
						||
| 
								 | 
							
										once(diff(X, Y, Diff)).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									diff(X, Y, Y * X ** Y2 * DX) :-
							 | 
						||
| 
								 | 
							
										integer(Y),
							 | 
						||
| 
								 | 
							
										Y2 is Y - 1,
							 | 
						||
| 
								 | 
							
										X::diff(DX).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									diff(X, Y, Y * X ** Y2 * DX) :-
							 | 
						||
| 
								 | 
							
										Y2 = Y - 1,
							 | 
						||
| 
								 | 
							
										X::diff(DX).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(S) :-
							 | 
						||
| 
								 | 
							
										this(X ** Y),
							 | 
						||
| 
								 | 
							
										once(simplify(X, Y, S)).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(_, 0, 1).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(X, 1, X).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(X, Y, S ** Y) :-
							 | 
						||
| 
								 | 
							
										integer(Y),
							 | 
						||
| 
								 | 
							
										X::simplify(S).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(X, Y, SX ** SY) :-
							 | 
						||
| 
								 | 
							
										X::simplify(SX),
							 | 
						||
| 
								 | 
							
										Y::simplify(SY).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								:- end_object.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								:- object(log(_),
							 | 
						||
| 
								 | 
							
									implements(symdiffp)).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									:- info([
							 | 
						||
| 
								 | 
							
										author is 'Paulo Moura',
							 | 
						||
| 
								 | 
							
										version is 1.0,
							 | 
						||
| 
								 | 
							
										date is 1999/12/29,
							 | 
						||
| 
								 | 
							
										parnames is ['Expression'],
							 | 
						||
| 
								 | 
							
										comment is 'Symbolic differentiation and simplification of log/1 expressions.',
							 | 
						||
| 
								 | 
							
										source is 'Example based on the Clocksin and Mellish Prolog book.']).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									diff(Diff) :-
							 | 
						||
| 
								 | 
							
										this(log(X)),
							 | 
						||
| 
								 | 
							
										once(diff(X, Diff)).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									diff(I, 0) :-
							 | 
						||
| 
								 | 
							
										integer(I).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									diff(X, DX * X ** -1) :-
							 | 
						||
| 
								 | 
							
										X::diff(DX).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(S) :-
							 | 
						||
| 
								 | 
							
										this(log(X)),
							 | 
						||
| 
								 | 
							
										once(simplify(X, S)).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(1, 0).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(I, Log) :-
							 | 
						||
| 
								 | 
							
										integer(I),
							 | 
						||
| 
								 | 
							
										Log is log(I).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									simplify(X, X).
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								:- end_object.
							 |