Кешируются ли сущности в jpa по умолчанию? - PullRequest
17 голосов
/ 25 апреля 2010

Я добавляю сущность в свою базу данных, и она отлично работает.Но когда я получаю список, я получаю старую сущность, новые добавленные сущности не отображаются, пока я не разверну приложение и снова разверну его.Это означает, что мои объекты кэшируются по умолчанию?Но я не установил никаких параметров для кэширования сущностей в моем файле persistence.xml или в любом таком файле.

Я даже пытался вызвать flush (), refresh () и merge ().Но все же это показывает только старые сущности.Я что-то пропустил?Пожалуйста, помогите мне.

Ответы [ 5 ]

19 голосов
/ 25 апреля 2010

Добро пожаловать в JPA. Если вы используете его, это означает, что у вас будут огромные проблемы, если вы обновите базу данных за пределами JPA, если вы не знаете, что делаете, и очень осторожны. Это означает, что вам нужно выяснить, как очистить любые кэшированные объекты, чтобы их можно было перезагрузить.

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

12 голосов
/ 26 апреля 2010

Это означает, что мои объекты кэшируются по умолчанию?

JPA 1.0 не определяет кэш L2 («разделяемый кэш»), JPA 1.0 определяет только L1 кэш («транзакционный кеш»), но поставщики JPA могут поддерживать общий кеш объекта, и большинство делают. Это случай TopLink Essentials, который поддерживает кэш L1 и L2 с помощью JPA Extensions для кэширования (для JVM).

Теперь, как объяснено в большой статье Понимание кеша TopLink Essentials (GlassFish JPA) :

  • Все EntityManager из одного и того же модуля персистентности совместно используют кэш сеанса (так TopLink вызывает кэш 2-го уровня).
  • Кэш сеанса включен по умолчанию.
  • Если есть изменения / удаления объектов в контексте постоянства, они синхронизируются с кешем сеанса после фиксации транзакции, поэтому состояние кеша сеанса обновляется (или такой кеш не будет использоваться на всех).

Так что должно быть что-то еще не так с вашей настройкой. Вы можете try отключить общий кэш сеанса для целей тестирования (и только для целей тестирования), добавив следующее свойство:

<property name="toplink.cache.shared.default" value="false"/>

Но я был бы удивлен, если это что-то изменит. Как я уже сказал, я думаю, что где-то есть другая проблема.

PS: это не отвечает на вопрос, но, если вы используете GlassFish v3, почему бы вам не использовать EclipseLink?

Обновление: ответ на комментарий ОП

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

Что ж, если вы не создадите ссылку между сущностями на уровне Java, JPA не сможет создать ее в базе данных (JPA делает только то, что вы ему сказали). Итак, да , вам нужно создать ссылку и, в случае двунаправленной связи, вам даже нужно установить обе стороны ссылки (например, добавить employee к коллекции сотрудников на Department и установите department для Employee).

1 голос
/ 28 июня 2012

JPA 2.0 определяет общий (L2) кэш, но не определяет значение по умолчанию. EclipseLink по умолчанию разрешает кэширование, другие провайдеры - нет.

EntityManager всегда будет иметь кэш постоянного уровня (L1), пока вы не вызовете clear () или не создадите новый.

Вы можете отключить общий кеш,

См. http://wiki.eclipse.org/EclipseLink/UserGuide/JPA/Basic_JPA_Development/Caching

Но ваша проблема в том, что вы не поддерживаете обе стороны своих отношений. Когда вы устанавливаете 1-1, вы должны добавить к 1-м, иначе ваши объекты будут недействительными.

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

http://wiki.eclipse.org/EclipseLink/UserGuide/JPA/Basic_JPA_Development/Caching

0 голосов
/ 14 июня 2017

Я использую контекст eclipselink и ejb , я обнаружил, что когда я запрашиваю объекты, он будет автоматически кэшироваться, но я также запускаю функцию для изменения записи в базе данных поэтому я могу получить старые данные в кеше, поэтому я отключаю кеш, добавив в файл следующее: persistence.xml:

<shared-cache-mode>NONE</shared-cache-mode>

это работает!

0 голосов
/ 28 июня 2012

Вы используете EntityManager? если нет, попробуйте http://docs.oracle.com/javaee/5/api/javax/persistence/EntityManager.html

Используете ли вы источник данных для управления подключениями к вашей базе данных? Вы должны попробовать это тоже. Это XML, который настроен на вашем сервере. Дайте больше информации о вашей архитектуре, чтобы мы могли вам помочь.

...