简体   繁体   English

如何在不使用 bean 或注释的情况下手动初始化 Spring Boot 存储库? (科特林)

[英]How do I manually initialize a Spring Boot repository without using beans or annotations? (Kotlin)

I want to initialize a repository object in Spring Boot without having to make it a bean or an Autowired property.我想在 Spring Boot 中初始化repository对象,而不必使其成为 bean 或Autowired属性。

I just want to do feedRepository = FeedRepository() in Kotlin.我只想在 Kotlin 中执行feedRepository = FeedRepository() Why won't Spring Boot allow me to do this?为什么 Spring Boot 不允许我这样做? Why does it have to be autowired or a bean?为什么它必须是自动装配的或 bean?

here's my code:这是我的代码:

@Component
class UriParser() {
    @Autowired
    lateinit var uriRepository: UriRepository

    @Autowired
    lateinit var hostRepository: HostRepository

    @Autowired
    lateinit var feedRepository: FeedRepository

    fun parseUri(uri: String) : Feed {
        val urlRepository = UriRepository()
        val uri = URI(uri)

        val uriRecord = uriRepository.save(Uri(scheme = uri.scheme, host = uri.host, port = uri.port))
        hostRepository.save(Host(host = uri.host))

        return feedRepository.save(Feed(uriId = uriRecord.id))
    }
}

Edit: ok so here's better context as to why I can't autowire things.编辑:好的,所以这里有更好的背景说明为什么我不能自动装配东西。 Basically I'm doing this within a static function so I have absolutely no access to beans or autowired properties:基本上我在静态函数中执行此操作,因此我绝对无法访问 bean 或自动装配的属性:

@SpringBootApplication
class Application

fun main(args: Array<String>) {
    runApplication<Application>(*args)
     KafkaScheduler().init()
}

The Kafka scheduler basically executes several threads which all listen to kafka topics: Kafka 调度程序基本上执行几个线程,这些线程都监听 kafka 主题:

class KafkaScheduler() {
    // this doesn't work....
    @Autowired
    lateinit var feedRepository: FeedRepository

    fun init() {
        val threads = arrayOf(
            {
                QueueListener().init()
            },
            {
                PrefetchListener().init()
            },
            {
                FetchListener().init()
            }
        )

        val service = Executors.newFixedThreadPool(threads.size)

        for (thread in threads) {
            service.submit(thread)
        }
    }
}

however, I can't autowire the KafkaScheduler class as a bean within a static method so I can't really make that a component or autowire any repositories, as far as I know.但是,我无法将 KafkaScheduler 类自动装配为静态方法中的 bean,因此据我所知,我无法真正将其作为组件或自动装配任何存储库。

Well I figured it out.好吧,我想通了。 A simple @PostConstruct annotation solves everything I need for this.一个简单的@PostConstruct注释解决了我需要的一切。 Essentially, I just mark KafkaScheduler as a @Component , autowire everything I need within it, the only difference is I added @PostConstruct to the init method so I have access to the autowired properties.本质上,我只是将KafkaScheduler标记为@Component ,在其中自动装配我需要的一切,唯一的区别是我将@PostConstruct添加到 init 方法,以便我可以访问自动装配的属性。 Then I can autowire the repositories wherever I need them (in this case within QueueListener which uses UriParser )然后,我可以自动装配存储库的地方,我需要他们(在中这种情况下QueueListener它采用UriParser

@Component
class KafkaScheduler() {
    @Autowired
    lateinit var queueListener: QueueListener

    @Autowired
    lateinit var prefetchListener: PrefetchListener

    @Autowired
    lateinit var fetchListener: FetchListener

    @PostConstruct
    fun init() {
        val threads = arrayOf(
            {
                queueListener.init()
            },
            {
                prefetchListener.init()
            },
            {
                fetchListener.init()
            }
        )

        val service = Executors.newFixedThreadPool(threads.size)

        for (thread in threads) {
            service.submit(thread)
        }
    }
}

Noone stops you from initializing objects by yourself inside the spring managed component.没有人会阻止您在 spring 管理的组件内自行初始化对象。

However, these objects:但是,这些对象:

  • won't be managed by spring (eg if they also have dependencies you'll have to resolve them by yourself)不会由 spring 管理(例如,如果它们也有依赖项,你必须自己解决它们)
  • It doesn't make sense to use @Autowired in this case because it works only between spring beans.在这种情况下使用@Autowired没有意义,因为它只适用于 spring bean。

For example, this is wrong because class A is not managed by spring:例如,这是错误的,因为类 A 不是由 spring 管理的:


// not managed by spring
class A {
}

@Component
class B {
  @Autowired 
  A a;
}

So I believe you aim for something like this:所以我相信你的目标是这样的:

@Component 
class UriParser() { 

    // no autowired here, you manage everything by youself
    // also possible from constructor
    lateinit var uriRepository = UriRepository()
    ...
}

Side note, I haven't learnt Kotlin, so the syntax might be wrong.旁注,我没有学过 Kotlin,所以语法可能是错误的。

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM