Spring:Datenbanken: Unterschied zwischen den Versionen
| Admin (Diskussion | Beiträge) | Admin (Diskussion | Beiträge)  | ||
| Zeile 1: | Zeile 1: | ||
| == Codesnippets == | |||
| <syntaxhighlight lang="java" line="1"> | |||
|     @Autowired | |||
|     private EntityManagerFactory entityManagerFactory; | |||
|     public Session getSession() | |||
|     { | |||
|         SessionFactory sessionFactory = entityManagerFactory.unwrap(SessionFactory.class); | |||
|         Session session = sessionFactory.openSession(); | |||
|         return session; | |||
|     } | |||
|     @Test | |||
|     void whenCreateOneRule_thenItShouldBePossible() | |||
|     { | |||
|         Transaction tx = getSession().beginTransaction(); | |||
|         // ... | |||
|         tx.commit(); | |||
|     } | |||
| </syntaxhighlight> | |||
| == Set Up or Tear Down Outside of a Transaction<ref>Marco Behler, [https://www.marcobehler.com/guides/spring-transaction-management-transactional-in-depth "Spring Transaction Management: @Transactional In-Depth"]</ref> == | == Set Up or Tear Down Outside of a Transaction<ref>Marco Behler, [https://www.marcobehler.com/guides/spring-transaction-management-transactional-in-depth "Spring Transaction Management: @Transactional In-Depth"]</ref> == | ||
| You can annotate a <code>public void</code> method with <code>@BeforeTransaction</code> or <code>@AfterTransaction</code>. This indicates to Spring that this method should be run before or after a test method is run within a transaction. | You can annotate a <code>public void</code> method with <code>@BeforeTransaction</code> or <code>@AfterTransaction</code>. This indicates to Spring that this method should be run before or after a test method is run within a transaction. | ||
Aktuelle Version vom 16. Mai 2023, 07:49 Uhr
Codesnippets
    @Autowired
    private EntityManagerFactory entityManagerFactory;
    
    public Session getSession()
    {
        SessionFactory sessionFactory = entityManagerFactory.unwrap(SessionFactory.class);
        Session session = sessionFactory.openSession();
        return session;
    }
    
    @Test
    void whenCreateOneRule_thenItShouldBePossible()
    {
        Transaction tx = getSession().beginTransaction();
        // ...
        tx.commit();
    }
Set Up or Tear Down Outside of a Transaction[1]
You can annotate a public void method with @BeforeTransaction or @AfterTransaction. This indicates to Spring that this method should be run before or after a test method is run within a transaction.
CGlib & JDK Proxies - @Transactional under the covers
Spring cannot really rewrite your Java class, like I did above, to insert the connection code (unless you are using advanced techniques like bytecode weaving, but we are ignoring that for now).
Your registerUser() method really just calls userDao.save(user), there’s no way to change that on the fly.
But Spring has an advantage. At its core, it is an IoC container. It instantiates a UserService for you and makes sure to autowire that UserService into any other bean that needs a UserService.
Now whenever you are using @Transactional on a bean, Spring uses a tiny trick. It does not just instantiate a UserService, but also a transactional proxy of that UserService.
It does that through a method called proxy-through-subclassing with the help of the Cglib library. There are also other ways to construct proxies (like Dynamic JDK proxies), but let’s leave it at that for the moment.
Let’s see proxies in action in this picture:
 
As you can see from that diagram, the proxy has one job.
- Opening and closing database connections/transactions.
- And then delegating to the real UserService, the one you wrote.
- And other beans, like your UserRestController will never know that they are talking to a proxy, and not the real thing.
Quellen:
- ↑ Marco Behler, "Spring Transaction Management: @Transactional In-Depth"

