Шаблон для повторной фиксации Hibernate после исключения StaleObjectStateException

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

К сожалению, то, что я сделал до сих пор, кажется неоптимальным и содержит то, что, как я полагаю, содержит много кода, что означает, что другие разработчики вряд ли будут использовать его в случае необходимости. Как улучшить нижеприведенный пример ExampleRetryableUpdate, особенно для исполняемого файла Update, в котором есть множество полей для одновременного обновления?

public class ExampleRetryableUpdate extends SafeRetryableUpdateRunnable<Model>{
    private Long id;
    private String originalState;
    private String targetState;

    public ExampleRetryableUpdate(Model targetState, Model currentState){
        this.originalState = currentState.getFieldToUpdate();
        this.targetState = targetState.getFieldToUpdate();
    }

    @Override
    boolean safeToUpdate(Model current) {
        return originalState.equals(current.getFieldToUpdate());
    }

    @Override
    String getStaleFields(Model current) {
        StringBuilder sb = new StringBuilder();
        if(!this.originalState.equals(current.getFieldToUpdate())){
            sb.append("FieldToUpdate differs: Stale - ");
            sb.append(this.originalState);
            sb.append(", Current - ");
            sb.append(current.getFieldToUpdate());
        }
        //Repeatable for each relevant field
        return sb.toString();
    }

    @Override
    public void run() {
        try{
            Model current = Model.findById(id);
            if(safeToUpdate(current)){
                current.setFieldToUpdate(targetState);
                Model.persist(current);
            } else{
                throw new UnsafeRetryException(getStaleFields(current));
            }
        } catch (StaleObjectStateException ex){
            System.out.println("Failed due to StaleObject exception, may retry.");
            throw ex;
        }

    }
}

Контекстные классы:

public abstract class SafeRetryableUpdateRunnable<T> implements Runnable {
    abstract boolean safeToUpdate(T Model);
    abstract String getStaleFields(T current);
}

public class Model {
    private Long id;
    private String fieldToUpdate;
    private String irrelevantField;
    private Integer oca;

    public String getFieldToUpdate() {
        return fieldToUpdate;
    }

    public void setFieldToUpdate(String fieldToUpdate) {
        this.fieldToUpdate = fieldToUpdate;
    }
...
}

public class BusinessLogicClass {
    public boolean someMethod() throws InterruptedException {
        //Business logic
        //...
        //Get current from Db, create a copy and alter fieldToUpdate
        //...
        ExampleRetryableUpdate runnable = new ExampleRetryableUpdate(targetState, currentState);
        int i = 0;
        do{
            try{
                runnable.run();
                return true;
            } catch(StaleObjectStateException ex){
                System.out.println("Expected, potentially retryable.");
                Thread.sleep(1000L);
            } catch(UnsafeRetryException ex){
                System.out.println("Field has been updated by an alternate thread/server. Aborting.");
                return false;
            }
        }while (i++ <= 3);
        return false;
    }
}

0 ответов

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