Гость
Map
Форумы / Java [игнор отключен] [закрыт для гостей] / как правильно внедрить bean в компонент spring-boot. / 25 сообщений из 40, страница 1 из 2
13.09.2021, 09:27
    #40097050
chron
Участник
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
как правильно внедрить bean в компонент spring-boot.
есть класс JdbcRepository:
Код: java
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
@Repository
@Scope("prototype")
public class JdbcRepository {
    private final NamedParameterJdbcTemplate jdbc;

    public JdbcRepository(NamedParameterJdbcTemplate jdbc) {
        this.jdbc = jdbc;
    }
...
}


который работает для одного источника данных в среде spring-boot.
Нужно расширить его применение для нескольких источников данных, для нескольких значений параметра NamedParameterJdbcTemplate jdbc.

В конфигураторе создаются beans для каждого источника данных, как напр., для источника postgres:
Код: java
1.
2.
3.
4.
@Bean(name = "sourcePg")
public NamedParameterJdbcTemplate jdbcPg(HikariDataSource dataSourcePg) {
    return new NamedParameterJdbcTemplate(dataSourcePg);
}


и далее начинаются непонятки.
Для манипулирования данными разных источников используются классы DataAccessService[xx], напр., для источника postgres:

Код: java
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
@Service
public class DataAccessServicePg {
    private final NamedParameterJdbcTemplate jdbcPg;
    private final JdbcRepository jdbcRepository;

    public DataAccessService(NamedParameterJdbcTemplate jdbcPg, JdbcRepository jdbcRepository) {
        this.jdbcPg = jdbcPg;
        this.jdbcRepository = new JdbcRepository(jdbcPg);
    }
...
}


я не понимаю, как правильно внедрить bean(name = "sourcePg").
строка конструктора
Код: java
1.
 this.jdbcRepository = new JdbcRepository(jdbcPg);


очевидно, не правильная.

Как правильно внедрить bean(name = "anySource") в класс DataAccessService[anySource] ?
...
Рейтинг: 0 / 0
13.09.2021, 09:51
    #40097061
как правильно внедрить bean в компонент spring-boot.
1. Твой бин не должен быть prototype, это обычный singleton. Ты ж не хочешь чтоб создавалось много одинаковых объектов JdbcRepository (с одинаковыми параметрами).
2. Ну и объяви несколько @Bean в контексте которые возвращают все тот же JdbcRepository:
Код: java
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
@Bean
public NamedParameterJdbcTemplate jdbcPg(HikariDataSource dataSourcePg) {
    return new NamedParameterJdbcTemplate(dataSourcePg);
}
@Bean
public NamedParameterJdbcTemplate jdbcMysql(HikariDataSource dataSourceMysql) {
    return new NamedParameterJdbcTemplate(dataSourceMysql);
}
@Bean
public NamedParameterJdbcTemplate jdbcOracle(HikariDataSource dataSourceOracle) {
    return new NamedParameterJdbcTemplate(dataSourceOracle);
}


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

Но архитектура конечно странноватая, я с трудом себе представляю требования.. Если расскажешь подробней зачем это все и как это будет использоваться, может тебе подскажут вариант по-удобней.
...
Рейтинг: 0 / 0
13.09.2021, 09:51
    #40097062
chpasha
Участник
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
как правильно внедрить bean в компонент spring-boot.
chron
Как правильно внедрить

а это ты для дяди пишешь?
@Bean (name = "sourcePg")

бин с конкретным именем можно инжектировать либо неявно, указав его как имя параметра/поля, т.е. вместо NamedParameterJdbcTemplate jdbcPg -> NamedParameterJdbcTemplate sourcePg либо явно с помощью аннотации Qualifier
...
Рейтинг: 0 / 0
13.09.2021, 09:57
    #40097064
PetroNotC Sharp
Участник
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
как правильно внедрить bean в компонент spring-boot.
chron,
Какой смысл жонглировать базами в одном сервисе?
...
Рейтинг: 0 / 0
13.09.2021, 10:00
    #40097065
chron
Участник
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
как правильно внедрить bean в компонент spring-boot.
chpasha,
прошу прощения, имя бина я указал просто для передачи смысла, в тексте .java этого нет.
...
Рейтинг: 0 / 0
13.09.2021, 10:10
    #40097068
chron
Участник
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
как правильно внедрить bean в компонент spring-boot.
Stanislav Bashkyrtsev,
я так и сделал: в конфигураторе объявил неск. бинов для разных источников.
и то же самое делаю для сервисов.
я тщательно не пробовал проект, т.к. пока занят простым редактированием, и строка
Код: java
1.
    this.jdbcRepository = new JdbcRepository(jdbcPg);


мне не понравилась.
вот и вопрос такой: так - можно ?
и @Scope("prototype") должен быть именно такой.
...
Рейтинг: 0 / 0
13.09.2021, 10:13
    #40097069
chron
Участник
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
как правильно внедрить bean в компонент spring-boot.
PetroNotC Sharp,
так ведь не всегда мы властны над судьбой.
но здесь для каждой БД - свой класс-сервис.
...
Рейтинг: 0 / 0
13.09.2021, 10:18
    #40097071
как правильно внедрить bean в компонент spring-boot.
chron , с такой формулировкой тебе не помогут или помогут неправильно. Либо ты расскажешь что и зачем, либо будешь продолжать есть кактус. Это ты еще до управления транзакциями не дошел..
chronи @Scope("prototype") должен быть именно такой.Ну раз ты познал все таинства, то ты и на другие вопросы наверно сам знаешь ответы :)
...
Рейтинг: 0 / 0
13.09.2021, 10:19
    #40097072
PetroNotC Sharp
Участник
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
как правильно внедрить bean в компонент spring-boot.
chron,
Заставляют?
Это плохо.
А как тогда пишите
sping.datasource.url.......driver?
...
Рейтинг: 0 / 0
13.09.2021, 10:43
    #40097080
chron
Участник
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
как правильно внедрить bean в компонент spring-boot.
Stanislav Bashkyrtsev,
да, с транзакциями всё впереди.
...
Рейтинг: 0 / 0
13.09.2021, 10:46
    #40097082
chron
Участник
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
как правильно внедрить bean в компонент spring-boot.
PetroNotC Sharp,
короче, я понял: нужно довести проект до формирования конкретной ошибки и тогда разсуждать.
возможно, я поторопился с вопросом. пусть пока так будет. Вот здесь уже Stanislav Bashkyrtsev предвидит шикарный трах.
...
Рейтинг: 0 / 0
13.09.2021, 10:48
    #40097084
PetroNotC Sharp
Участник
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
как правильно внедрить bean в компонент spring-boot.
chron,
Ну дак Hello world же.
Сначала нужно сделать демку как все делают.
А потом уже фантазии из сна постановщика или программиста.
Удачи!
...
Рейтинг: 0 / 0
13.09.2021, 18:59
    #40097340
chron
Участник
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
как правильно внедрить bean в компонент spring-boot.
не инициализируется bean одного из источников данных.
структура проекта прилагается.

конфигураторы написаны для каждого источника данных, для pg и h2.
для БД H2 класс DataSourcesConfigurationH2:
Код: java
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
14.
15.
16.
17.
18.
19.
20.
21.
22.
23.
24.
25.
26.
27.
28.
29.
30.
31.
32.
33.
34.
35.
36.
37.
38.
@Configuration
@EnableTransactionManagement
public class DataSourcesConfigurationH2 {

    @Bean
    @Primary
    @ConfigurationProperties("app.datasource.h2")
    public DataSourceProperties dataSourcePropertiesH2() {
        return new DataSourceProperties();
    }

    @Bean(name = "dataSourceH2")
    @Primary
    @ConfigurationProperties("app.datasource.h2.configuration")
    public HikariDataSource dataSourceH2(DataSourceProperties dataSourcePropertiesH2) {
        return dataSourcePropertiesH2.initializeDataSourceBuilder().type(HikariDataSource.class).build();
    }

    @Bean
    @Primary
    public NamedParameterJdbcTemplate jdbcH2(HikariDataSource dataSourceH2) {
        return new NamedParameterJdbcTemplate(dataSourceH2);
    }


    @Bean
    @Primary
    TransactionManager transactionManagerH2(HikariDataSource dataSourceH2) {
        return new DataSourceTransactionManager(dataSourceH2);
    }

    @Bean
    @Primary
    JdbcRepository jdbcRepositoryH2(NamedParameterJdbcTemplate jdbcH2) {
        return new JdbcRepository(jdbcH2);
    };

}



для БД postgres класс DataSourcesConfigurationPg:
Код: java
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
14.
15.
16.
17.
18.
19.
20.
21.
22.
23.
24.
25.
26.
27.
28.
29.
30.
31.
@Configuration
@EnableTransactionManagement
//@ComponentScan(basePackages = {"mag.jdbc.repository"})
public class DataSourcesConfigurationPg {

    @Bean
    @ConfigurationProperties("app.datasource.pg")
    public DataSourceProperties dataSourcePropertiesPg() {
        return new DataSourceProperties();
    }

    @Bean(name = "dataSourcePg")
    @ConfigurationProperties("app.datasource.pg.configuration")
    public HikariDataSource dataSourcePg(DataSourceProperties dataSourcePropertiesPg) {
        return dataSourcePropertiesPg.initializeDataSourceBuilder().type(HikariDataSource.class).build();
    }

    @Bean
    public NamedParameterJdbcTemplate jdbcPg(HikariDataSource dataSourcePg) {
        return new NamedParameterJdbcTemplate(dataSourcePg);
    }
    @Bean
    TransactionManager transactionManagerPg(HikariDataSource dataSourcePg) {
        return new DataSourceTransactionManager(dataSourcePg);
    }

    @Bean
    JdbcRepository jdbcRepositoryPg(NamedParameterJdbcTemplate jdbcPg) {
        return new JdbcRepository(jdbcPg);
    };
}


класс JdbcRepository подключается из локальной библиотеки .m2
строка в начале листинга закомментирована
Код: java
1.
//@ComponentScan(basePackages = {"mag.jdbc.repository"})


потому, что класс находится без этого указателя.
--------
класс манипулирования данными DataAccessServicePg:
Код: java
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
14.
15.
@Service
public class DataAccessServicePg {

    private final JdbcRepository jdbcRepositoryPg;

    public DataAccessServicePg(JdbcRepository jdbcRepositoryPg) {
        this.jdbcRepositoryPg = jdbcRepositoryPg;
    }

    public Actor actorOne(String actorName) {
        return jdbcRepositoryPg.presentByString("actorList",
                actorName, ActorMapper::new, Actor::new);
    }

}



контроллер BaseController, который вызывает метод Actor actorOne:
Код: java
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
14.
15.
16.
17.
18.
19.
20.
21.
22.
23.
24.
25.
26.
27.
28.
29.
30.
31.
32.
@Controller
@RequestMapping("")
public class BaseController {
    private final DataAccessServicePg servicePg;

    public BaseController(DataAccessServicePg servicePg) {
        this.servicePg = servicePg;
    }

    @GetMapping("")
    public String primary(
            Model model) {
        return "primary";
    }
    @GetMapping("/dataH2")
    public String spaceH2(
            HttpServletRequest request, Model model) {

        return "spaceData";
    }

    @GetMapping("/dataPg")
    public String spacePg(
            HttpServletRequest request, Model model) {

        Actor actor = servicePg.actorOne("palomnic");
        model.addAttribute("pageTitle", "pg: palomnic");
        model.addAttribute("actor", actor);
        return "spaceData";
    }

}


остальное — стандартно.

вызываю метод контроллера @GetMapping("/dataPg") и получаю сообщение:
Код: html
1.
2.
[500] Internal Server Error 
PreparedStatementCallback; bad SQL grammar [select * from actorList(?);]; nested exception is org.h2.jdbc.JdbcSQLSyntaxErrorException: Функция "ACTORLIST" не найдена Function "ACTORLIST" not found; SQL statement: select * from actorList(?); [90022-200]


из которого ясно, что вызывается метод сервиса H2, а не pg. Функция actorlist есть в БД pg, и её нет в БД H2. Другими словами, не выполнился код инициализации в конфигураторе DataSourcesConfigurationPg.

Кто знает, куда нужно посмотреть ?
...
Рейтинг: 0 / 0
13.09.2021, 21:36
    #40097373
chpasha
Участник
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
как правильно внедрить bean в компонент spring-boot.
поставь брейпоинты в Pg-конфигурации и посмотри что туда реально прилетает при создании бинов, возможно ты что-то с пропертями намутил и туда приходит не то, что ты думаешь. Так сложно на глаз косяк заметить, вроде бы (вроде бы) все норм
...
Рейтинг: 0 / 0
13.09.2021, 21:56
    #40097376
chron
Участник
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
как правильно внедрить bean в компонент spring-boot.
chpasha,
вот с каким application.properties всё запускается:

Код: html
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
14.
15.
16.
17.
18.
19.
20.
server.servlet.context-path=/transData
server.error.include-message=always
spring.devtools.add-properties=false
spring.main.banner-mode=off

spring.thymeleaf.cache=false
spring.thymeleaf.enabled=true
spring.thymeleaf.prefix=classpath:/templates/
spring.thymeleaf.suffix=.html

server.error.whitelabel.enabled=false
server.error.path=/error

app.datasource.h2.jdbc-url=jdbc:h2:~/database/tester
app.datasource.h2.username=name
app.datasource.h2.password=pass

app.datasource.pg.jdbc-url=jdbc:postgresql://localhost:5432/tester
app.datasource.pg.username=name
app.datasource.pg.password=pass



в pg-конфигурации пробовал так:
Код: java
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
    @Bean(name = "dataSourcePg")
    @ConfigurationProperties("app.datasource.pg.configuration")
    public HikariDataSource dataSourcePg(DataSourceProperties dataSourcePropertiesPg) {
        return dataSourcePropertiesPg.initializeDataSourceBuilder().type(HikariDataSource.class).build();
    }
    @Bean(name = "jdbcPg")
    public NamedParameterJdbcTemplate jdbcPg(@Qualifier("dataSourcePg") HikariDataSource dataSourcePg) {
        return new NamedParameterJdbcTemplate(dataSourcePg);
    }
    @Bean(name = "jdbcRepositoryPg")
    JdbcRepository jdbcRepositoryPg(@Qualifier("jdbcPg") NamedParameterJdbcTemplate jdbcPg) {
        return new JdbcRepository(jdbcPg);
    };


но тоже не помогло.
Конечно, дело в этом классе - конфигураторе БД pg.
...
Рейтинг: 0 / 0
13.09.2021, 22:20
    #40097382
chron
Участник
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
как правильно внедрить bean в компонент spring-boot.
установлено: не включается инициализатор initializeDataSourceBuilder() для бина
Код: java
1.
2.
3.
4.
5.
@Bean(name = "dataSourcePg")
@ConfigurationProperties("app.datasource.pg.configuration")
public HikariDataSource dataSourcePg(DataSourceProperties dataSourcePropertiesPg) {
    return dataSourcePropertiesPg.initializeDataSourceBuilder().type(HikariDataSource.class).build();
}
...
Рейтинг: 0 / 0
14.09.2021, 09:59
    #40097478
chpasha
Участник
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
как правильно внедрить bean в компонент spring-boot.
chron
установлено: не включается инициализатор initializeDataSourceBuilder() для бина

откуда вообще префикс "app.datasource.pg.configuration" - у тебя в пропертях такого нет, есть "app.datasource.h2" и "app.datasource.pg". и зачем он вообще над методом, если бин DataSourceProperties создается отдельно строчкой выше?
...
Рейтинг: 0 / 0
14.09.2021, 10:26
    #40097491
chron
Участник
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
как правильно внедрить bean в компонент spring-boot.
chpasha, да, это лишнее.
я оставил:
-- в классе DataSourcesConfigurationH2
Код: java
1.
2.
3.
4.
5.
6.
    @Bean(name = "dataSourceH2")
    @Primary
    @ConfigurationProperties(prefix="app.datasource.h2")
    public HikariDataSource dataSourceH2(DataSourceProperties dataSourcePropertiesH2) {
        return dataSourcePropertiesH2.initializeDataSourceBuilder().type(HikariDataSource.class).build();
    }


-- и в классе DataSourcesConfigurationPg
Код: java
1.
2.
3.
4.
5.
    @Bean(name = "dataSourcePg")
    @ConfigurationProperties(prefix="app.datasource.pg")
    public HikariDataSource dataSourcePg(DataSourceProperties dataSourcePropertiesPg) {
        return dataSourcePropertiesPg.initializeDataSourceBuilder().type(HikariDataSource.class).build();
    }


но в обоих классах значение HikariDataSource datasource.getDriverClassName() после инициализации равно "org.h2.Driver".
...
Рейтинг: 0 / 0
14.09.2021, 12:07
    #40097523
chpasha
Участник
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
как правильно внедрить bean в компонент spring-boot.
нужно значит дебажить что происходит в методе dataSourcePg - какие DataSourceProperties туда реально попадают и почему. Магия спринга нах. Я если честно, ни видал такого способа конфигурации через DataSourceProperties, по идее (по идее) должно работать и так

Код: java
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
@Bean
    @ConfigurationProperties(prefix="spring.first-datasource")
    public DataSource firstDataSource() {
        return DataSourceBuilder.create().build();
    }

@Bean
@ConfigurationProperties(prefix="spring.second-datasource")
    public DataSource secondDataSource() {
        return DataSourceBuilder.create().build();
    }
...
Рейтинг: 0 / 0
14.09.2021, 14:38
    #40097572
chron
Участник
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
как правильно внедрить bean в компонент spring-boot.
chpasha, всё предельно просто, но не работает:
application.properties
Код: html
1.
2.
3.
4.
5.
6.
7.
app.datasource.h2.jdbc-url=jdbc:h2:~/database/tester
app.datasource.h2.username=usr
app.datasource.h2.password=pwd

app.datasource.pg.jdbc-url=jdbc:postgresql://localhost:5432/tester
app.datasource.pg.username=usr
app.datasource.pg.password=pwd



config h2
Код: java
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
14.
15.
16.
17.
18.
19.
20.
21.
22.
23.
24.
    @Bean
    @Primary
    @ConfigurationProperties(prefix="app.datasource.h2")
    public DataSource dataSourceH2() {
        return DataSourceBuilder.create().build();
    }

    @Bean
    @Primary
    public NamedParameterJdbcTemplate jdbcH2(HikariDataSource dataSource) {
        return new NamedParameterJdbcTemplate(dataSource);
    }

    @Bean
    @Primary
    JdbcRepository jdbcRepositoryH2(NamedParameterJdbcTemplate jdbc) {
        return new JdbcRepository(jdbc);
    };

    @Bean
    @Primary
    TransactionManager transactionManagerH2(HikariDataSource dataSource) {
        return new DataSourceTransactionManager(dataSource);
    }



config pg
Код: java
1.
2.
3.
4.
5.
6.
    @Bean
    @ConfigurationProperties(prefix="app.datasource.pg")
    public DataSource dataSourcePg() {
        return DataSourceBuilder.create().build();
    }
...


не инициализируется dataSourcePg ну, никак. без дебаггера ясно, что spring`у нужно что-то подсказать.
...
Рейтинг: 0 / 0
14.09.2021, 15:24
    #40097583
PetroNotC Sharp
Участник
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
как правильно внедрить bean в компонент spring-boot.
chron,
Сворачивайся и делай как все.
...
Рейтинг: 0 / 0
14.09.2021, 16:19
    #40097598
chron
Участник
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
как правильно внедрить bean в компонент spring-boot.
PetroNotC Sharp,
как это - как все ?
я не могу получить ответ на простой вопрос: как инициализировать в spring два источника данных. Даже в документации об этом есть http://docs.spring.io/spring-boot/docs/current/reference/htmlsingle/#howto-two-datasources . Значит - есть в этом потребность. Но указанный в доках код не работает и никто не знает ответа - почему. Я считал, что spring при его популярности, такие вопросы уже закрыл.
В данном случае выполняется "ручная инициализация", которая полностью в доках не описана, отсюда и непонятки.
...
Рейтинг: 0 / 0
14.09.2021, 16:29
    #40097601
PetroNotC Sharp
Участник
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
как правильно внедрить bean в компонент spring-boot.
chron,
Смотри первый пост.
Жонглировать коннектами бд в одном приложении глупость.
Сделай микросервисы. Или jdbc. Но спринг и jpa и ОРМ и хибер не заточен на это.
Имхо
...
Рейтинг: 0 / 0
14.09.2021, 16:31
    #40097603
PetroNotC Sharp
Участник
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
как правильно внедрить bean в компонент spring-boot.
chron
Значит - есть в этом потребность.
у тебя одного?
Прогеры часто выдумывают себе самому работу. Бывает.
...
Рейтинг: 0 / 0
14.09.2021, 16:33
    #40097605
chron
Участник
Скрыть профиль Поместить в игнор-лист Сообщения автора в теме
как правильно внедрить bean в компонент spring-boot.
PetroNotC Sharp,
да, про микро я уже думал. побарахтаюсь ещё немного и буду всё переделывать.

chpasha спрашивал, откуда взялось использование
Код: java
1.
2.
3.
4.
5.
    @Bean
    @ConfigurationProperties("app.datasource.pg")
    public DataSourceProperties dataSourcePropertiesPg() {
        return new DataSourceProperties();
    }


вот отсюда взялось https://docs.spring.io/spring-boot/docs/current/reference/htmlsingle/#howto.data-access.configure-two-datasources .
...
Рейтинг: 0 / 0
Форумы / Java [игнор отключен] [закрыт для гостей] / как правильно внедрить bean в компонент spring-boot. / 25 сообщений из 40, страница 1 из 2
Целевая тема:
Создать новую тему:
Автор:
Найденые пользователи ...
Разблокировать пользователей ...
Читали форум (0):
Пользователи онлайн (0):
x
x
Закрыть


Просмотр
0 / 0
Close
Debug Console [Select Text]