CYK algorithm
Class  Parsing with contextfree grammars 

Data structure  String 
Worstcase performance  ${\mathcal {O}}\left(n^{3}\cdot \leftG\right\right)$, where:

In computer science, the Cocke–Younger–Kasami algorithm (alternatively called CYK, or CKY) is a parsing algorithm for contextfree grammars published by Itiroo Sakai in 1961.^{[1]} The algorithm is named after some of its rediscoverers: John Cocke, Daniel Younger, Tadao Kasami, and Jacob T. Schwartz. It employs bottomup parsing and dynamic programming.
The standard version of CYK operates only on contextfree grammars given in Chomsky normal form (CNF). However any contextfree grammar may be transformed (after convention) to a CNF grammar expressing the same language (Sipser 1997).
The importance of the CYK algorithm stems from its high efficiency in certain situations. Using big O notation, the worst case running time of CYK is ${\mathcal {O}}\left(n^{3}\cdot \leftG\right\right)$, where $n$ is the length of the parsed string and $\leftG\right$ is the size of the CNF grammar $G$ (Hopcroft & Ullman 1979, p. 140). This makes it one of the most efficient parsing algorithms in terms of worstcase asymptotic complexity, although other algorithms exist with better average running time in many practical scenarios.
Standard form
The dynamic programming algorithm requires the contextfree grammar to be rendered into Chomsky normal form (CNF), because it tests for possibilities to split the current sequence into two smaller sequences. Any contextfree grammar that does not generate the empty string can be represented in CNF using only production rules of the forms $A\rightarrow \alpha$, $A\rightarrow BC$, and $S\to \varepsilon$ where $S$ is the start symbol.^{[2]}
Algorithm
As pseudocode
The algorithm in pseudocode is as follows:
let the input be a string I consisting of n characters: a_{1} ... a_{n}. let the grammar contain r nonterminal symbols R_{1} ... R_{r}, with start symbol R_{1}. let P[n,n,r] be an array of booleans. Initialize all elements of P to false. let back[n,n,r] be an array of lists of backpointing triples. Initialize all elements of back to the empty list. for each s = 1 to n for each unit production R_{v} → a_{s} set P[1,s,v] = true for each l = 2 to n  Length of span for each s = 1 to nl+1  Start of span for each p = 1 to l1  Partition of span for each production R_{a} → R_{b} R_{c} if P[p,s,b] and P[lp,s+p,c] then set P[l,s,a] = true, append <p,b,c> to back[l,s,a] if P[n,1,1] is true then I is member of language return back  by retracing the steps through back, one can easily construct all possible parse trees of the string. else return "not a member of language"
Probabilistic CYK (for finding the most probable parse)
Allows to recover the most probable parse given the probabilities of all productions.
let the input be a string I consisting of n characters: a_{1} ... a_{n}. let the grammar contain r nonterminal symbols R_{1} ... R_{r}, with start symbol R_{1}. let P[n,n,r] be an array of real numbers. Initialize all elements of P to zero. let back[n,n,r] be an array of backpointing triples. for each s = 1 to n for each unit production R_{v} →a_{s} set P[1,s,v] = Pr(R_{v} →a_{s}) for each l = 2 to n  Length of span for each s = 1 to nl+1  Start of span for each p = 1 to l1  Partition of span for each production R_{a} → R_{b} R_{c} prob_splitting = Pr(R_{a} →R_{b} R_{c}) * P[p,s,b] * P[lp,s+p,c] if prob_splitting > P[l,s,a] then set P[l,s,a] = prob_splitting set back[l,s,a] = <p,b,c> if P[n,1,1] > 0 then find the parse tree by retracing through back return the parse tree else return "not a member of language"
As prose
In informal terms, this algorithm considers every possible substring of the input string and sets $P[l,s,v]$ to be true if the substring of length $l$ starting from $s$ can be generated from the nonterminal $R_{v}$. Once it has considered substrings of length 1, it goes on to substrings of length 2, and so on. For substrings of length 2 and greater, it considers every possible partition of the substring into two parts, and checks to see if there is some production $A\to B\;C$ such that $B$ matches the first part and $C$ matches the second part. If so, it records $A$ as matching the whole substring. Once this process is completed, the input string is generated by the grammar if the substring containing the entire input string is matched by the start symbol.
Example
This is an example grammar:
 ${\begin{aligned}{\ce {S}}&\ {\ce {>NP\ VP}}\\{\ce {VP}}&\ {\ce {>VP\ PP}}\\{\ce {VP}}&\ {\ce {>V\ NP}}\\{\ce {VP}}&\ {\ce {>eats}}\\{\ce {PP}}&\ {\ce {>P\ NP}}\\{\ce {NP}}&\ {\ce {>Det\ N}}\\{\ce {NP}}&\ {\ce {>she}}\\{\ce {V}}&\ {\ce {>eats}}\\{\ce {P}}&\ {\ce {>with}}\\{\ce {N}}&\ {\ce {>fish}}\\{\ce {N}}&\ {\ce {>fork}}\\{\ce {Det}}&\ {\ce {>a}}\end{aligned}}$
Now the sentence she eats a fish with a fork is analyzed using the CYK algorithm. In the following table, in $P[i,j,k]$, i is the number of the row (starting at the bottom at 1), and j is the number of the column (starting at the left at 1).
S  
VP  
S  
VP  PP  
S  NP  NP  
NP  V, VP  Det.  N  P  Det  N 
she  eats  a  fish  with  a  fork 
For readability, the CYK table for P is represented here as a 2dimensional matrix M containing a set of nonterminal symbols, such that R_{k} is in $M[i,j]$ if, and only if, $P[i,j,k]$. In the above example, since a start symbol S is in $M[7,1]$, the sentence can be generated by the grammar.
Extensions
Generating a parse tree
The above algorithm is a recognizer that will only determine if a sentence is in the language. It is simple to extend it into a parser that also constructs a parse tree, by storing parse tree nodes as elements of the array, instead of the boolean 1. The node is linked to the array elements that were used to produce it, so as to build the tree structure. Only one such node in each array element is needed if only one parse tree is to be produced. However, if all parse trees of an ambiguous sentence are to be kept, it is necessary to store in the array element a list of all the ways the corresponding node can be obtained in the parsing process. This is sometimes done with a second table B[n,n,r] of socalled backpointers. The end result is then a sharedforest of possible parse trees, where common trees parts are factored between the various parses. This shared forest can conveniently be read as an ambiguous grammar generating only the sentence parsed, but with the same ambiguity as the original grammar, and the same parse trees up to a very simple renaming of nonterminals, as shown by Lang (1994).
Parsing nonCNF contextfree grammars
As pointed out by Lange & Leiß (2009), the drawback of all known transformations into Chomsky normal form is that they can lead to an undesirable bloat in grammar size. The size of a grammar is the sum of the sizes of its production rules, where the size of a rule is one plus the length of its righthand side. Using $g$ to denote the size of the original grammar, the size blowup in the worst case may range from $g^{2}$ to $2^{2g}$, depending on the transformation algorithm used. For the use in teaching, Lange and Leiß propose a slight generalization of the CYK algorithm, "without compromising efficiency of the algorithm, clarity of its presentation, or simplicity of proofs" (Lange & Leiß 2009).
Parsing weighted contextfree grammars
It is also possible to extend the CYK algorithm to parse strings using weighted and stochastic contextfree grammars. Weights (probabilities) are then stored in the table P instead of booleans, so P[i,j,A] will contain the minimum weight (maximum probability) that the substring from i to j can be derived from A. Further extensions of the algorithm allow all parses of a string to be enumerated from lowest to highest weight (highest to lowest probability).
Numerical stability
When the probabilistic CYK algorithm is applied to a long string, the splitting probability can become very small due to multiplying many probabilities together. This can be dealt with by summing logprobability instead of multiplying probabilities.
Valiant's algorithm
The worst case running time of CYK is $\Theta (n^{3}\cdot G)$, where n is the length of the parsed string and G is the size of the CNF grammar G. This makes it one of the most efficient algorithms for recognizing general contextfree languages in practice. Valiant (1975) gave an extension of the CYK algorithm. His algorithm computes the same parsing table as the CYK algorithm; yet he showed that algorithms for efficient multiplication of matrices with 01entries can be utilized for performing this computation.
Using the Coppersmith–Winograd algorithm for multiplying these matrices, this gives an asymptotic worstcase running time of $O(n^{2.38}\cdot G)$. However, the constant term hidden by the Big O Notation is so large that the Coppersmith–Winograd algorithm is only worthwhile for matrices that are too large to handle on presentday computers (Knuth 1997), and this approach requires subtraction and so is only suitable for recognition. The dependence on efficient matrix multiplication cannot be avoided altogether: Lee (2002) has proved that any parser for contextfree grammars working in time $O(n^{3\varepsilon }\cdot G)$ can be effectively converted into an algorithm computing the product of $(n\times n)$matrices with 01entries in time $O(n^{3\varepsilon /3})$, and this was extended by Abboud et al.^{[3]} to apply to a constantsize grammar.
See also
References
 ^ Grune, Dick (2008). Parsing techniques : a practical guide (2nd ed.). New York: Springer. p. 579. ISBN 9780387202488.
 ^ Sipser, Michael (2006). Introduction to the theory of computation (2nd ed.). Boston: Thomson Course Technology. Definition 2.8. ISBN 0534950973. OCLC 58544333.
 ^ Abboud, Amir; Backurs, Arturs; Williams, Virginia Vassilevska (20151105). "If the Current Clique Algorithms are Optimal, so is Valiant's Parser". arXiv:1504.01431 [cs.CC].
Sources
 Sakai, Itiroo (1962). Syntax in universal translation. 1961 International Conference on Machine Translation of Languages and Applied Language Analysis, Teddington, England. Vol. II. London: Her Majesty’s Stationery Office. pp. 593–608.
 Cocke, John; Schwartz, Jacob T. (April 1970). Programming languages and their compilers: Preliminary notes (PDF) (Technical report) (2nd revised ed.). CIMS, NYU.
 Hopcroft, John E.; Ullman, Jeffrey D. (1979). Introduction to Automata Theory, Languages, and Computation. Reading/MA: AddisonWesley. ISBN 020102988X.
 Kasami, T. (1965). An efficient recognition and syntaxanalysis algorithm for contextfree languages (Technical report). AFCRL. 65758.
 Knuth, Donald E. (November 14, 1997). The Art of Computer Programming Volume 2: Seminumerical Algorithms (3rd ed.). AddisonWesley Professional. p. 501. ISBN 0201896842.
 Lang, Bernard (1994). "Recognition can be harder than parsing". Comput. Intell. 10 (4): 486–494. CiteSeerX 10.1.1.50.6982. doi:10.1111/j.14678640.1994.tb00011.x. S2CID 5873640.
 Lange, Martin; Leiß, Hans (2009). "To CNF or not to CNF? An Efficient Yet Presentable Version of the CYK Algorithm". Informatica Didactica. 8.
 Lee, Lillian (2002). "Fast contextfree grammar parsing requires fast Boolean matrix multiplication". J. ACM. 49 (1): 1–15. arXiv:cs/0112018. doi:10.1145/505241.505242. S2CID 1243491.
 Sipser, Michael (1997). Introduction to the Theory of Computation (1st ed.). IPS. p. 99. ISBN 053494728X.
 Valiant, Leslie G. (1975). "General contextfree recognition in less than cubic time". J. Comput. Syst. Sci. 10 (2): 308–314. doi:10.1016/s00220000(75)800468.
 Younger, Daniel H. (February 1967). "Recognition and parsing of contextfree languages in time n^{3}". Inform. Control. 10 (2): 189–208. doi:10.1016/s00199958(67)80007x.
External links
 CYK parsing demo in JavaScript
 Exorciser is a Java application to generate exercises in the CYK algorithm as well as Finite State Machines, Markov algorithms etc