@@ -593,7 +593,7 @@ public static Automaton loadAutomata(String path){
593
593
}
594
594
595
595
Automaton a = new Automaton (initialStates ,delta ,states );
596
- a .minimize ();
596
+ a .hopcroftMinimize ();
597
597
return a ;
598
598
}
599
599
@@ -1206,9 +1206,9 @@ public static HashSet<String> getAlphabet(Automaton a){
1206
1206
return alphabet ;
1207
1207
}
1208
1208
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 ) ){
1212
1212
return t .getTo ();
1213
1213
}
1214
1214
}
@@ -1219,15 +1219,15 @@ public void hopcroftremoveUnreachableStates(){
1219
1219
HashSet <State > unreachableStates = new HashSet <>();
1220
1220
HashSet <State > reachableStates = (HashSet <State >) this .getInitialStates ().clone ();
1221
1221
HashSet <State > newStates = (HashSet <State >) this .getInitialStates ().clone ();
1222
+ HashSet <Transition > transitionstoRemove = new HashSet <>();
1222
1223
HashSet <State > temp ;
1223
1224
1224
1225
1225
1226
do {
1226
1227
temp = new HashSet <>(Collections .<State >emptySet ());
1227
1228
for (State s : newStates ){
1228
1229
for (String a : getAlphabet (this )){
1229
- HashSet <Transition > transition = this .getOutgoingTransitionsFrom (s );
1230
- State to = getOutgoingStatefromTransitionSymbol (transition , a );
1230
+ State to = getOutgoingStatefromTransitionSymbol (s , a );
1231
1231
if (to != null ) temp .add (to );
1232
1232
}
1233
1233
}
@@ -1244,32 +1244,64 @@ public void hopcroftremoveUnreachableStates(){
1244
1244
1245
1245
states .removeAll (unreachableStates );
1246
1246
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
+
1247
1256
}
1248
1257
1249
- private HashSet <State > getXSet (HashSet <State > A , String symbol ){
1258
+ private HashSet <State > getXSet (HashSet <State > A , String c ){
1250
1259
HashSet <State > s = new HashSet <State >();
1251
1260
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 ());
1254
1263
}
1255
1264
}
1256
1265
return s ;
1257
1266
}
1258
1267
1259
1268
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 ){
1261
1293
HashSet <State > intersection = new HashSet <State >();
1262
1294
for (State s1 : first )
1263
1295
for (State s2 : second ){
1264
1296
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 () );
1267
1299
}
1268
1300
}
1269
1301
return intersection ;
1270
1302
}
1271
1303
1272
- private Set <State > setSubtraction (HashSet <State > first , HashSet <State > second ){
1304
+ private HashSet <State > setSubtraction (HashSet <State > first , HashSet <State > second ){
1273
1305
HashSet <State > firstCopy = (HashSet <State >) first .clone ();
1274
1306
1275
1307
for (State s : second ){
@@ -1281,69 +1313,108 @@ private Set<State> setSubtraction(HashSet<State> first, HashSet<State> second){
1281
1313
1282
1314
public void hopcroftMinimize (){
1283
1315
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
+
1286
1326
HashSet <State > A = new HashSet <>();
1287
1327
HashSet <State > X ;
1288
- HashSet <State > Y = new HashSet <>() ;
1328
+ LinkedList < HashSet <State >> listYs ;
1289
1329
Random r =new Random ();
1290
1330
1291
1331
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 ;
1298
1339
}
1299
- }
1340
+ W .remove (A );
1341
+ }while (A .isEmpty ());
1300
1342
1301
1343
for (String c : getAlphabet (this )){
1344
+ // select a X set for which a transition in c leads to a state in A
1302
1345
X = getXSet (A ,c );
1303
1346
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 );
1311
1353
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 );
1323
1362
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 );
1338
1368
}
1339
-
1340
-
1341
1369
}
1342
-
1370
+
1343
1371
}
1344
1372
}
1345
1373
1374
+ // construct the minimum automata
1375
+ constructMinimumAutomatonFromPartition (P );
1346
1376
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 ();
1347
1418
}
1348
1419
1349
1420
/**
0 commit comments