dubbo轉(zhuǎn)http方式調(diào)用方式
業(yè)務(wù)背景
在當(dāng)前項(xiàng)目下,所有前端請求均通過外層網(wǎng)關(guān)轉(zhuǎn)發(fā)到后端這邊的dubbo服務(wù),現(xiàn)計(jì)劃去掉網(wǎng)關(guān)層,由前端直接http調(diào)用后端dubbo。
解決方案
在前端調(diào)用方式不變的前提下,后端服務(wù)新建controller層(原后端服務(wù)無任何controller),做統(tǒng)一請求兼容轉(zhuǎn)發(fā):
package cn.***********.controller;
import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.TypeReference;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang3.StringUtils;
import org.apache.dubbo.config.ApplicationConfig;
import org.apache.dubbo.config.ReferenceConfig;
import org.apache.dubbo.config.RegistryConfig;
import org.apache.dubbo.config.utils.ReferenceConfigCache;
import org.apache.dubbo.rpc.service.GenericService;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import javax.servlet.ServletInputStream;
import javax.servlet.http.HttpServletRequest;
import java.io.IOException;
import java.lang.reflect.Method;
import java.lang.reflect.Parameter;
import java.util.List;
import java.util.Map;
/**
* Dubbo調(diào)用controller
* @Version:1.0
* @Desc
*/
@Slf4j
@RestController
@RequestMapping("/api/test/")
public class GenericController {
@PostMapping("/{service}/{method}")
public Object invoke(HttpServletRequest request,
@PathVariable("service") String service,
@PathVariable("method") String method) {
Object result;
try {
GenericService tarService = this.getTargetService(service);
if(tarService == null){
throw new RuntimeException("獲取service失敗");
}
// 1.解析傳入?yún)?shù)
String paramStr = this.parseInputStream(request);
// 獲取目標(biāo)接口的 Class 對象
Class<?> serviceInterface = Class.forName(service);
// 獲取方法的參數(shù)類型
String[] parameterTypes = this.getMethodParameterTypes(serviceInterface, method);
// 將 JSON 字符串轉(zhuǎn)換為目標(biāo)參數(shù)值
Object[] parameters = this.convertParameters(paramStr, parameterTypes);
// 調(diào)用方法
result = tarService.$invoke(method, parameterTypes, parameters);
} catch (Exception e) {
log.error("服務(wù)調(diào)用失敗, service={},method={}",service, method, e);
throw new ErrorCodeException("000000", "系統(tǒng)異常,請稍后重試");
}
return result;
}
/**
* 獲取請求文本
*
* @return
*/
public String parseInputStream(HttpServletRequest request) {
try {
ServletInputStream inputStream = request.getInputStream();
StringBuilder content = new StringBuilder();
int size = request.getContentLength();
byte[] b = new byte[size];
int lens;
while ((lens = inputStream.read(b)) > 0) {
content.append(new String(b, 0, lens, "utf-8"));
}
return content.toString();
} catch (IOException e) {
log.error("請求報文解析失敗", e);
throw new RuntimeException("請求報文解析異常");
}
}
/**
* 獲取目標(biāo)dubbo服務(wù)
* @params [service]
* @return org.apache.dubbo.rpc.service.GenericService
* @desc
*/
private GenericService getTargetService(String service){
// 應(yīng)用信息
ApplicationConfig application = SpringBeanUtils.getBean(ApplicationConfig.class);
// 注冊中心發(fā)現(xiàn)
RegistryConfig registry = SpringBeanUtils.getBean(RegistryConfig.class);
// 引用遠(yuǎn)程服務(wù)
ReferenceConfig<GenericService> reference = new ReferenceConfig<>();
reference.setApplication(application);
reference.setRegistry(registry);
reference.setProtocol("dubbo");
reference.setInterface(service);
reference.setTimeout(10000);
reference.setRetries(0);
// 聲明為泛化接口
reference.setGeneric(true);
ReferenceConfigCache cache = ReferenceConfigCache.getCache();
GenericService genericService = cache.get(reference);
if(genericService == null){
genericService = reference.get();
}
return genericService;
}
/**
* 將 JSON 字符串轉(zhuǎn)換為目標(biāo)參數(shù)值
* @params [paramStr, parameterTypes]
* @return java.lang.Object[]
* @author wu.zeng
* @date 2025/2/21 13:44
* @desc
*/
private Object[] convertParameters(String paramStr, String[] parameterTypes) {
Object[] parameters = new Object[parameterTypes.length];
for (int i = 0; i < parameterTypes.length; i++) {
String paramType = parameterTypes[i];
switch (paramType) {
case "java.lang.String":
parameters[i] = paramStr;
break;
case "java.lang.Integer":
parameters[i] = Integer.parseInt(paramStr);
break;
case "java.lang.Long":
parameters[i] = Long.parseLong(paramStr);
break;
case "java.util.Map":
parameters[i] = JSON.parseObject(paramStr, new TypeReference<Map<String, Object>>() {});
break;
case "java.util.List":
parameters[i] = JSON.parseObject(paramStr, new TypeReference<List<Object>>() {});
break;
default:
// 如果是自定義類型,使用反射或 JSON 反序列化
try {
Class<?> clazz = Class.forName(paramType);
parameters[i] = JSON.parseObject(paramStr, clazz);
} catch (ClassNotFoundException e) {
throw new RuntimeException("不支持的類型: " + paramType, e);
}
}
}
return parameters;
}
/*
* 獲取目標(biāo)方法的參數(shù)類型
* @params [serviceInterface, methodName]
* @return java.lang.String[]
* @desc
*/
public String[] getMethodParameterTypes(Class<?> serviceInterface, String methodName) {
for (Method method : serviceInterface.getMethods()) {
// 由于不支持重載,因此這里可以根據(jù) methodName 來匹配方法
if (method.getName().equals(methodName)) {
Parameter[] parameters = method.getParameters();
String[] parameterTypes = new String[parameters.length];
for (int i = 0; i < parameters.length; i++) {
parameterTypes[i] = parameters[i].getType().getName();
}
return parameterTypes;
}
}
throw new RuntimeException("方法未找到: " + methodName);
}
}總結(jié)
以上為個人經(jīng)驗(yàn),希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
Java編程實(shí)現(xiàn)從給定范圍內(nèi)隨機(jī)N個不重復(fù)數(shù)生成隨機(jī)數(shù)的方法小結(jié)
這篇文章主要介紹了Java編程實(shí)現(xiàn)從給定范圍內(nèi)隨機(jī)N個不重復(fù)數(shù)生成隨機(jī)數(shù)的方法,結(jié)合實(shí)例形式較為詳細(xì)的分析了java根據(jù)指定范圍生成不重復(fù)隨機(jī)數(shù)的相關(guān)操作技巧,需要的朋友可以參考下2017-04-04
使用RestTemplate 調(diào)用遠(yuǎn)程接口上傳文件方式
這篇文章主要介紹了使用RestTemplate 調(diào)用遠(yuǎn)程接口上傳文件方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-09-09
Java?web開發(fā)環(huán)境的搭建超完整步驟
這篇文章主要介紹了如何安裝和配置IDEA?2020.1.1?X64版本軟件,包括創(chuàng)建Java?Web項(xiàng)目、配置Tomcat、部署Tomcat?API以及創(chuàng)建和配置Servlet,通過這些步驟,新手可以快速搭建起Javaweb開發(fā)環(huán)境,需要的朋友可以參考下2024-11-11

