Communities

Writing
Writing
Codidact Meta
Codidact Meta
The Great Outdoors
The Great Outdoors
Photography & Video
Photography & Video
Scientific Speculation
Scientific Speculation
Cooking
Cooking
Electrical Engineering
Electrical Engineering
Judaism
Judaism
Languages & Linguistics
Languages & Linguistics
Software Development
Software Development
Mathematics
Mathematics
Christianity
Christianity
Code Golf
Code Golf
Music
Music
Physics
Physics
Linux Systems
Linux Systems
Power Users
Power Users
Tabletop RPGs
Tabletop RPGs
Community Proposals
Community Proposals
tag:snake search within a tag
answers:0 unanswered questions
user:xxxx search by author id
score:0.5 posts with 0.5+ score
"snake oil" exact phrase
votes:4 posts with 4+ votes
created:<1w created < 1 week ago
post_type:xxxx type of post
Search help
Notifications
Mark all as read See all your notifications »
Q&A

Welcome to Software Development on Codidact!

Will you help us build our independent community of developers helping developers? We're small and trying to grow. We welcome questions about all aspects of software development, from design to code to QA and more. Got questions? Got answers? Got code you'd like someone to review? Please join us.

Comments on Understanding "de Morgan's laws"

Parent

Understanding "de Morgan's laws"

+2
−0

While trying to understand logical 'or'/'and', I encountered another problem (I'm writing Python code here, but my question is about the logic, not about any given programming language). I have some code like:

if x or y:
    pass
else:
    do_something()

(Of course, following the other Q&A, I made sure that the real x and y make sense independently as conditions.)

I wanted to focus on the else case, because it's actually interesting. I figured out that I could negate the condition to flip the cases, and then the else: pass part would be unnecessary:

if not(x or y):
    do_something()

However, other obvious attempts did not work:

if not x or y:
    do_something()

if not x or not y:
    do_something()

if (not x) or (not y): # not even with more parentheses!
    do_something()

I asked about this elsewhere and was told it had something to do with "de Morgan's laws". What does this mean? Can I use these "laws" to understand how to write the code correctly? How exactly do they apply?

History
Why does this post require attention from curators or moderators?
You might want to add some details to your flag.
Why should this post be closed?

1 comment thread

Useful references (1 comment)
Post
+3
−0

Overview

de Morgan's laws are rules of logic that allow you to transform one conditional expression into another, equivalent expression. Because (in Boolean logic) two negations cancel each other out (i.e. not(not x) is equivalent to x), we can also infer rules for negating an expression.

The core idea is that, given a simple logical-and or logical-or with a left-hand side and a right-hand side, we can get an equivalent result by:

  1. negating both sides;

  2. swapping "and" for "or", and vice-versa;

  3. negating the result.

Examples:

By applying the steps, we find that x or y is equivalent to not((not x) and (not y)):

x or y
(not x) or (not y) # negate each side independently
(not x) and (not y) # replace 'or' with 'and'
not((not x) and (not y)) # negate the entire expression

Similarly, m and n is equivalent to not((not m) or (not n)):

m and n
(not m) and (not n) # negate each side independently
(not m) or (not n) # replace 'and' with 'or'
not((not m) or (not n)) # negate the entire expression

We can also start with negated sub-expressions:

(not x) or (not y)
not((not x)) or not((not y)) # negate each side independently
x or y # cancel double negations (we can do this freely any time)
x and y # replace 'or' with 'and'
not(x and y) # negate the entire expression

and:

(not m) and (not n)
not((not m)) and not((not n)) # negate each side independently
m and n # cancel double negations (we can do this freely any time)
m or n # replace 'and' with 'or'
not(m or n) # negate the entire expression

Notice that if we apply the procedure twice and make sure to cancel all pairs of negations, we'll get the original result back.

Even though new programmers often get this wrong, the rule actually makes sense in natural English: "one of these things is true" means the same thing as "these things aren't both false".

Proof

We can prove this simply by inspection (building some truth tables).

For example:
x and y x true x false
y true true false
y false false false

Therefore, by the double-negation principle:

x and y (not x) false (not x) true
(not y) false true false
(not y) true false false

If we negate the expression, that is equivalent to negating all results:

not (x and y) (not x) false (not x) true
(not y) false false true
(not y) true true true

And swapping rows and columns for clarity:

not (x and y) (not x) true (not x) false
(not y) true true true
(not y) false true false

But now we can see that the truth table for not (x and y) is identically the truth table for (not x) or (not y), QED.

Importance

These laws entail that we always have two fundamental ways to write the same expression (notwithstanding useless pairs of nots). Thus, we can choose between them as we see fit, to make the code easier to read and understand.

Because, again, two negations cancel each other out in Boolean logic, and because the last step is negation, this also gives us an easy way to negate an expression (which is useful for inverting if/else logic). Given the condition x or y, we can get the opposite condition by applying the first two steps - thus, (not x) and (not y).

Finally, this rule has important applications in hardware. A NOT gate can be created from a NAND gate by simply connecting both inputs to the same source; and an AND gate is of course just the inverse of NAND, so we can create it by connecting NOT to the output of NAND. Then, because of de Morgan's laws, we see that we can construct an OR gate by connecting NOT to each input of NAND. Further following this chain of logic, we can demonstrate that every combinatorial logic circuit can be created from a combination of NAND gates.

Applications

  1. Here is a two and a half hour video exposition of the functional completeness of NAND.

  2. Relational operators also have negations. In particular, != and == are typically defined as opposites (some programming languages use different symbols); not(x == y) could equivalently be written x != y Thus, we can transform (a == b) or (c == d) into not((a != b) and (c != d)).

    Here is a quick reference table:

    Original expression Negation
    x == y x != y
    x != y x == y
    x < y x >= y
    x > y x <= y
    x <= y x > y
    x >= y x < y

(We can think of these operators as implementing 3-input truth tables, where the inputs are x < y, x == y and x > y - except skipping the ones for "always true" and "always false".)

  1. A common logical error in code looks like (again, this is a Python example, but the principle applies to other languages):

    if x != 1 or x != 2:
        print("neither 1 nor 2")
    

    The intent was to negate the condition x == 1 or x == 2. But because of de Morgan's law, the or should be replaced with and. The consequence of this error is that the condition is always met. (To see that more easily, apply the law: our wrong condition is thus the negation of x == 1 and x == 2, which obviously is never met.)

History
Why does this post require attention from curators or moderators?
You might want to add some details to your flag.

1 comment thread

Python-specific addendum (1 comment)
Python-specific addendum

In addition to the arithmetic-relational operators (to compare whether values are equal and/or which is larger), Python also has an in relational operator - x in y checks whether y "contains" x in some sense (depending on the types of the values). This operation also has a negation: not (x in y) is equivalent to x not in y. Notice that this not in is treated as a single operator spelled out in two separate words.