Обработка мягких ограничений для расписания университета (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;
}
эта функция может обрабатывать первое и второе мягкие ограничения для небольшого набора данных, но эта функция занимает слишком много времени для средних и больших наборов данных, и причина, по-моему, из-за глубокого копирования.