Skip to content

Commit 5972ba9

Browse files
author
Andrea
committed
Fixed wrong implementation of Hopcroft's algorithm #7
1 parent e2f625c commit 5972ba9

File tree

2 files changed

+147
-58
lines changed

2 files changed

+147
-58
lines changed

src/it/univr/fsm/machine/Automaton.java

+129-58
Original file line numberDiff line numberDiff line change
@@ -593,7 +593,7 @@ public static Automaton loadAutomata(String path){
593593
}
594594

595595
Automaton a= new Automaton(initialStates,delta,states);
596-
a.minimize();
596+
a.hopcroftMinimize();
597597
return a;
598598
}
599599

@@ -1206,9 +1206,9 @@ public static HashSet<String> getAlphabet(Automaton a){
12061206
return alphabet;
12071207
}
12081208

1209-
private State getOutgoingStatefromTransitionSymbol(HashSet<Transition> transitions, String symbol){
1210-
for(Transition t : transitions){
1211-
if(t.getInput().equals(symbol)){
1209+
private State getOutgoingStatefromTransitionSymbol(State s, String symbol){
1210+
for(Transition t : delta){
1211+
if(t.getInput().equals(symbol) && t.getFrom().equals(s)){
12121212
return t.getTo();
12131213
}
12141214
}
@@ -1219,15 +1219,15 @@ public void hopcroftremoveUnreachableStates(){
12191219
HashSet<State> unreachableStates = new HashSet<>();
12201220
HashSet<State> reachableStates = (HashSet<State>) this.getInitialStates().clone();
12211221
HashSet<State> newStates = (HashSet<State>) this.getInitialStates().clone();
1222+
HashSet<Transition> transitionstoRemove = new HashSet<>();
12221223
HashSet<State> temp;
12231224

12241225

12251226
do{
12261227
temp = new HashSet<>(Collections.<State>emptySet());
12271228
for(State s : newStates){
12281229
for(String a : getAlphabet(this)){
1229-
HashSet<Transition> transition = this.getOutgoingTransitionsFrom(s);
1230-
State to = getOutgoingStatefromTransitionSymbol(transition, a);
1230+
State to = getOutgoingStatefromTransitionSymbol(s, a);
12311231
if(to != null) temp.add(to);
12321232
}
12331233
}
@@ -1244,32 +1244,64 @@ public void hopcroftremoveUnreachableStates(){
12441244

12451245
states.removeAll(unreachableStates);
12461246

1247+
for(Transition t: delta)
1248+
if(!states.contains(t.getFrom()))
1249+
transitionstoRemove.add(t);
1250+
1251+
delta.removeAll(transitionstoRemove);
1252+
1253+
1254+
this.adjacencyList = this.computeAdjacencyList();
1255+
12471256
}
12481257

1249-
private HashSet<State> getXSet(HashSet<State> A, String symbol){
1258+
private HashSet<State> getXSet(HashSet<State> A, String c){
12501259
HashSet<State> s = new HashSet<State>();
12511260
for(Transition t: delta){
1252-
if(A.contains(t.getTo()) && t.getInput().equals(symbol)){
1253-
s.add(t.getTo());
1261+
if(A.contains(t.getTo()) && t.getInput().equals(c)){
1262+
s.add(t.getFrom());
12541263
}
12551264
}
12561265
return s;
12571266
}
12581267

12591268

1260-
private Set<State> setIntersection(HashSet<State> first, HashSet<State> second){
1269+
1270+
private LinkedList<HashSet<State>> getYList(HashSet<HashSet<State>> P, HashSet<State> X){
1271+
LinkedList<HashSet<State>> Ys = new LinkedList<>();
1272+
HashSet<State> Ytemp ;
1273+
1274+
for(HashSet<State> s : P){
1275+
1276+
//try to select a set, see if condition is respected
1277+
Ytemp = s;
1278+
1279+
if(!setIntersection(X,Ytemp).isEmpty() && !setSubtraction(Ytemp,X).isEmpty()){
1280+
Ys.add(Ytemp);
1281+
}
1282+
1283+
}
1284+
1285+
1286+
1287+
return Ys;
1288+
1289+
}
1290+
1291+
1292+
private HashSet<State> setIntersection(HashSet<State> first, HashSet<State> second){
12611293
HashSet<State> intersection = new HashSet<State>();
12621294
for(State s1: first)
12631295
for(State s2: second){
12641296
if(first.contains(s2) && second.contains(s1)){
1265-
intersection.add(s1);
1266-
intersection.add(s2);
1297+
intersection.add(s1.clone());
1298+
intersection.add(s2.clone());
12671299
}
12681300
}
12691301
return intersection;
12701302
}
12711303

1272-
private Set<State> setSubtraction(HashSet<State> first, HashSet<State> second){
1304+
private HashSet<State> setSubtraction(HashSet<State> first, HashSet<State> second){
12731305
HashSet<State> firstCopy = (HashSet<State>) first.clone();
12741306

12751307
for(State s: second){
@@ -1281,69 +1313,108 @@ private Set<State> setSubtraction(HashSet<State> first, HashSet<State> second){
12811313

12821314
public void hopcroftMinimize(){
12831315
this.hopcroftremoveUnreachableStates();
1284-
HashSet<State> P = (HashSet<State>) states.clone();
1285-
HashSet<State> W = this.getFinalStates();
1316+
1317+
// the partition P
1318+
HashSet<HashSet<State>> P = new HashSet<>();
1319+
P.add(this.getFinalStates());
1320+
P.add(setSubtraction(this.states, this.getFinalStates()) );
1321+
1322+
//the partition W
1323+
HashSet<HashSet<State>> W = new HashSet<>();
1324+
W.add(this.getFinalStates());
1325+
12861326
HashSet<State> A = new HashSet<>();
12871327
HashSet<State> X;
1288-
HashSet<State> Y = new HashSet<>();
1328+
LinkedList<HashSet<State>> listYs;
12891329
Random r=new Random();
12901330

12911331
while(!W.isEmpty()){
1292-
//choose and remove a random set A from W
1293-
for(State s : W){
1294-
int insert=r.nextInt(2);
1295-
if(insert > 0){
1296-
A.add(s);
1297-
W.remove(s);
1332+
//choose and remove a set A from W
1333+
do{
1334+
for(HashSet<State> s : W){
1335+
int insert=r.nextInt(2);
1336+
if(insert > 0)
1337+
A = s;
1338+
else break;
12981339
}
1299-
}
1340+
W.remove(A);
1341+
}while(A.isEmpty());
13001342

13011343
for(String c : getAlphabet(this)){
1344+
// select a X set for which a transition in c leads to a state in A
13021345
X = getXSet(A,c);
13031346

1304-
while(!setIntersection(X,Y).isEmpty() && !setSubtraction(Y,X).isEmpty()){
1305-
for(State s : X ){
1306-
int insert=r.nextInt(2);
1307-
if(insert > 0 && !X.equals(Y)){
1308-
Y.add(s);
1309-
}
1310-
}
1347+
// list of set Y in P such that X intersect Y != empty and Y \ X != empty
1348+
listYs = getYList(P, X);
1349+
1350+
for(HashSet<State> Y : listYs){
1351+
HashSet<State> xyintersection = setIntersection(X,Y);
1352+
HashSet<State> yxsubtraction = setSubtraction(Y,X);
13111353

1312-
if(!setIntersection(X,Y).isEmpty() && !setSubtraction(Y,X).isEmpty()){
1313-
HashSet<State> Z = new HashSet<>();
1314-
Z.addAll(setIntersection(X,Y));
1315-
Z.addAll(setSubtraction(Y,X));
1316-
1317-
for(State s: P){
1318-
if(Y.contains(s)){
1319-
P.remove(s);
1320-
P.addAll(Z);
1321-
}
1322-
}
1354+
P.remove(Y);
1355+
P.add(xyintersection);
1356+
P.add(yxsubtraction);
1357+
1358+
if(W.contains(Y)){
1359+
W.remove(Y);
1360+
W.add(xyintersection);
1361+
W.add(yxsubtraction);
13231362

1324-
if(W.contains(Y)){
1325-
for(State s: W){
1326-
if(Y.contains(s)){
1327-
W.remove(s);
1328-
W.addAll(Z);
1329-
}
1330-
}
1331-
1332-
}else{
1333-
if(setIntersection(X,Y).size() <= setSubtraction(Y,X).size()){
1334-
W.addAll(setIntersection(X,Y));
1335-
}else
1336-
W.addAll(setSubtraction(Y,X));
1337-
}
1363+
}else{
1364+
if(xyintersection.size() <= yxsubtraction.size()){
1365+
W.add(xyintersection);
1366+
}else
1367+
W.add(yxsubtraction);
13381368
}
1339-
1340-
13411369
}
1342-
1370+
13431371
}
13441372
}
13451373

1374+
// construct the minimum automata
1375+
constructMinimumAutomatonFromPartition(P);
13461376

1377+
1378+
1379+
1380+
}
1381+
1382+
private void constructMinimumAutomatonFromPartition(HashSet<HashSet<State>> P) {
1383+
HashMap<State, State> automatonStateBinding = new HashMap<>();
1384+
1385+
for(HashSet<State> macroState : P){
1386+
String macroStatename = new String("");
1387+
boolean isInitialState = false;
1388+
boolean isFinalState = false;
1389+
1390+
// get the name and the properties of the states and merge it
1391+
for(State s : macroState){
1392+
macroStatename += s.getState();
1393+
isInitialState = isInitialState || s.isInitialState();
1394+
isFinalState = isFinalState || s.isFinalState();
1395+
}
1396+
1397+
State mergedMacroState = new State(macroStatename, isInitialState, isFinalState);
1398+
1399+
for(State s : macroState)
1400+
automatonStateBinding.put(s, mergedMacroState);
1401+
1402+
}
1403+
1404+
HashSet<Transition> newDelta = new HashSet<>();
1405+
1406+
for(Transition t : this.delta){
1407+
Transition tcopy = t.clone();
1408+
tcopy.setFrom(automatonStateBinding.get(t.getFrom()));
1409+
tcopy.setTo(automatonStateBinding.get(t.getTo()));
1410+
newDelta.add(tcopy);
1411+
1412+
}
1413+
1414+
this.states = new HashSet<State>(automatonStateBinding.values());
1415+
this.delta = newDelta;
1416+
1417+
this.adjacencyList = this.computeAdjacencyList();
13471418
}
13481419

13491420
/**

src/it/univr/fsm/main/Main.java

+18
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,7 @@ public static void main(String[] args) {
2727
Automaton epsilon=Automaton.makeAutomaton("");
2828

2929

30+
/*
3031
Automaton a1= Automaton.loadAutomaton("/Users/andreaperazzoli/Desktop/SPY/java-fsm-library/automata/" + "automaton0001" );
3132
Automaton a2= Automaton.loadAutomaton("/Users/andreaperazzoli/Desktop/SPY/java-fsm-library/automata/" + "automaton0002" );
3233
@@ -69,6 +70,23 @@ public static void main(String[] args) {
6970
System.out.println(a);
7071
a.hopcroftMinimize();
7172
System.out.println(a);
73+
*/
74+
75+
Automaton a3 = Automaton.loadAutomata("/Users/andreaperazzoli/Desktop/SPY/java-fsm-library/automata/" + "automaton0003");
76+
System.out.println("a3:");
77+
System.out.println(a3);
78+
79+
Automaton a4 = Automaton.loadAutomata("/Users/andreaperazzoli/Desktop/SPY/java-fsm-library/automata/" + "automaton0004");
80+
System.out.println("a4:");
81+
System.out.println(a4);
82+
83+
Automaton a2 = Automaton.loadAutomata("/Users/andreaperazzoli/Desktop/SPY/java-fsm-library/automata/" + "automaton0002");
84+
System.out.println("a2:");
85+
System.out.println(a2);
86+
87+
Automaton a1 = Automaton.loadAutomata("/Users/andreaperazzoli/Desktop/SPY/java-fsm-library/automata/" + "automaton0001");
88+
System.out.println("a1:");
89+
System.out.println(a1);
7290

7391
}
7492

0 commit comments

Comments
 (0)