Assessing Parson's problems and proofs
General remarks
This document explains how to assess Parson's problems. There are three sets of tools available in STACK.
- When a tree is used to represent the structure of a teacher's proof, we can automatically create all alternative proofs. Then we can establish whether a student's proof is one of these. If not we automatically provide feedback explaining how to modify the student's proof to transform it to the closest correct proof. E.g. swap two lines. Generating alternatives and feedback is fully automated. However, we assume a proof can be fully represented as a tree.
- Generally we have a directed graph representing dependencies of lines in a proof. We have general tools for recording these dependencies, establishing if a student's proof satisfies them, and if not generating automatic feedback based on dependencies specified in a dependency graph. The teacher has to create a dependency graph for each proof, but feedback is automatic.
- Teachers can encode specific dependency checks, and provide specific feedback when a student's proof meets/fails to meet certain criteria. E.g. "This line uses the variable but you have not yet introduced it in your proof".
Each option above independent, and any/all could be used with a particular proof.
1. Automatic assessment of proof defined as a tree
The Damerau-Levenshtein distance is a metric for measuring the difference between two strings. Informally, this is the edit distance measuring the minimum number of single-character edits (insertions, deletions, transition or substitutions) required to change one string into the other. STACK uses this metric to assess answers which are text strings. The problem of assessing Parson's problems is very similar.
- We want to establish the distance between the student's proof and a teacher's proof, and identify the closest proof from a list deemed acceptable by the teacher.
- We want to automatically provide feedback detailing which edits will transform the student's proof into a "correct" proof, e.g. "Swap these two lines", "Insert a line here".
Rather than a string of characters, we apply the metric to a list of keys
tags in the proof_steps
list, as defined in the CAS library for representing text-based proofs. Further, by tracking the steps in the algorithm we can provide automatic feedback about which edits are required to transform one list into another.
We assume that the teacher's answer is ta
is expressed using "proof construction functions" e.g.
ta:proof_iff("A","B");
where the keys are part of a proof_steps
list.
Define tal
to be the list of teacher's answers. This can be created automatically, e.g. tal:proof_alternatives(ta)
or a teacher can define a bespoke list of proofs.
The assessment function proof_assessment(sa, tal)
takes a proof provided by a student and a list of acceptable proofs and does the following.
- All proofs are flattened to lists of keys. The Damerau-Levenshtein distance only applies to lists, not trees.
- Calculate the distance between the student's proof and each of the teacher's proofs (in
tal
), and identify the closest proof. - Return a list
[d, [edits]]
, whered
is the shortest distance andedits
is the list of edits (if any) needed for each step.
To provide feedback the list of edits can be sent directly to a display function.
To use this in practice define the following feedback variables.
sa:parsons_decode(ans1);
[pd, saa]:proof_assessment(sa, proof_alternatives(ta));
The variable pd
now contains the edit distance from the student's proof to the closest teacher's. The teacher can decide if this is close enough (zero, of course means exact match) and whether to display the feedback. For example, use ATAlgEquiv(pd,0)
to check if the student's proof matches one of the teacher's proofs. If the number of edits required is at least the length of the teacher's proof then everything needs editing and there is little point displaying feedback!
To display feedback use {@proof_assessment_display(saa, proof_steps)@}
in a PRT feedback (or other castext).
2. Proof assessment when steps within separate sub-hypotheses can be interleaved
Proofs often contain sub-proofs, in blocks. In our design for the representation of proofs as tree structures, a teacher might decide one block precedes another, or that the order of these blocks is interchangeable. In some situations it is possible to interleave parts of two or more blocks without invalidating the proof from a strictly logical point of view. For example, in the teacher's proof we might have separate blocks such as
Assume n is odd.
There exists j such that n=2j-1.
...
Assume m is even.
There exists k such that m=2k.
...
However, a student might choose to interleave these clauses without a loss of strict logical correctness.
Assume n is odd.
Assume m is even.
This section explains general tools to assess proofs which allows steps to be interleaved.
Consider the following proof.
H1. Assume that \(3 \cdot 2^{172} + 1\) is a perfect square.
S2. There is a positive integer \(k\) such that \(3 \cdot 2^{172} + 1 = k^2\).
S3. Since \(3 \cdot 2^{172} + 1 > 2^{172} = (2^{86})^2 > 172^2\), we have \(k > 172\).
S4. We have \(k^2 = 3 \cdot 2^{172} + 1 < 3 \cdot 2^{172} + 173\).
S5. Also, \(3 \cdot 2^{172} + 173 = (3 \cdot 2^{172} + 1) + 172 < k^2 + k\). Further, \(k^2 + k < (k + 1)^2\).
C6. Since \(3 \cdot 2^{172} + 173\) is strictly between two successive squares \(k^2\) and \((k + 1)^2\), it cannot be a perfect square.
In this proof has the following dependency structure.
S1 | S2 / | S3 | | S4 S5 | | / C6
In particular, we have the following options for "correct" answers.
[S1, S2, S3, S4, S5, C6]
[S1, S2, S4, S3, S5, C6]
[S1, S2, S3, S5, S4, C6]
This graph dependency cannot be inferred from a proof tree type structure. While it is considerably simpler for a teacher to write a proof as a tree, interleaved steps in a student's proof may not be wrong.
Therefore we provide a function which takes (a) the student's list (of Parsons keys), and (b) a representation of the directed graph and uses the teacher's graph to conduct assessment.
To author the graph we create lists of key-lists in Maxima as follows. This is a list of lists, representing the edges of the graph..
tdag: [
["S1", "S2"],
["S2", "S3", "S5"],
["S2", "S4"],
["S4", "C6"],
["S5", "C6"]
];
- Each key used in the teacher's proof must occur in the student's list. Missing keys (i.e. steps) will be flagged.
- Only keys used in the teacher's proof should occur in the student's list. Extra keys (i.e. steps) will be flagged.
- For each list, we check that the keys occur in the specified order in the student's proof. E.g.
- in the first list
["S1", "S2"]
we check that"S1"
comes before"S2"
in the student's proof. - in the second list
["S2", "S3", "S5"]
we check that"S2"
comes before"S3"
, and"S3"
comes before"S4"
. Note, by allowing lists with more than two keys we reduce the complexity of expressing long chains of steps. - We do not specify that nothing can be between steps. That's a separate property which this test does not establish. (Separate tools are needed to establish, e.g. "No other steps should occur between X and Y".)
Hence, we could also write this graph as follows.
tdag: [
["S1", "S2", "S3", "S5", "C6"],
["S2", "S4", "C6"]
];
Writing a graph is considerably more complex for a teacher than using the proof()
functions, but of course it gives the teacher more flexibility with what to accept. The two approaches can be combined. If a student's answer is found to be incorrect, then we can still establish the closest distance to a proof the teacher considers to be correct to give automatic feedback on how a student should change their proof.
STACK provides functions to support assessment where the teacher specified a dependency graph.
saprob: proof_assessment_dag(sa, tdag)
takes (1) the student's answer (a list of tags) and (2) a graph specified as lists of key-lists and returns a list of "problems". Problems with a student's graph can take three forms.
proof_step_missing({"C6"})
indicates the set of steps{"C6"}
is missing fromsa
.proof_step_extra({"H0"})
indicates the set of steps{"H0"}
are not needed insa
.proof_step_must("S1", "S2")
indicates that the step"S1"
must come before step"S2"
in the proof specified by the graph, but does not insa
.
If the result of proof_assessment_dag
is an empty list, the sa
matches with the graph in tdag
.
If, for any reason, you don't want all three checks, then you can filter the list to retain only the relevant properties from that listed above. E.g. if you only want proof_step_must
to be established then use
saprob: proof_dag_check(sa, tdag);
saprob:sublist(saprob, lambda([ex], op(ex)=proof_step_must));
To use this in a potential response tree, check if saprob
is empty. If not, you can display feedback based on the properties described above using the following in the feedback.
{@proof_dag_problem_display(saprob, poof_steps)@}
An example question illustrating these features is given in the sample questions library under Topics/Parsons-DAG.xml
.
3. General tools in builder.mac
to provide bespoke feedback
The last option is to encode specific dependencies, with feedback for each. The advantage is that teachers can mange meaningful feedback for individual steps, but of course that is more work than using the auto-generated feedback.
You can load the optional library builder.mac
to use tools to manage bespoke feedback.
An example question is given in the stacklibrary Doc-Examples\Parsons-3-builder.xml
This example combines both the automatic feedback in Section 1 where the proof is defined as a tree (a list in our example), with the bespoke feedback for many of the steps.