archivesOfToyLang/tshunhue/newFolder/nfatree/NFAGraph.java

506 lines
No EOL
15 KiB
Java

// package NFAGraph;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.Hashtable;
import dir.Tree;
/*
* Exception for NFA
*/
class NFAException extends Exception
{
private static final long serialVersionUID = 1L;
public NFAException(String message)
{
super(message);
}
}
/*
* structure for NFAGraph
*/
class NFAGraph{
private Integer start;
private ArrayList<Integer> endList; // List of end node num.
private ArrayList<NFAGraphPath> graphTable; // graph shown as table
public String toString(){
String returnString = "";
returnString += String.format("%d [shape=Diamond]\n" , this.start);
if (this.graphTable.length == 0){
returnString += "\n";
}
else{
for (int i=0; i < this.graphTable.size() ; i++ ){
Integer orig = this.graphTable.get(i).getOrig();
Integer dest = this.graphTable.get(i).getDest();
returnString += String.format("%d -> %d\n" , orig , dest);
}
}
String endString = "";
for (int i=0; i<this.endList.size(); i++){
endString += String.format("%d [shape=square]\n", this.endList.get(i));
}
returnString += endString;
}
public ArrayList<Integer> getEndList(){
return this.endList;
}
/**
* @return the start
*/
public Integer getStart() {
return this.start;
}
public ArrayList<NFAGraphPath> getGraphTable(){
return this.graphTable;
}
public setEndList(ArrayList<Integer> endList){
this.endList = endList;
}
public setStart(Integer startId){
this.start = startId;
}
public addGraphPath(NFAGraphPath path){
this.graphTable.add(path);
}
public static void main(String[] args) {
NFAGraph _a = new NFAGraph();
_a.start = 1;
NFAGraphPath p1 = new NFAGraphPath(1, 2, "a");
NFAGraphPath p2 = new NFAGraphPath(2, 3, "g");
NFAGraphPath p3 = new NFAGraphPath(2, 4, "c");
Tree<String> pattern = new Tree<String>("SET");
pattern.appendChildrenList(Arrays.asList("F", "G", "H"));
NFAGraphPath p4 = new NFAGraphPath(2, 5, pattern);
_a.graphTable.add(p1);
_a.graphTable.add(p2);
_a.graphTable.add(p3);
_a.graphTable.add(p4);
try{
Hashtable h = _a.nfaTableToHashMap();
int z = 1 + 1;
} catch (NFAException e) {
System.out.println(e);
}
}
public NFAGraph(int orig, int dest, NFAGraphPath path){
this.start = (Integer) orig;
this.endList = new ArrayList<Integer>(Arrays.asList((Integer) dest));
this.graphTable = new ArrayList<NFAGraphPath>(Arrays.asList(path));
}
public NFAGraph(){
this.start = (Integer)-1 ; // impossible value
this.endList = new ArrayList<Integer>();
this.graphTable = new ArrayList<NFAGraphPath>();
}
public NFAGraph(int orig, ArrayList<Integer> destList, NFAGraphPath path){
this.start = (Integer)orig;
this.endList = destList;
this.graphTable = new ArrayList<NFAGraphPath>(Arrays.asList(path));
}
/* return a Hashtable in the form of:
* Hashtable< OrigId, <Pattern, DestId>>
*
*/
Hashtable<Integer, Hashtable<Tree<String>,Integer>> nfaTableToHashMap () throws NFAException{
Hashtable<Integer, Hashtable<Tree<String>,Integer>> graphHashMap = new Hashtable();
for (int i=0;i<this.graphTable.size(); i++){
Integer orig = this.graphTable.get(i).getOrig();
Integer dest = this.graphTable.get(i).getDest();
Tree<String> pattern = this.graphTable.get(i).getPatt();
if (graphHashMap.containsKey(orig)){
if (graphHashMap.get(orig).containsKey(pattern)){
throw new NFAException(
"Each 2 links from the same link shouldn't have the same value.");
}
}else{
// graph_dict[orig] = dict()
graphHashMap.put(orig, new Hashtable());
}
graphHashMap.get(orig).put(pattern, dest);
}
return graphHashMap;
}
/*
* (value_match)
* Match a char string with char pattern string (1-char) "pattern".
*/
boolean charPatternMatch(String patt, String c) throws NFAException{
if (c.length()>1){
throw NFAException("Char or pattern input too many.");
}
else if(patt.length()>1){
throw NFAException("patt or pattern input too many.");
}
else{
if(c == patt){
return true;
}
else{
return false;
}
}
}
/*
* Match a char string with char pattern (tree) "pattern".
*/
boolean charPatternMatch(Tree<String> patt, String c) throws NFAException{
if (c.length()>1){
throw NFAException("Char input too many.");
}
else{
head = patt.getNode();
if (head == "NOT"){
if(patt.getChildrenList().size()>1){
throw NFAException("in char pattern NOT contains only 1 sub-pattern");
return not(this.charPatternMatch(patt.getChildrenList().get(0),c));
}
}
else if(patt.getNode() == "SET"){
this.charPatternMatchSet(patt, c);
}
else if (patt.getNode() == "RANGE"){
this.charPatternMatchRange(patt, c);
}
else if (patt.getNode().length() == 1){
thischarPatternMatch(patt.getNode(),c); // char-to-char matching
}
else{
throw NFAException("The form of a char pattern tree is not acceptable.");
}
}
}
/*
* check part for patt of which head node is "SET".
*/
boolean charPatternMatchSet(Tree<String> patt, String c) throws NFAException{
if (patt.getChildrenList().size() == 0){
throw NFAException("in char pattern SET contains at least 1 sub-pattern");
}else{
for (int i=0;i<patt.getChildrenList().size();i++){
if (this.charPatternMatch(patt, c)){
return true;
}
else{
return false;
}
}
}
}
/*
* check part for patt of which head node is "RANGE".
*/
boolean charPatternMatchRange(Tree<String> patt, String c) throws NFAException{
if (patt.getChildrenList().size() > 2){
throw NFAException("in char pattern RANGE contains only 2 chars");
}
else if (patt.getChildrenList().get(0).length()>1 || patt.getChildrenList().get(1).length()>1){
throw NFAException("In char pattern RANGE contains no string of which length > 1.");
}
else {
char lowerBound = patt.getChildrenList().get(0).charAt(0);
char upperBound = patt.getChildrenList().get(1).charAt(0);
if (lowerBound > upperBound){
throw NFAException("In char pattern RANGE the lower_bound must not bigger than upper bound.");
}
else{
if (c.charAt(0) >= lowerBound && c.charAt(0) <= upperBound){
return true;
}
else{
return false;
}
}
}
}
public Integer transitStrInGraph(int startPoint, String str){
Integer status = (Integer) startPoint;
Hashtable<Integer, Hashtable<Tree<String>,Integer>> nfaHashMap = this.nfaTableToHashMap();
for (int i=0;i<str.length();i++){
String c = str.substring(i,i+1);
// treeKey : the key is a tree
for(Tree<String> treeKey : nfaHashMap.get(status).keySet()){
boolean treeKeyMatched = this.charPatternMatch(treeKey, c);
if (treeKeyMatched){
status = nfaHashMap.get(status).get(treeKey);
}
}
}
return status;
}
}
/*
* Path for NFAGraph.graph
* [2]-- a ->[3]
* orig = 2;
* dest = 3;
* patt = ['a'];
*/
class NFAGraphPath {
private Integer orig;
private Integer dest;
private Tree<String> patt;
public NFAGraphPath(int ori, int des, String ptn) {
this.orig = (Integer) ori;
this.dest = (Integer) des;
this.patt = new Tree<String>(ptn);
}
public NFAGraphPath(int ori, int des, Tree<String> ptnTree) {
this.orig = (Integer) ori;
this.dest = (Integer) des;
this.patt = ptnTree;
}
Integer getOrig() {
return this.orig;
}
Integer getDest() {
return this.dest;
}
Tree<String> getPatt() {
return this.patt;
}
}
/**
* containing mainaulations for the nfa graph genarating.
*/
class NfaManipulation{
Integer id; // the id number that will be returned by nextId
public NNfaManipulation(){
this.id = (Integer) 0;
}
Integer nextId (){
Integer retId = this.id;
this.id += 1;
return retId;
}
public NFAGraph makeSimpleNFA(String pattern){
Integer start = this.nextId();
Integer end = this.nextId();
NFAGraphPath path = NFAGraphPath(start, end, pattern);
NFAGraph nfa = NFAGraph(start, end, path);
return nfa;
}
public NFAGraph makeSimpleNFA(Tree<String> pattern){
Integer start = this.nextId();
Integer end = this.nextId();
NFAGraphPath path = NFAGraphPath(start, end, pattern);
NFAGraph nfa = NFAGraph(start, end, path);
return nfa;
}
/**
* xy
*/
public nfaConcat(NFAGraph nfa1, NFAGraph nfa2){
Integer nfa1Start = nfa1.getStart();
ArrayList<Integer> nfa1EndList = nfa1.getEndList();
ArrayList<NFAGraphPath> nfa1NFAGraphList = nfa1.getGraphTable();
ArrayList<NFAGraphPath> nfa2NFAGraphList = nfa2.getGraphTable();
NFAGraph resultNFA = new NFAGraph();
resultNFA.setStart(nfa1Start);
concatNFAGraph(nfa2, nfa1EndList, nfa1NFAGraphList, nfa2NFAGraphList, resultNFA);
resultNFA.setEndList(nfa2.getEndList());
return resultNFA;
}
private void concatNFAGraph(NFAGraph nfa2, ArrayList<Integer> nfa1EndList, ArrayList<NFAGraphPath> nfa1NFAGraphList,
ArrayList<NFAGraphPath> nfa2NFAGraphList, NFAGraph resultNFA) {
for(int i=0; i<nfa1NFAGraphList.size();i++){
resultNFA.addGraphPath(nfa1NFAGraphList.get(i));
}
for(int i=0; i<nfa1EndList.size();i++){
for (int j=0; i<nfa2NFAGraphList.size();j++){
NFAGraphPath path = nfa2NFAGraphList.get(j);
if(path.getOrig() == (Integer) nfa1EndList.get(i)){
NFAGraphPath newPath =
new NFAGraphPath(nfa2.getStart(), path.getDest(), path.getPatt());
resultNFA.addGraphPath(newPath);
}
else{
resultNFA.addGraphPath(path);
}
}
}
}
/**
* x | y
*/
public nfaOr(NFAGraph nfa1, NFAGraph nfa2){
Integer nfa1Start = nfa1.getStart();
Integer nfa2Start = nfa2.getStart();
ArrayList<Integer> nfa1EndList = nfa1.getEndList();
ArrayList<Integer> nfa2EndList = nfa2.getEndList();
ArrayList<NFAGraphPath> nfa1NFAGraphList = nfa1.getGraphTable();
ArrayList<NFAGraphPath> nfa2NFAGraphList = nfa2.getGraphTable();
NFAGraph resultNFA = new NFAGraph();
resultNFA.setStart(nfa1Start);
ArrayList<Integer> NewEndList = nfa1EndList;
for(int i=0; i<nfa1NFAGraphList.size();i++){
resultNFA.addGraphPath(nfa1NFAGraphList.get(i));
}
for(int i=0;i<nfa2NFAGraphList.size();i++){
NFAGraphPath path = nfa2NFAGraphList.get(i);
if(path.getOrig() == nfa2Start){
NFAGraphPath newPath = new NFAGraphPath(nfa1Start, path.getDest(), path.getPatt());
resultNFA.addGraphPath(newPath);
}
else{
resultNFA.addGraphPath(path);
}
}
for (int i=0;i<nfa2EndList.size();i++){
if (nfa2EndList.get(i) == nfa2Start){
NewEndList.add(nfa1Start);
}
else{
NewEndList.add(nfa2EndList.get(i));
}
}
resultNFA.setEndList(NewEndList);
return resultNFA;
}
/**
* xy?
*/
public NFAGraph nfaOnceOrNone(NFAGraph nfa1, NFAGraph nfa2){
NFAGraph resultNFA = new NFAGraph();
resultNFA.setStart(nfa1.getStart());
ArrayList<Integer> resultNFAEndList = nfa1.getEndList();
for (int i=0; i<nfa1.getGraphTable().size();i++){
resultNFA.addGraphPath(nfa1.getGraphTable.get(i));
}
for(int i=0; i<nfa2.getEndList().size();i++){
Integer item = nfa2.getEndList().get(i);
if(item != nfa2.getStart()){
resultNFAEndList.add(item);
}
}
resultNFA.setEndList(resultNFAEndList);
concatNFAGraph(nfa2, nfa1.getEndList(), nfa1.getGraphTable(), nfa2.getGraphTable(), resultNFA);
return resultNFA;
}
/**
* xy*
*/
public NFAGraph nfaRepeatOrNone(NFAGraph nfa1, NFAGraph nfa2){
NFAGraph resultNFA = new NFAGraph();
resultNFA.setEndList(nfa1.getEndList());
resultNFA.setStart(nfa1.getStart());
Integer nfa2Start = nfa2.getStart();
ArrayList<Integer> nfa1EndList = nfa1.getEndList();
ArrayList<Integer> nfa2EndList = nfa2.getEndList();
for(int i=0;i<nfa1.getGraphTable().size();i++){
resultNFA.addGraphPath(nfa1.getGraphTable().get(i));
}
for(int i=0;i<nfa1EndList.size();i++){
Integer currentNfa1End = nfa1EndList.get(i);
for(int j=0;j<nfa2.getGraphTable().size();j++){
NFAGraphPath path = nfa2.getGraphTable().get(j);
Integer pathOrig = path.getOrig();
Integer pathDest = path.getDest();
if(pathOrig == nfa2.getStart()){
pathOrig = currentNfa1End;
}
if(nfa2EndList.contains(pathDest)){
pathDest = currentNfa1End;
}
path = new NFAGraphPath((int)pathOrig,(int) pathDest, path.getPatt());
if (!resultNFA.getGraphTable().contains(path)){
resultNFA.addGraphPath(path);
}
}
}
return resultNFA;
}
}
/*
*/