Skip to content

WA5. Propositional Logic

Statement

A set of symbols is commonly used to express logical representation. Let’s define the semantics of each propositional symbol:

  • P means “It is hot”
  • Q means “It is humid”
  • R means “It is raining”

Examples of Propositional Logic (PL) sentences

  • (P ^ Q) => R (meaning “If it is hot and humid, then it is raining”)
  • Q => P (meaning “If it is humid, then it is hot”)
  • Q (meaning “It is humid.”)

Using the “weather” sentences from above, consider the following Knowledge Base (KB):

  • KB = (((P ^ Q) => R) ^ (Q => P) ^ Q)

Corresponding to the three facts we know about the weather:

  1. If it is hot and humid, then it is raining.
  2. If it is humid, then it is hot, and
  3. It is humid.

Now let’s ask the query “Is it raining?” That is, is the query sentence R entailed by KB.

P Q R (P ^ Q) => R Q => P Q KB R KB => R
T T T T T T T T T
T T F F T T F F T
T F T T T F F T T
T F F T T F F F T
F T T T F T F T T
F T F T F T F F T
F F T T T F F T T
F F F T T F F F T

Hence, in this problem there is only one model of KB, when P, Q, and R are all True, and in this case R is also True, so R is entailed by KB.

For this programming assignment, you need to write a program in Java or Python which will output the above truth table. Here are the detailed requirements:

  • Your program must be complete, compile able, and produce output.
  • Your program will not take any input.
  • You will have three variables P, Q, R, and dynamically generate those 8 combinations.
  • You will define an efficient data structure which will hold the values of result variables: (p ^ Q) => R, Q=>P, KB, and KB=>R
  • In your algorithm, you must define the rule for the conjunction (^) and implication (=>) operator, so that given two operands, they produce the correct result.
  • In the above truth table, Q and R columns appeared twice because of convenience. In your output, you don’t have to have these duplicates.
  • You don’t have to display the output in a table with borders, however, consider proper alignment, padding, and formatting for each row and column.
  • Make sure your output matches with the above table.

Table of Contents


Introduction

This assignment provides a source code in java, an explanation of the source code, and example output. To run the code, you need to compile the source code and then run it. All files should be passed with the attachments.

To run the code, you need to:

  1. Compile the java code using the command javac Wa5.java.
  2. Run the compiled code using the command java Wa5.

Source Code

Below is the source code for the assignment, all in one file, to make easy to run and test. The main class is Wa5 which contains other helper classes and methods to implement the truth table for the given knowledge base.

class Wa5 {

    public static class KnowledgeBase {

        private boolean P;
        private boolean Q;
        private boolean R;
        private boolean p1; // (P ^ Q) => R
        private boolean p2; // Q => P
        private boolean p3; // Q
        private boolean kb; // KB = (((P ^ Q) => R) ^ (Q => P) ^ Q)
        private boolean result; // KB => R

        public static boolean conjunction(boolean a, boolean b) {
            return a && b;
        }


        // See  https://www.geeksforgeeks.org/proposition-logic/
        public static boolean implication(boolean a, boolean b) {
            return !a || b;
        }

        // Evaluate the knowledge base KB = (((P ^ Q) => R) ^ (Q => P) ^ Q)
        public static boolean evaluateKb(boolean P, boolean Q, boolean R) {
            boolean p1 = implication(conjunction(P, Q), R);
            boolean p2 = implication(Q, P);
            boolean p3 = Q;
            return p1 && p2 && p3;
        }

        KnowledgeBase(boolean P, boolean Q, boolean R) {
            this.P = P;
            this.Q = Q;
            this.R = R;
            this.p1 = implication(conjunction(P, Q), R);
            this.p2 = implication(Q, P);
            this.p3 = Q;
            boolean kb = evaluateKb(P, Q, R);
            this.kb = kb;
            this.result = implication(kb, R);
        }

        // Get the truth values of the knowledge base according
        // to the current values of P, Q, and R
        public boolean[] getTruthValues() {
            return new boolean[] { P, Q, R, p1, p2, p3, kb, result };
        }
    }


    // returns "T" for true and "F" for false
    public static String boolToString(boolean b) {
        return b ? "T" : "F";
    }


    // centers a string in a field of a given width
    public static String center(String s, int width) {
        if (s.length() >= width) {
            return s;
        }
        int leftPadding = (width - s.length()) / 2;
        int rightPadding = width - s.length() - leftPadding;
        return " ".repeat(leftPadding) + s + " ".repeat(rightPadding);
    }

    // prints a truth table in Markdown format
    public static void printTruthTable(boolean[][] truthTable) {
        // Build Markdown table header row
        String[] headers = { "P", "Q", "R", "(P ^ Q) => R", "Q => P", "Q", "KB", "KB => R" };
        int[] widths = { 5, 5, 5, 15, 10, 5, 5, 10 };
        StringBuilder headerRow = new StringBuilder("|");
        for (int i = 0; i < headers.length; i++) {
            headerRow.append(" ").append(center(headers[i], widths[i])).append(" |");
        }
        System.out.println(headerRow);

        // Build Markdown table alignment row (centered)
        StringBuilder alignRow = new StringBuilder("|");
        for (int i = 0; i < headers.length; i++) {
            // Create a string with dashes of the column width
            String dashes = "-".repeat(Math.max(0, widths[i]));
            alignRow.append(":").append(dashes).append(":|");
        }
        System.out.println(alignRow);

        // Print each row of the truth table
        for (int i = 0; i < truthTable.length; i++) {
            boolean[] row = truthTable[i];
            StringBuilder rowBuilder = new StringBuilder("|");
            // For each column, center the value in the corresponding width
            rowBuilder
                .append(" ")
                .append(center(boolToString(row[0]), widths[0]))
                .append(" |")
                .append(" ")
                .append(center(boolToString(row[1]), widths[1]))
                .append(" |")
                .append(" ")
                .append(center(boolToString(row[2]), widths[2]))
                .append(" |")
                .append(" ")
                .append(center(boolToString(row[3]), widths[3]))
                .append(" |")
                .append(" ")
                .append(center(boolToString(row[4]), widths[4]))
                .append(" |")
                .append(" ")
                .append(center(boolToString(row[5]), widths[5]))
                .append(" |")
                .append(" ")
                .append(center(boolToString(row[6]), widths[6]))
                .append(" |")
                .append(" ")
                .append(center(boolToString(row[7]), widths[7]))
                .append(" |");
            System.out.println(rowBuilder);
        }
    }

    public static void main(String[] args) {
        // generate all possible truth values for P, Q, R
        boolean[][] permutations = {
            { true, true, true },
            { true, true, false },
            { true, false, true },
            { true, false, false },
            { false, true, true },
            { false, true, false },
            { false, false, true },
            { false, false, false }
        };

        // generate truth table
        boolean[][] truthTable = new boolean[8][8];
        for (int i = 0; i < 8; i++) {
            boolean[] input = permutations[i];
            KnowledgeBase kb = new KnowledgeBase(input[0], input[1], input[2]);
            truthTable[i] = kb.getTruthValues();
        }

        printTruthTable(truthTable);
    }
}

Source Code Explanation

  • The main class Wa5 contains a main method and nested class KnowledgeBase which represents the knowledge base for the given problem.
  • The KnowledgeBase class contains the following fields:
    • P, Q, R: The truth values of the propositional symbols.
    • p1: The truth value of the expression (P ^ Q) => R.
    • p2: The truth value of the expression Q => P.
    • p3: The truth value of the expression Q.
    • kb: The truth value of the knowledge base KB = (((P ^ Q) => R) ^ (Q => P) ^ Q).
    • result: The truth value of the expression KB => R.
    • conjunction: represents an AND operation between two boolean values.
    • implication: represents logical implication between two boolean values according to (GeeksforGeeks, 2015).
    • evaluateKb: evaluates the knowledge base according to the current values of P, Q, and R.
    • getTruthValues: returns the truth values of the knowledge base as an array of booleans in the order P, Q, R, (P ^ Q) => R, Q => P, Q, KB, KB => R.
  • The Wa5 class also includes the following helper methods:
    • boolToString: converts a boolean value to a string representation ("T" for true and "F" for false).
    • center: centers a string in a field of a given width.
    • printTruthTable: prints the truth table in Markdown format with proper alignment and padding.
  • The main method generates all possible truth values for P, Q, and R and then generates the truth table by evaluating the knowledge base for each combination of truth values.

Important Notes

  • The code does not generate the combinations of P, Q, and R dynamically. Instead, it uses a predefined set of permutations.
  • This is just to simplify the output to exactly match the provided table in the problems statement.
  • To dynamically generate the combinations, you can modify the code to generate the permutations of P, Q, and R using nested loops as follows:
// generate all possible truth values for P, Q, R
boolean[][] permutations = new boolean[8][3];
for (int i = 0; i < 8; i++) {
    permutations[i][0] = i % 2 == 1;
    permutations[i][1] = (i / 2) % 2 == 1;
    permutations[i][2] = (i / 4) % 2 == 1;
}

Output

By compiling and running the code, you will get the following output:

Image 1: Output of the code
Output of the code

And here is the output converted to a table:

P Q R (P ^ Q) => R Q => P KB KB => R
T T T T T T T
T T F F T F T
T F T T T F T
T F F T T F T
F T T T F F T
F T F T F F T
F F T T T F T
F F F T T F T

Conclusion

We have provided the source code, source code explanation, and example output. Here are our checks for the assignment requirements:

Requirement Check Comments
Does the program compile? See the Output section.
Does the program dynamically generate those 8 combinations? See the Important Notes section.
Does the data structure used to hold the values efficient enough? KnowledgeBase class; we preferred clarity and simplicity.
Does the algorithm set the rule for conjunction (^) and implication (=>) operator? Conjunction: (a,b) => a AND b. Implication: (a,b) => !a OR b.
Does the program output correctly for (p ^ Q) => R, Q=>P, KB, and KB=>R? See the Output section.
Does the output follow proper alignment, formatting, and padding? Table formatted according to markdown rules.

References