/* * Copyright 2012 Red Hat, Inc. and/or its affiliates. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package org.optaplanner.core.impl.heuristic.selector.entity; import java.util.Iterator; import java.util.List; import java.util.ListIterator; import org.optaplanner.core.config.heuristic.selector.common.SelectionCacheType; import org.optaplanner.core.impl.domain.entity.descriptor.EntityDescriptor; import org.optaplanner.core.impl.heuristic.selector.common.iterator.CachedListRandomIterator; import org.optaplanner.core.impl.phase.scope.AbstractPhaseScope; import org.optaplanner.core.impl.phase.scope.AbstractStepScope; import org.optaplanner.core.impl.score.director.InnerScoreDirector; /** * This is the common {@link EntitySelector} implementation. */ public class FromSolutionEntitySelector extends AbstractEntitySelector { protected final EntityDescriptor entityDescriptor; protected final SelectionCacheType minimumCacheType; protected final boolean randomSelection; protected List<Object> cachedEntityList = null; protected Long cachedEntityListRevision = null; protected boolean cachedEntityListIsDirty = false; public FromSolutionEntitySelector(EntityDescriptor entityDescriptor, SelectionCacheType minimumCacheType, boolean randomSelection) { this.entityDescriptor = entityDescriptor; this.minimumCacheType = minimumCacheType; this.randomSelection = randomSelection; } @Override public EntityDescriptor getEntityDescriptor() { return entityDescriptor; } /** * @return never null, at least {@link SelectionCacheType#STEP} */ @Override public SelectionCacheType getCacheType() { SelectionCacheType intrinsicCacheType = SelectionCacheType.STEP; return (intrinsicCacheType.compareTo(minimumCacheType) > 0) ? intrinsicCacheType : minimumCacheType; } // ************************************************************************ // Cache lifecycle methods // ************************************************************************ @Override public void phaseStarted(AbstractPhaseScope phaseScope) { super.phaseStarted(phaseScope); InnerScoreDirector scoreDirector = phaseScope.getScoreDirector(); cachedEntityList = entityDescriptor.extractEntities(scoreDirector.getWorkingSolution()); cachedEntityListRevision = scoreDirector.getWorkingEntityListRevision(); cachedEntityListIsDirty = false; } @Override public void stepStarted(AbstractStepScope stepScope) { super.stepStarted(stepScope); InnerScoreDirector scoreDirector = stepScope.getScoreDirector(); if (scoreDirector.isWorkingEntityListDirty(cachedEntityListRevision)) { if (minimumCacheType.compareTo(SelectionCacheType.STEP) > 0) { cachedEntityListIsDirty = true; } else { cachedEntityList = entityDescriptor.extractEntities(scoreDirector.getWorkingSolution()); cachedEntityListRevision = scoreDirector.getWorkingEntityListRevision(); } } } @Override public void phaseEnded(AbstractPhaseScope phaseScope) { super.phaseEnded(phaseScope); cachedEntityList = null; cachedEntityListRevision = null; cachedEntityListIsDirty = false; } // ************************************************************************ // Worker methods // ************************************************************************ @Override public boolean isCountable() { return true; } @Override public boolean isNeverEnding() { // CachedListRandomIterator is neverEnding return randomSelection; } @Override public long getSize() { return (long) cachedEntityList.size(); } @Override public Iterator<Object> iterator() { checkCachedEntityListIsDirty(); if (!randomSelection) { return cachedEntityList.iterator(); } else { return new CachedListRandomIterator<>(cachedEntityList, workingRandom); } } @Override public ListIterator<Object> listIterator() { checkCachedEntityListIsDirty(); if (!randomSelection) { return cachedEntityList.listIterator(); } else { throw new IllegalStateException("The selector (" + this + ") does not support a ListIterator with randomSelection (" + randomSelection + ")."); } } @Override public ListIterator<Object> listIterator(int index) { checkCachedEntityListIsDirty(); if (!randomSelection) { return cachedEntityList.listIterator(index); } else { throw new IllegalStateException("The selector (" + this + ") does not support a ListIterator with randomSelection (" + randomSelection + ")."); } } @Override public Iterator<Object> endingIterator() { checkCachedEntityListIsDirty(); return cachedEntityList.iterator(); } private void checkCachedEntityListIsDirty() { if (cachedEntityListIsDirty) { throw new IllegalStateException("The selector (" + this + ") with minimumCacheType (" + minimumCacheType + ")'s workingEntityList became dirty between steps but is still used afterwards."); } } @Override public String toString() { return getClass().getSimpleName() + "(" + entityDescriptor.getEntityClass().getSimpleName() + ")"; } }