V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
牛客网
jiobanma
V2EX  ›  Java

springboot 中 controller 中通过一个 key 值动态调用 service

  •  
  •   jiobanma · 47 天前 · 2501 次点击
    这是一个创建于 47 天前的主题,其中的信息可能已经有所发展或是发生改变。

    想在服务启动的时候,将所有的 service 存放到一个缓存 Map 中,然后给定一个 key 值绑定。 在 controller 中通过 key 值找到对应的 service 对象该如何实现。 因为不同 service 的类型不一样,所以 Map 在定义的时候只能定义成 Map<String,Object>value 是个 Object, 这样的话通过 key 值取出来的也就是 Object 的类型,无法调用到 service 下的方法。 有没有好的实现思路呢?

    50 条回复    2020-09-07 17:34:35 +08:00
    tsening
        1
    tsening   47 天前
    我累了,小白吧。。。
    mosliu
        2
    mosliu   47 天前
    反射呗
    chendy
        3
    chendy   47 天前
    为啥要这么搞,别的不说,service 的参数和返回也不一样啊
    jiobanma
        4
    jiobanma   47 天前
    @chendy #3 因为服务要做成动态的,service 里面只有一个通用接口,参数都是 Map 所以不存在返回值类型和参数不同的问题
    wysnylc
        5
    wysnylc   47 天前
    @jiobanma #4 我只能说,作死
    说了多少回参数和返回值不要用 map,不撞南墙不回头
    pushback
        6
    pushback   47 天前
    method.invoke
    pushback
        7
    pushback   47 天前
    存储 class 就行,对应 value 到 BeanFactory 下去取
    idoggy
        8
    idoggy   47 天前 via Android   ❤️ 1
    入参都一样吗?要调用的 service 全部继承同一个接口就行了,用 applicationcontext.getbean 去调。
    ily433664
        9
    ily433664   47 天前
    通过 applicationContext 获取,可以根据 bean 类型或者名称
    YzSama
        10
    YzSama   47 天前
    @idoggy #8 这个方式可取。 不过,入参和出参不建议使用 map 。
    lipcao
        11
    lipcao   47 天前
    哈哈,你这个标准的策略+工厂模式 解决 if else 啊,前一段时间公众号嗷嗷推这个。。。
    zhuweiyou
        12
    zhuweiyou   47 天前   ❤️ 1
    建议楼主学 PHP 或 JS,他们就是这么玩的。
    DelayNoMay
        13
    DelayNoMay   47 天前
    用 golang 的 interface
    yuan7712
        14
    yuan7712   47 天前
    460881773
        15
    460881773   47 天前
    花里胡哨
    aguesuka
        16
    aguesuka   47 天前 via Android
    service 都带个 controller,原来的 controller 里通过 key 转发到对应的 service 的 controller 。param bind 是 spring 的工作,不要实现一个低配版的
    aguesuka
        17
    aguesuka   47 天前 via Android   ❤️ 1
    以前我还是个傻逼的时候,map 里面存的是个 lambda 函数,比如 map.put(key,xxxxService::doxxxxxxx)
    huifer
        18
    huifer   47 天前
    添加一个 接口比如
    ```java
    public interface KeyInterface{
    String key();
    }

    ```
    - 所有 service 都实现这个接口,重写 key()

    - 在启动阶段(ApplicationRunner 或者 CommandLineRunner)通过 context.getBeanOfTypes(KeyInterface.class) 获取到一个 map
    key: beanName value: KeyInterface 的实现
    - 调用 value 的 key()方法. 将结果映射成 key:key() ,value: service 即可
    dutianze
        19
    dutianze   47 天前
    如果所有 sevice 都是同样的接口的话,

    1. 所有 service 都继承一个接口 CommonService

    2.
    @Autowired
    private Map<String, CommonService> commonServiceMap;

    k 默认是 service 的类名小写,v 默认是对应的 service
    butterfly1211
        20
    butterfly1211   47 天前
    服务注册与服务发现
    24bit
        21
    24bit   47 天前
    不如直接用注解标注要缓存的方法,直接在 Map 里面存 Method,取的时候直接调用
    Cyron
        22
    Cyron   47 天前
    Service service = applicationContext.getBean("serviceImplA");
    Service service = applicationContext.getBean("serviceImplB");
    jintianfengda
        23
    jintianfengda   46 天前
    用 map 这么玩开发一时爽,维护火葬场
    ixx
        24
    ixx   46 天前
    liuguangcuican
        25
    liuguangcuican   46 天前 via Android
    定义一个 interface,在其每个实现类上使用 @Component("实现类的别名")。在 controller 里使用 @Autowrid 定义一个 Map<String,你定义的 interface>,在入口方法里使用 map.get("实现类的别名")就能获取到你对应的实现类。
    d460686680
        26
    d460686680   46 天前
    实现同一个接口 + getBean()
    levizheng
        27
    levizheng   46 天前
    简单工厂就可以解决了。。 放一 map 里
    xuanbg
        28
    xuanbg   46 天前
    我司的 php 写的老系统就是这样,接口参数里面有个 action,1234567……分别对应一个方法。。。。各方法传入一个 object,其实就是一个 json,各个方法自己负责解析这个参数。楼主莫非要的是这个?

    可是为啥不把接口拆开呢?
    KevinBlandy
        29
    KevinBlandy   46 天前
    ApplicationContext.getBean();
    ocean1477
        30
    ocean1477   46 天前
    策略模式了解下
    caotian
        31
    caotian   46 天前
    不要手工维护 map, 不然每次加一个 serivce 还要修改这个 map
    可以这样试试: 每个 service 定义自己的 key, 实现同一个接口 IService, 使用的时候通过 Autowired 注入拿到 List<IService>对象, 通过 key 来获得具体的 service 实例, 再调用方法
    jiobanma
        32
    jiobanma   46 天前
    @wysnylc #5 业务需要
    pierswu
        33
    pierswu   46 天前
    这个用 mvc 的 forward 不就行了吗?

    每一个 service 都有各自的 controller,就按照正常的写 ,然后总的 controller 根据不同的 key,forward 到不同的 controller
    至于 key,最好能放在 url 中或者 header 中,好取一些
    pierswu
        34
    pierswu   46 天前
    用 forward 的话,每个 service 的参数类型,参数个数,返回值,都可以不一样的
    pengfei
        35
    pengfei   46 天前
    策略模式 ?
    venpong
        36
    venpong   46 天前
    建议学学设计模式,比如楼上说的策略模式就可以很优雅的解决你这个问题。
    oneisall8955
        37
    oneisall8955   46 天前 via Android   ❤️ 1
    我最喜欢搞这些花里胡哨得了,经常写工厂模式+模板
    securityCoding
        38
    securityCoding   46 天前
    applicationContext.getBean()

    策略模式+泛型可解

    不过在业务层做这种事情这种看起来很无脑....中间件还情有可原
    BigBunny
        39
    BigBunny   46 天前
    做那种对外开放平台标准化接口有用过。当时的想法是 controller 里所有接口的请求参数返回参数都一样。加签验签逻辑也都一致。索性共用一个接口,调用 service 里的方法通过自定义注解去区分。半路转的 java,也不知道这么做了对不对好不好。
    supermoonie
        40
    supermoonie   46 天前 via iPhone
    注解?
    chachae
        41
    chachae   46 天前
    策略模式
    gaius
        42
    gaius   46 天前 via Android
    以前掉各种银行接口就是用反射,配置全限定名
    1ffree
        43
    1ffree   46 天前
    19l 正解 +1
    HolmLoh
        44
    HolmLoh   46 天前
    工厂模式加模板模式,小的单人开发的项目这么搞搞还行,偷偷懒还挺快乐,人多起来不好维护
    yisheyuanzhang
        45
    yisheyuanzhang   46 天前
    想到两种方法
    1 、8L 的方法,Service 实现同一接口,key 作为 bean 的名称,用 applicationcontext.getbean 去调。缺点是每个 service 只能实现同一个方法。
    2 、定义 service 的类注解和方法注解, 启动时通过 BeanPostProcessor,根据类注解的 key 存入 Map<key,bean>中。调用时根据类注解 key,找到 bean,根据方法注解 key 找到方法,通过反射执行

    我们生产中用过第二种
    CoderGeek
        46
    CoderGeek   46 天前
    简单点
    1. <String,String> key 是你的策略 value 是你的 beanName 直接 get 就好
    2. <String,Class> key 是你的策略 value 是你的所有加载的 service 取出来强制转型
    3.直接用 xml 更简单创建一个 map 对象 value 是多个 bean
    举个常见栗子
    private Map<Object, Object> targetDataSources;
    <bean id="xxx" class="xxxxx">
    <property name="targetDataSources">
    <map>
    <entry key="xxxxx" value-ref="xxxxx"></entry>
    </map>
    </property>
    </bean>
    xgfan
        47
    xgfan   46 天前
    别听楼上 BB 。
    把那些 service 合成一个函数方法就 ok 了。
    一个函数就解决所有问题了,高内聚。
    传来 map,返回 map,改动也只需要改这一个函数,低耦合。
    xiaoxinshiwo
        48
    xiaoxinshiwo   45 天前
    instance of
    yazinnnn
        49
    yazinnnn   44 天前
    ```kotlin
    @RestController
    class ControllerA(
    var serviceA: ServiceA,
    var serviceB: ServiceB
    ) {

    @RequestMapping("/")
    fun foo(type: String, param: String) = serviceMap[type]!!(param)


    private val funcA = { it: String -> serviceA.foo(it) }
    private val funcB = { it: String -> serviceB.foo(it) }


    private val serviceMap by lazy {
    mutableMapOf(
    "a" to funcA,
    "b" to funcB
    )
    }
    }
    ```

    kotlin 可以这么干,java 没有 by lazy,估计比较啰嗦
    Relims
        50
    Relims   42 天前
    value 存 service 的实例化对象,通过对象来调用方法,如果你的 sevice 有共同的方法的化,还可以用到工厂模式
    关于   ·   FAQ   ·   API   ·   我们的愿景   ·   广告投放   ·   感谢   ·   实用小工具   ·   4617 人在线   最高记录 5168   ·     Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 · 21ms · UTC 05:57 · PVG 13:57 · LAX 22:57 · JFK 01:57
    ♥ Do have faith in what you're doing.