Обработка мягких ограничений для расписания университета (JAVA) Генетический алгоритм

В настоящее время я работаю над своим дипломным проектом, который решает проблему с расписанием университетских курсов с использованием генетического алгоритма. В этом проекте я использовал метаэвристические сетевые наборы данных, которые содержат несколько жестких и мягких ограничений. моя программа была в состоянии справиться со всеми жесткими ограничениями, и я использовал локальный оператор перекрестного поиска, чтобы удовлетворить все жесткие ограничения. теперь у меня есть проблема обработки мягких ограничений:

у ученика есть класс в последнем слоте дня. студент имеет более двух классов подряд. у студента есть один класс в день.

Во-первых, у меня проблема с глубоким копированием объекта расписания, содержащего другие объекты, и я использовал метод сериализации, но он выдает ошибку переполнения стека со средними и большими наборами данных. во-вторых, я хотел бы спросить, каков наилучший алгоритм или метод поиска для обработки мягких ограничений.

это моя первая функция для подсчета количества нарушений мягких ограничений

public int CalcViolations() {
    violation = 0;
    int consecutiveClasses;
    boolean oneClassPenalty;
    int studentNo=data.getStudents().size();
    for(int i=0;i<studentNo;i++){
        int stCoSize=data.getStudents().get(i).getCourseRecord().size();
        consecutiveClasses = 0;
        for(int j=0;j<stCoSize;j++) {
            if (data.getStudents().get(i).getCourseRecord().get(j).getMeetingTime().getId() % 9 == 0) {
                violation++;
            }
            for (int n = 0; n < stCoSize; n++) {
                    if (n != j) {
                        if (data.getStudents().get(i).getCourseRecord().get(j).getMeetingTime().getId() % 9 != 0 && data.getStudents().get(i).getCourseRecord().get(n).getMeetingTime().getId() % 9 != 0) {
                        if (Math.abs((data.getStudents().get(i).getCourseRecord().get(j).getMeetingTime().getId()) - (data.getStudents().get(i).getCourseRecord().get(n).getMeetingTime().getId())) == 1) {
                            consecutiveClasses++;
                        }
                    }
                }
            }
            if (consecutiveClasses > 2) {
                violation++;
            }
        }
        for(int j=0;j<data.getStudents().get(i).getCourseRecord().size();j++){
            oneClassPenalty=true;
            if(j==0){
                if(data.getStudents().get(i).getCourseRecord().get(j).getMeetingTime().getDay()==data.getStudents().get(i).getCourseRecord().get(j+1).getMeetingTime().getDay())
                    oneClassPenalty=false;
            }
            if(j>0){
                if((data.getStudents().get(i).getCourseRecord().get(j).getMeetingTime().getDay()==data.getStudents().get(i).getCourseRecord().get(j-1).getMeetingTime().getDay()))
                    oneClassPenalty=false;
                if(j+1 < data.getStudents().get(i).getCourseRecord().size()){
                    if((data.getStudents().get(i).getCourseRecord().get(j).getMeetingTime().getDay()==data.getStudents().get(i).getCourseRecord().get(j+1).getMeetingTime().getDay())) oneClassPenalty=false;
                }
            }
            if(oneClassPenalty) {
                violation++;
            }
        }
    }
    return violation;

}

и вот метод, который используется для обработки нарушений мягких ограничений

public Schedule SoftConstraints(Schedule s) {
    int consecutiveClasses;
    boolean oneClassPenalty;
    for(int i=0;i<data.getStudents().size();i++){
        consecutiveClasses = 0;
        for(int j=0;j<data.getStudents().get(i).getCourseRecord().size();j++) {
            for (int n = 0; n < data.getStudents().get(i).getCourseRecord().size(); n++) {
                if (n != j) {
                    if (data.getStudents().get(i).getCourseRecord().get(j).getMeetingTime().getId() % 9 != 0 && data.getStudents().get(i).getCourseRecord().get(n).getMeetingTime().getId() % 9 != 0) {
                        if (Math.abs((data.getStudents().get(i).getCourseRecord().get(j).getMeetingTime().getId()) - (data.getStudents().get(i).getCourseRecord().get(n).getMeetingTime().getId())) == 1) {
                            consecutiveClasses++;
                            if (consecutiveClasses > 2) {
                                int c = s.getClasses().indexOf(data.getStudents().get(i).getCourseRecord().get(j));
                                s=shuffle(s, c);
                            }
                        }
                    }
                }
                if (data.getStudents().get(i).getCourseRecord().get(j).getMeetingTime().getId() % 9 == 0) {
                    int c = s.getClasses().indexOf(data.getStudents().get(i).getCourseRecord().get(j));
                    s=shuffle(s, c);
                }
            }
        }
        for(int j=0;j<data.getStudents().get(i).getCourseRecord().size();j++){
            oneClassPenalty=true;
            if(j==0){
                if(data.getStudents().get(i).getCourseRecord().get(j).getMeetingTime().getDay()==data.getStudents().get(i).getCourseRecord().get(j+1).getMeetingTime().getDay())
                    oneClassPenalty=false;
            }
            if( j > 0){
                if((data.getStudents().get(i).getCourseRecord().get(j).getMeetingTime().getDay()==data.getStudents().get(i).getCourseRecord().get(j-1).getMeetingTime().getDay()))
                    oneClassPenalty=false;
                if(j+1<data.getStudents().get(i).getCourseRecord().size()){
                    if((data.getStudents().get(i).getCourseRecord().get(j).getMeetingTime().getDay()==data.getStudents().get(i).getCourseRecord().get(j+1).getMeetingTime().getDay())) oneClassPenalty = false;
                }
            }
            if(oneClassPenalty) {
                int c = s.getClasses().indexOf(data.getStudents().get(i).getCourseRecord().get(j));
                    shuffle(s,c);
            }
        }
    }
    return s;
}

private ArrayList<MeetingTime> getStudentOccupyTime(Schedule s, Class c){
    ArrayList<MeetingTime> MT=new ArrayList<>();
    for(int i=0;i<c.getCourse().getStudent().size();i++){
        for(int j=0;j<s.getClasses().size();j++){
            for(int n=0;n<s.getClasses().get(j).getCourse().getStudent().size();n++){
                if(s.getClasses().get(j).getCourse().getStudent().get(n)==c.getCourse().getStudent().get(i)){
                    MT.add(s.getClasses().get(j).getMeetingTime());
                }
            }
        }
    }
    Set<MeetingTime> removeDups = new LinkedHashSet<>(MT);
    MT.clear();
    MT.addAll(removeDups);
    return MT;
}

private ArrayList<MeetingTime> getStudentFreeTime(ArrayList<MeetingTime> MT){
    ArrayList<MeetingTime> studentFreeTime=new ArrayList<>();
    for(int i=0;i<data.getMeetingTimes().size();i++){
        if(!MT.contains(data.getMeetingTimes().get(i))) {
            studentFreeTime.add(data.getMeetingTimes().get(i));
        }
    }
    return studentFreeTime;
}

private ArrayList<Room> ChangeRoomAvailability(Schedule s,int c){
    Room currentRoom=s.getClasses().get(c).getRoom();
    ArrayList<Room> AvailableRooms=new ArrayList<>();
    AvailableRooms.clear();
    for(int r=0;r<data.getRooms().size();r++) {
        if ((data.getRooms().get(r).getFeatures().containsAll(s.getClasses().get(c).getCourse().getFeatures())) && (data.getRooms().get(r).getSeatingCapacity() >= s.getClasses().get(c).getCourse().getMaxNumbOfStudents())) {
            if (data.getRooms().get(r) != currentRoom)
                if(checkAvailability(s,data.getRooms().get(r))) AvailableRooms.add(data.getRooms().get(r));
        }
    }
    return AvailableRooms;
}

private ArrayList<MeetingTime> checkRoomAvailableTime(Schedule s, Room currentRoom){//Room currentRoom=r;
    ArrayList<MeetingTime> RoomOccupyTime=new ArrayList<>();
    ArrayList<MeetingTime> roomAvailableTime=new ArrayList<>();
    roomAvailableTime.clear();
    for (int i = 0; i < s.getClasses().size(); i++) {
        if (s.getClasses().get(i).getRoom() == currentRoom) {
            RoomOccupyTime.add(s.getClasses().get(i).getMeetingTime());
        }
    }
    Set<MeetingTime> removeDups = new LinkedHashSet<>(RoomOccupyTime);
    RoomOccupyTime.clear();
    RoomOccupyTime.addAll(removeDups);
    if(RoomOccupyTime.containsAll(data.getMeetingTimes())) return roomAvailableTime;
    else{
        for(int i=0;i<data.getMeetingTimes().size();i++){
            if(!RoomOccupyTime.contains(data.getMeetingTimes().get(i))){
                roomAvailableTime.add(data.getMeetingTimes().get(i));
            }
        }
        return roomAvailableTime;
    }
}

private Schedule shuffle(Schedule s, int c)  {
    int currentVio=s.getViolation();
    Boolean status=false;
    MeetingTime rand;
    ArrayList<Room> candidateRooms = new ArrayList<>();
    Room newRoom=null;
    ArrayList<MeetingTime> checkedTime = new ArrayList<>();
    ArrayList<MeetingTime> MT;
    ArrayList<MeetingTime> studentAvailableTime;
    ArrayList<MeetingTime> roomAvailableTime;
    ArrayList<MeetingTime> AvailableTime=new ArrayList<>();

    MT=getStudentOccupyTime(s,s.getClasses().get(c));
    studentAvailableTime=getStudentFreeTime(MT);
    roomAvailableTime=checkRoomAvailableTime(s,s.getClasses().get(c).getRoom());
    AvailableTime.clear();
    if(roomAvailableTime.size()>0){
        for (MeetingTime aStudentAvailableTime : studentAvailableTime) {
            if (roomAvailableTime.contains(aStudentAvailableTime))
                AvailableTime.add(aStudentAvailableTime);
        }
    }
    checkedTime.clear();
    if(AvailableTime.size()>0) {
        while (!status ) {
            rand = AvailableTime.get((int) (AvailableTime.size() * Math.random()));
            while (checkedTime.contains(rand)&&checkedTime.size()!=AvailableTime.size()){
                rand = AvailableTime.get((int) (AvailableTime.size() * Math.random()));
            }
            s.getClasses().get(c).setMeetingTime(rand);
            if(!checkedTime.contains(rand)) checkedTime.add(rand);
            if(s.getViolation()<currentVio && s.getFitness()==1.0){
                status=true;
                break;
            }
            if(checkedTime.size()==AvailableTime.size())break;
        }

    }
    if(status){
        return s;
    }
    else ShuffleCourseTime( s, c);
    if(s.getViolation()<currentVio){
        return s;
    }
    else {
        candidateRooms.clear();
        candidateRooms=ChangeRoomAvailability(s,c);
        if(candidateRooms.size()>0){
            newRoom=checkSwapping(s,c,candidateRooms);
        }
        if(candidateRooms.size()>0 && newRoom!=null){
            shuffle(s,c,newRoom);
        }
        return s;
    }
}
private Schedule shuffle(Schedule s, int c, Room r){
    s.getClasses().get(c).setRoom(r);
    MeetingTime rand;
    ArrayList<MeetingTime> MT;
    ArrayList<MeetingTime> studentAvailableTime;
    ArrayList<MeetingTime> roomAvailableTime;
    ArrayList<MeetingTime> AvailableTime=new ArrayList<>();
    MT=getStudentOccupyTime(s,s.getClasses().get(c));
    studentAvailableTime=getStudentFreeTime(MT);
    roomAvailableTime=checkRoomAvailableTime(s,r);
    if(roomAvailableTime.size()>0){
        for (MeetingTime aStudentAvailableTime : studentAvailableTime) {
            if (roomAvailableTime.contains(aStudentAvailableTime))
                AvailableTime.add(aStudentAvailableTime);
        }
    }
    if(AvailableTime.size()>0) {
        rand = AvailableTime.get((int) (AvailableTime.size() * Math.random()));
        s.getClasses().get(c).setMeetingTime(rand);
    }
    return s;
}
private Room checkSwapping(Schedule s,int c,ArrayList<Room> availableRooms){
    ArrayList<MeetingTime> MT;
    ArrayList<MeetingTime> studentAvailableTime;
    Room randomRoom;
    Room NewRoom=null;
    ArrayList<MeetingTime> meetingTime=new ArrayList<>();
    ArrayList<Room> checkRoom=new ArrayList<>();
    ArrayList<MeetingTime> availableTime=new ArrayList<>();
    Boolean status=false;
    MT=getStudentOccupyTime(s,s.getClasses().get(c));
    studentAvailableTime=getStudentFreeTime(MT);
    meetingTime.clear();
    availableTime.clear();
    while (checkRoom.size()!=availableRooms.size()){
        randomRoom=availableRooms.get((int) (availableRooms.size()*Math.random()));
        meetingTime = checkRoomAvailableTime(s, randomRoom);
        for (MeetingTime aStudentAvailableTime : studentAvailableTime) {
            if (meetingTime.contains(aStudentAvailableTime)) {
                availableTime.add(aStudentAvailableTime);
            }
        }
        if(!checkRoom.contains(randomRoom)) checkRoom.add(randomRoom);
        if(availableTime.size()>0){
            NewRoom=randomRoom;
            status=true;
            break;
        }
        if(checkRoom.size()==availableRooms.size()) break;
    }
    if(status) return NewRoom;
    else    return null;
}

public Schedule Serialization(Schedule s){
    Object obj = null;
    try {
        FastByteArrayOutputStream fbos =
                new FastByteArrayOutputStream();
        ObjectOutputStream out = new ObjectOutputStream(fbos);
        out.writeObject(s);
        out.flush();
        out.close();
        ObjectInputStream in =
                new ObjectInputStream(fbos.getInputStream());
        obj = in.readObject();
    }
    catch(IOException e) {
        e.printStackTrace();
    }
    catch(ClassNotFoundException cnfe) {
        cnfe.printStackTrace();
    }
    return (Schedule) obj;

}

public Schedule ShuffleCourseTime(Schedule s,int c){
    int currentViolation=s.getViolation();
    ArrayList<Class> checkedClass=new ArrayList<>();
    Boolean Status=false;
    Room currentRoom=s.getClasses().get(c).getRoom();
    ArrayList<Class> roomClasses=new ArrayList<>();
    for(int i=0;i<s.getClasses().size();i++){
        if(s.getClasses().get(i).getRoom()==currentRoom&&s.getClasses().get(i)!=s.getClasses().get(c)) roomClasses.add(s.getClasses().get(i));
    }
    while (!Status){
        if(roomClasses.size()==0)break;
        MeetingTime T1=s.getClasses().get(c).getMeetingTime();
        Class c2=roomClasses.get((int) (Math.random() * roomClasses.size()));
        while (checkedClass.contains(c2)&&checkedClass.size()!=roomClasses.size()) {
            c2 = roomClasses.get((int) (Math.random() * roomClasses.size()));
        }
        MeetingTime T2=c2.getMeetingTime();
        s.getClasses().get(c).setMeetingTime(data.getMeetingTimes().get((int)(data.getMeetingTimes().size()*Math.random())));
        s.getClasses().get(s.getClasses().indexOf(c2)).setMeetingTime(data.getMeetingTimes().get((int)(data.getMeetingTimes().size()*Math.random())));
        if(s.getFitness()==1){
            if(s.getViolation()<currentViolation){
                Status=true;
                break;
            }
        }
        s.getClasses().get(c).setMeetingTime(T1);
        s.getClasses().get(s.getClasses().indexOf(c2)).setMeetingTime(T2);
        if(!checkedClass.contains(c2))checkedClass.add(c2);
        if (checkedClass.size()==roomClasses.size())break;
    }
    if(Status)return s;
    else return s;

}

эта функция может обрабатывать первое и второе мягкие ограничения для небольшого набора данных, но эта функция занимает слишком много времени для средних и больших наборов данных, и причина, по-моему, из-за глубокого копирования.

0 ответов

Другие вопросы по тегам