spring @Transactional не обновляется в db с помощью hibernate [duplicate]

Необходимость объяснения виртуальной функции [Легко понять]

#include<iostream>

using namespace std;

class A{
public: 
        void show(){
        cout << " Hello from Class A";
    }
};

class B :public A{
public:
     void show(){
        cout << " Hello from Class B";
    }
};


int main(){

    A *a1 = new B; // Create a base class pointer and assign address of derived object.
    a1->show();

}

Выход будет:

Hello from Class A.

Но с виртуальной функцией:

#include<iostream>

using namespace std;

class A{
public:
    virtual void show(){
        cout << " Hello from Class A";
    }
};

class B :public A{
public:
    virtual void show(){
        cout << " Hello from Class B";
    }
};


int main(){

    A *a1 = new B;
    a1->show();

}

Выход будет:

Hello from Class B.

Следовательно, с виртуальной функцией вы можете добиться полиморфизма во время выполнения.

70
задан Gray 30 October 2012 в 22:47
поделиться

6 ответов

С пружиной 4 возможно автоподключение

@Service
@Transactional
public class UserServiceImpl implements UserService{
    @Autowired
    private  UserRepositroy repositroy;

    @Autowired
    private UserService userService;

    @Override
    public void update(int id){
       repository.findOne(id).setName("ddd");
    }

    @Override
    public void save(Users user) {
        repositroy.save(user);
        userService.update(1)l
    }
}
2
ответ дан Almas Abdrazak 27 August 2018 в 23:41
поделиться

Это ограничение с Spring AOP. (динамические объекты и CGLIB)

Если вы настроите Spring на использование AspectJ для обработки транзакций, ваш код будет работать.

Простая и, вероятно, лучшая альтернатива - реорганизовать ваш код. Например, один класс, который обрабатывает пользователей, и один, обрабатывающий каждого пользователя. Тогда обработка транзакций по умолчанию с Spring AOP будет работать.

Советы по настройке для обработки транзакций с AspectJ

Чтобы включить Spring для использования AspectJ для транзакций, вы должны установить режим AspectJ:

<tx:annotation-driven mode="aspectj"/>

Если вы используете Spring с более старой версией, чем 3.0, вы также должны добавить это в свою конфигурацию Spring:

<bean class="org.springframework.transaction.aspectj
        .AnnotationTransactionAspect" factory-method="aspectOf">
    <property name="transactionManager" ref="transactionManager" />
</bean>
71
ответ дан Elnur Abdurrakhimov 27 August 2018 в 23:41
поделиться

Это мое решение для самозапуска.

private SBMWSBL self;
@Autowired private ApplicationContext applicationContext;

@PostConstruct
public  void postContruct(){
    self =applicationContext.getBean(SBMWSBL.class);
}
5
ответ дан Hlex 27 August 2018 в 23:41
поделиться

Проблема заключается в том, что прокси-серверы AOP Spring не распространяются, а скорее завершают ваш экземпляр службы, чтобы перехватывать вызовы. Это приводит к тому, что любой вызов «этого» из вашего экземпляра службы непосредственно вызывается в этом экземпляре и не может быть перехвачен прокси-сервером (прокси-сервер даже не знает о таком вызове). Уже упоминалось одно решение. Еще одним из излюбленных было бы просто заставить Spring вставить экземпляр службы в саму службу и вызвать ваш метод на введенном экземпляре, который будет прокси-сервером, который обрабатывает ваши транзакции. Но имейте в виду, что это может иметь и плохие побочные эффекты, если ваш сервисный компонент не является одиночным:

<bean id="userService" class="your.package.UserService">
  <property name="self" ref="userService" />
    ...
</bean>

public class UserService {
    private UserService self;

    public void setSelf(UserService self) {
        this.self = self;
    }

    @Transactional
    public boolean addUser(String userName, String password) {
        try {
        // call DAO layer and adds to database.
        } catch (Throwable e) {
            TransactionAspectSupport.currentTransactionStatus()
                .setRollbackOnly();

        }
    }

    public boolean addUsers(List<User> users) {
        for (User user : users) {
            self.addUser(user.getUserName, user.getPassword);
        }
    } 
}
46
ответ дан Kai 27 August 2018 в 23:41
поделиться

Вы можете авторизовать BeanFactory внутри одного класса и делать

getBean(YourClazz.class)

. Он автоматически проксимизирует ваш класс и учитывает вашу @Transactional или другую аннотацию aop.

0
ответ дан LionH 27 August 2018 в 23:41
поделиться

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

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

0
ответ дан Ujjwal Choudhari 27 August 2018 в 23:41
поделиться
Другие вопросы по тегам:

Похожие вопросы: