Chomsky normal form should not be confused with conjunctive normal form.
In formal language theory, a context-free grammar, G, is said to be in Chomsky normal form (first described by Noam Chomsky)[1] if all of its production rules are of the form:[2] [3]
A → BC, or
A → a, or
S → ε,where A, B, and C are nonterminal symbols, the letter a is a terminal symbol (a symbol that represents a constant value), S is the start symbol, and ε denotes the empty string. Also, neither B nor C may be the start symbol, and the third production rule can only appear if ε is in L(G), the language produced by the context-free grammar G.[4]
Every grammar in Chomsky normal form is context-free, and conversely, every context-free grammar can be transformed into an equivalent one[5] which is in Chomsky normal form and has a size no larger than the square of the original grammar's size.
To convert a grammar to Chomsky normal form, a sequence of simple transformations is applied in a certain order; this is described in most textbooks on automata theory.[4] [6] [7] [8] The presentation here follows Hopcroft, Ullman (1979), but is adapted to use the transformation names from Lange, Leiß (2009).[9] [10] Each of the following transformations establishes one of the properties required for Chomsky normal form.
Introduce a new start symbol S0, and a new rule
S0 → S, where S is the previous start symbol.This does not change the grammar's produced language, and S0 will not occur on any rule's right-hand side.
To eliminate each rule
A → X1 ... a ... Xnwith a terminal symbol a being not the only symbol on the right-hand side, introduce, for every such terminal, a new nonterminal symbol Na, and a new rule
Na → a. Change every rule
A → X1 ... a ... Xn to
A → X1 ... Na ... Xn.If several terminal symbols occur on the right-hand side, simultaneously replace each of them by its associated nonterminal symbol.This does not change the grammar's produced language.[4]
Replace each rule
A → X1 X2 ... Xn with more than 2 nonterminals X1,...,Xn by rules
A → X1 A1,
A1 → X2 A2,
...,
An-2 → Xn-1 Xn, where Ai are new nonterminal symbols.Again, this does not change the grammar's produced language.[4]
An ε-rule is a rule of the form
A → ε, where A is not S0, the grammar's start symbol.
To eliminate all rules of this form, first determine the set of all nonterminals that derive ε.Hopcroft and Ullman (1979) call such nonterminals nullable, and compute them as follows:
Obtain an intermediate grammar by replacing each rule
A → X1 ... Xn by all versions with some nullable Xi omitted.By deleting in this grammar each ε-rule, unless its left-hand side is the start symbol, the transformed grammar is obtained.[4]
For example, in the following grammar, with start symbol S0,
S0 → AbB | C
B → AA | AC
C → b | c
A → a | εthe nonterminal A, and hence also B, is nullable, while neither C nor S0 is.Hence the following intermediate grammar is obtained:[11]
S0 → b | b | b | b | C
B → | | | ε | C | C
C → b | c
A → a | εIn this grammar, all ε-rules have been "inlined at the call site".[12] In the next step, they can hence be deleted, yielding the grammar:
S0 → AbB | Ab | bB | b | C
B → AA | A | AC | C
C → b | c
A → aThis grammar produces the same language as the original example grammar, viz., but has no ε-rules.
A unit rule is a rule of the form
A → B, where A, B are nonterminal symbols.To remove it, for each rule
B → X1 ... Xn, where X1 ... Xn is a string of nonterminals and terminals, add rule
A → X1 ... Xn unless this is a unit rule which has already been (or is being) removed. The skipping of nonterminal symbol B in the resulting grammar is possible due to B being a member of the unit closure of nonterminal symbol A.[13]
+ Mutual preservation of transformation results | ||||||
Transformation X resp. the result of Y: | ||||||
START | TERM | BIN | DEL | UNIT | ||
---|---|---|---|---|---|---|
START | ||||||
TERM | ||||||
BIN | ||||||
DEL | ||||||
UNIT | ||||||
|
When choosing the order in which the above transformations are to be applied, it has to be considered that some transformations may destroy the result achieved by other ones. For example, START will re-introduce a unit rule if it is applied after UNIT. The table shows which orderings are admitted.
Moreover, the worst-case bloat in grammar size[14] depends on the transformation order. Using |G| to denote the size of the original grammar G, the size blow-up in the worst case may range from |G|2 to 22 |G|, depending on the transformation algorithm used.[9] The blow-up in grammar size depends on the order between DEL and BIN. It may be exponential when DEL is done first, but is linear otherwise. UNIT can incur a quadratic blow-up in the size of the grammar.[9] The orderings START,TERM,BIN,DEL,UNIT and START,BIN,DEL,UNIT,TERM lead to the least (i.e. quadratic) blow-up.
The following grammar, with start symbol Expr, describes a simplified version of the set of all syntactical valid arithmetic expressions in programming languages like C or Algol60. Both number and variable are considered terminal symbols here for simplicity, since in a compiler front end their internal structure is usually not considered by the parser. The terminal symbol "^" denoted exponentiation in Algol60.
Expr | → Term | Expr AddOp Term | AddOp Term | ||||
Term | → Factor | Term MulOp Factor | |||||
Factor | → Primary | Factor ^ Primary | |||||
Primary | → number | variable | (Expr) | ||||
AddOp | → + | - | |||||
MulOp | → * | / |
In step "START" of the above conversion algorithm, just a rule S0→Expr is added to the grammar.After step "TERM", the grammar looks like this:
S0 | → Expr | ||||||
Expr | → Term | Expr AddOp Term | AddOp Term | ||||
Term | → Factor | Term MulOp Factor | |||||
Factor | → Primary | Factor PowOp Primary | |||||
Primary | → number | variable | Open Expr Close | ||||
AddOp | → + | - | |||||
MulOp | → * | / | |||||
PowOp | → ^ | ||||||
Open | → ( | ||||||
Close | →) |
After step "BIN", the following grammar is obtained:
S0 | → Expr | ||||||
Expr | → Term | Expr AddOp_Term | AddOp Term | ||||
Term | → Factor | Term MulOp_Factor | |||||
Factor | → Primary | Factor PowOp_Primary | |||||
Primary | → number | variable | Open Expr_Close | ||||
AddOp | → + | - | |||||
MulOp | → * | / | |||||
PowOp | → ^ | ||||||
Open | → ( | ||||||
Close | →) | ||||||
AddOp_Term | → AddOp Term | ||||||
MulOp_Factor | → MulOp Factor | ||||||
PowOp_Primary | → PowOp Primary | ||||||
Expr_Close | → Expr Close |
Since there are no ε-rules, step "DEL" does not change the grammar.After step "UNIT", the following grammar is obtained, which is in Chomsky normal form:
S0 | → number | variable | Open Expr_Close | Factor PowOp_Primary | Term MulOp_Factor | Expr AddOp_Term | AddOp Term | ||||||||||||
Expr | → number | variable | Open Expr_Close | Factor PowOp_Primary | Term MulOp_Factor | Expr AddOp_Term | AddOp Term | ||||||||||||
Term | → number | variable | Open Expr_Close | Factor PowOp_Primary | Term MulOp_Factor | ||||||||||||||
Factor | → number | variable | Open Expr_Close | Factor PowOp_Primary | |||||||||||||||
Primary | → number | variable | Open Expr_Close | ||||||||||||||||
AddOp | → + | - | |||||||||||||||||
MulOp | → * | / | |||||||||||||||||
PowOp | → ^ | ||||||||||||||||||
Open | → ( | ||||||||||||||||||
Close | →) | ||||||||||||||||||
AddOp_Term | → AddOp Term | ||||||||||||||||||
MulOp_Factor | → MulOp Factor | ||||||||||||||||||
PowOp_Primary | → PowOp Primary | ||||||||||||||||||
Expr_Close | → Expr Close |
The Na introduced in step "TERM" are PowOp, Open, and Close.The Ai introduced in step "BIN" are AddOp_Term, MulOp_Factor, PowOp_Primary, and Expr_Close.
Another way[4] [15] to define the Chomsky normal form is:
A formal grammar is in Chomsky reduced form if all of its production rules are of the form:
A → BC
A → a
A
B
C
a
B
C
In a letter where he proposed a term Backus–Naur form (BNF), Donald E. Knuth implied a BNF "syntax in which all definitions have such a form may be said to be in 'Floyd Normal Form'",
\langleA\rangle::=\langleB\rangle\mid\langleC\rangle
\langleA\rangle::=\langleB\rangle\langleC\rangle
\langleA\rangle::=a
\langleA\rangle
\langleB\rangle
\langleC\rangle
a
Besides its theoretical significance, CNF conversion is used in some algorithms as a preprocessing step, e.g., the CYK algorithm, a bottom-up parsing for context-free grammars, and its variant probabilistic CKY.[18]