First, we show that we can convert any DFA into DFA-er code. Suppose we are given a deterministic finite automaton
, where
is the set of states,
is the set of input symbols,
is the transition function,
is the initial state, and
is the set of accepting states.
Let [
]
denote the binary representation of
. To convert
to DFA-er code, we use the following algorithm:
- If
write ..[
].
, otherwise write .[
].
- For each symbol
:
- Let
data:image/s3,"s3://crabby-images/46ebb/46ebb85b65a707f1fca5deaf64a37e2999861740" alt="{\displaystyle q_{j}=\delta (q_{0},\sigma )}"
- Write
-[
]-[
]-
- For each state
:
- If
write ..[
].
, otherwise write .[
].
- For each symbol
:
- Let
data:image/s3,"s3://crabby-images/ce996/ce996ae1ee3678b30d5634a858fd358521c12bed" alt="{\displaystyle q_{j}=\delta (q_{i},\sigma )}"
- Write
-[
]-[
]-
- Write
!
The code created by this process will have
as a starting state, since it is the first state specified. For every
, it will be accepting, since it will be written as ..[
].
, and for every
, it will be failing, since it will be written as .[
].
. Because
is a DFA, we know that
is defined for all
and all
. We therefore know that every transition of
will be encoded, since we iterate through all
and all
, encoding the proper transition. Therefore, the code correctly encodes
.
Assuming that we can look up
in
time, this algorithm runs in time
, since we iterate through each
, and for each state, we iterate through each
. It uses no additional storage.
The DFA-er code produced by this algorithm will encode each state once and each transition once, meaning it is the shortest code that completely encodes the transition function
. It is possible that a shorter DFA-er representation of
exists, however, since many DFAs include a "fail state," meaning a state
for which
. Since DFA-er does not require that every state have a defined transition for every symbol in
,
would not need to be expressed in the code, and neither would every transition to
.
Next, we show that we can convert any valid DFA-er code into a DFA.
To do this, we use the following algorithm (see DFA-er's Python interpreter for a code example):
- Create a DFA
data:image/s3,"s3://crabby-images/61659/61659fedb03601dafddd2699fcef8b5bdda6c2a8" alt="{\displaystyle M=(Q=\{\},\Sigma =\{\},\delta =\{\},q_{0}=\mathrm {NULL} ,F=\{\})}"
- Let
data:image/s3,"s3://crabby-images/15420/15420b9fc4a89e7db70b6e4712422395d1a0e490" alt="{\displaystyle c=0}"
- While there is still code to read (i.e., we have not yet encountered a
!
):
- If we encounter code of the form
..[
].
:
- If
, let data:image/s3,"s3://crabby-images/74fc9/74fc99a86941466646f6da26a308f71aa397ffba" alt="{\displaystyle q_{0}=q}"
- If
, let data:image/s3,"s3://crabby-images/0d0f1/0d0f114251e6d7a0b9ca45add828b6074bef61d4" alt="{\displaystyle Q=Q\cup \{q\}}"
- If
, let data:image/s3,"s3://crabby-images/4bbf4/4bbf4370f5bfd3ade01649f63fbdf8a62e580a08" alt="{\displaystyle F=F\cup \{q\}}"
- Let
data:image/s3,"s3://crabby-images/5096a/5096a76c73106cb1b875b1948c90399332c09019" alt="{\displaystyle c=q}"
- If we encounter code of the form
.[
].
:
- If
, let data:image/s3,"s3://crabby-images/74fc9/74fc99a86941466646f6da26a308f71aa397ffba" alt="{\displaystyle q_{0}=q}"
- If
, let data:image/s3,"s3://crabby-images/0d0f1/0d0f114251e6d7a0b9ca45add828b6074bef61d4" alt="{\displaystyle Q=Q\cup \{q\}}"
- If
, let data:image/s3,"s3://crabby-images/92537/9253791653d0b64cdf784e5ed94156d1dedee0c0" alt="{\displaystyle F=F-\{q\}}"
- Let
data:image/s3,"s3://crabby-images/5096a/5096a76c73106cb1b875b1948c90399332c09019" alt="{\displaystyle c=q}"
- If we encounter code of the form
-[
]-[
]-
:
- If
, let data:image/s3,"s3://crabby-images/14982/149825f0ee0a7b2833b56b63eb1c4caef10ca030" alt="{\displaystyle \Sigma =\Sigma \cup \{\sigma \}}"
- If
, let data:image/s3,"s3://crabby-images/bde10/bde10db3db657588e5fd55a600d5721dfdef8514" alt="{\displaystyle Q=Q\cup \{c\}}"
- If
, let data:image/s3,"s3://crabby-images/0d0f1/0d0f114251e6d7a0b9ca45add828b6074bef61d4" alt="{\displaystyle Q=Q\cup \{q\}}"
- Let
data:image/s3,"s3://crabby-images/39741/397417d37c41557feda00217edb58c96b47294d7" alt="{\displaystyle \delta (c,\sigma )=q}"
- Let
data:image/s3,"s3://crabby-images/7be2f/7be2f1e9a8e43237b71b2377b54171d3901eb602" alt="{\displaystyle Q=Q\cup \{q_{f}\}}"
- For every
:
- If
is undefined, let data:image/s3,"s3://crabby-images/bf5ef/bf5efdd7944c53687c82d729a817a97aa557f6b3" alt="{\displaystyle \delta (q,\sigma )=q_{f}}"
This algorithm will produce a valid DFA, since it will have a
that is defined for every possible
; either going to the state given by the code, or to
(the failing state) if it is not specified.
This algorithm will run in time
, where
is the length of the code. We also know that
because
and
. Therefore, the running time, expressed solely with respect to the input (the code), is
.
Since any DFA can be converted to DFA-er code, and since any valid DFA-er code can be converted to a DFA, DFA-er must have the computational power of a DFA.