记一次在java项目中动态调用接口实现

需求:

  • 提供一个统一的对外接口,调用者传入统一格式的json来进行调用到不同的接口并返回统一格式的json

请求url:

127.0.0.1:30072/openapi/api/v1/json

请求示例代码:

{
	"method": "system_ping",
	"timestamp": "1576565002744",
	"sign": "7D57278C74EC01DF33F8F826A8683F83",
	"data": "{\"str\":\"测试参数\"}"
}

返回示例代码:

{
    "code": 0,
    "msg": "成功",
    "data": {
        "msg": "请求参数str为:测试参数"
    }
}

实现

说明:以下很多代码为说明如何一个接口获取不同服务的方法,没做异常处理等其他代码优化,请注意.

controller提供web方法

第一种方式

定义一个枚举RequestEnum,保存method,并根据method获取到对应枚举,来进行switch循环(不推荐)
优点: 小白都看得懂
缺点: 每次加service都需要改controller来加case
具体代码如下:

import lombok.AllArgsConstructor;
import lombok.Getter;

/**
 * @author hs
 * @Description: {枚举}
 * @date 2019/12/16 17:19
 */
@Getter
@AllArgsConstructor
public enum RequestEnum {
    SYSTEM_PING("system_ping"),
    CATEGORY_ONE_QUERY("category_write_query"),
    CATEGORY_TWO_QUERY("category_two_query"),
    GOODS_QUERY("goods_write_query""),
    ORDERS_QUERY("orders_write_query");

    private String method;

    /**
     * 获取对应枚举类型
     *
     * @param method
     * @return
     */
    public static RequestEnum getEnum(String method) {
        for (RequestEnum requestEnum : RequestEnum.values()) {
            if (requestEnum.getMethod().equals(method)) {
                return requestEnum;
            }
        }
        return null;
    }

}
	/**
	*	controller方法
	*	service方法统一为map形参
	*
	/
    @PostMapping("/api/v1/json")
    @ApiOperation(value = "json", notes = "json")
    public ApiResponseResult openapiv1(@RequestBody ApiRequestParam apiRequestParam){
        RequestEnum requestEnum = RequestEnum.getEnum(apiRequestParam.getMethod());
        if (requestEnum == null) {
            return ApiResponseResult.send(ResultEnum.FAILURE_4);
        }
        Map<String, String> param = JSONUtil.toBean(apiRequestParam.getData(), Map.class);
        // 根据method获取对应内容
        switch (requestEnum) {
            case SYSTEM_PING:
                return systemService.ping(param);
            case CATEGORY_ONE_QUERY:
                return categoryOneService.query(param);
            case CATEGORY_TWO_QUERY:
                return categoryTwoService.query(param);
            case GOODS_QUERY:
                return goodsService.query(param);
            case ORDERS_QUERY:
                return ordersService.query(param);
            default:
                return ApiResponseResult.send(ResultEnum.OTHER);
        }
    }

第二种方式

定义一个枚举RequestEnum,保存method,和对应完全限定名和方法名并根据method获取到对应枚举,来进行反射获取对应类(不推荐)
优点: 使用反射不需要每次改controller层
缺点: 直接获取到对应类,没有spring特性了
具体代码如下:

import lombok.AllArgsConstructor;
import lombok.Getter;

/**
 * @author hs
 * @Description: {枚举}
 * @date 2019/12/16 17:19
 */
@Getter
@AllArgsConstructor
public enum RequestEnum {
    SYSTEM_PING("system_ping","com.handy.xxx.xxx.methodxx","xxx"),
	CATEGORY_ONE_QUERY("category_query","com.handy.methodxx","xxx"),
    CATEGORY_TQUERY("category_two_query","com.handy.methodxx","xxx"),
    GOODS_QUERY("goods_query","com.handy.xxx.xxx.methodxx","xxx"),
    ORDERS_QUERY("orders__query","com.handy.xxx.xxx.methodxx","xxx");

    private String method;
	private String interfaceName;
    private String methodName;

    /**
     * 获取对应枚举类型
     *
     * @param method
     * @return
     */
    public static RequestEnum getEnum(String method) {
        for (RequestEnum requestEnum : RequestEnum.values()) {
            if (requestEnum.getMethod().equals(method)) {
                return requestEnum;
            }
        }
        return null;
    }

}
	/**
	*	controller方法
	*	service方法统一为map形参
	*
	/
    @PostMapping("/api/v1/json")
    @ApiOperation(value = "json", notes = "json")
    public ApiResponseResult openapiv2(@RequestBody ApiRequestParam apiRequestParam) throws Exception {
        RequestEnum requestEnum = RequestEnum.getEnum(apiRequestParam.getMethod());
        if (requestEnum == null) {
            return ApiResponseResult.send(ResultEnum.FAILURE_4);
        }
        Class<?> mClass = Class.forName(requestEnum.getInterfaceName());
        Method method = mClass.getMethod(requestEnum.getMethodName(), Map.class);
        return (ApiResponseResult) method.invoke(mClass.newInstance(), JSONUtil.toBean(apiRequestParam.getData(), Map.class));
    }

第三种方式

定义一个枚举RequestEnum,保存method,和对应完全限定名和方法名并根据method获取到对应枚举,使用spring的方式获取对应类(推荐)
优点: 使用反射不需要每次改controller层,保留spring特性
缺点: 暂无
具体代码如下:

import lombok.AllArgsConstructor;
import lombok.Getter;

/**
 * @author hs
 * @Description: {枚举}
 * @date 2019/12/16 17:19
 */
@Getter
@AllArgsConstructor
public enum RequestEnum {
    SYSTEM_PING("system_ping","com.handy.xxx.xxx.methodxx","xxx"),
	CATEGORY_ONE_QUERY("category_query","com.handy.methodxx","xxx"),
    CATEGORY_TQUERY("category_two_query","com.handy.methodxx","xxx"),
    GOODS_QUERY("goods_query","com.handy.xxx.xxx.methodxx","xxx"),
    ORDERS_QUERY("orders__query","com.handy.xxx.xxx.methodxx","xxx");

    private String method;
	private String interfaceName;
    private String methodName;

    /**
     * 获取对应枚举类型
     *
     * @param method
     * @return
     */
    public static RequestEnum getEnum(String method) {
        for (RequestEnum requestEnum : RequestEnum.values()) {
            if (requestEnum.getMethod().equals(method)) {
                return requestEnum;
            }
        }
        return null;
    }

}
@PostMapping("/v1/json")
    @ApiOperation(value = "json", notes = "json")
    public ApiResponseResult openapiv3(@RequestBody ApiRequestParam apiRequestParam) throws Exception {
        RequestEnum requestEnum = RequestEnum.getEnum(apiRequestParam.getMethod());
        if (requestEnum == null) {
            return ApiResponseResult.send(ResultEnum.FAILURE_4);
        }
        Class<?> c = ClassUtils.forName(requestEnum.getInterfaceName(), ClassUtils.getDefaultClassLoader());
        Object obj = ContextUtil.getBean(c);

        Map map = null;
        if (StrUtil.isEmpty(apiRequestParam.getData())) {
            map = Maps.newHashMapWithExpectedSize(1);
        } else {
            map = JSONUtil.toBean(apiRequestParam.getData(), Map.class);
        }
        return (ApiResponseResult) obj.getClass().getDeclaredMethod(requestEnum.getMethodName(), Map.class).invoke(obj, map);
    }
  • 2
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在Java Web项目,可以通过以下步骤来限制接口每天只允许调用5万次: 1. 创建一个计数器类来接口调用次数,并在计数器类设置一个计数器变量,用于存储接口调用次数。 2. 在计数器类创建一个定时任务,用于在每天的零点重置计数器变量。 3. 在每个需要进行接口调用次数限制的接口方法,首先从计数器类获取当前接口调用次数,如果当前次数小于5万,则增加计数器变量的值,并继续执行接口的逻辑;否则,返回一个错误提示信息,告知用户接口调用次数已经达到上限。 4. 在项目启动时,创建计数器类的单例实例,并将其存储在全局变量,以便在所有的接口方法共享计数器实例。 下面是一个示例代码,演示如何实现接口每天只允许调用5万次的限制: ```java public class ApiCounter { private static ApiCounter instance = new ApiCounter(); private int count = 0; private Date lastResetTime; private ApiCounter() { reset(); scheduleResetTask(); } public static ApiCounter getInstance() { return instance; } public int getCount() { return count; } public void increaseCount() { count++; } public void reset() { count = 0; lastResetTime = new Date(); } private void scheduleResetTask() { TimerTask task = new TimerTask() { @Override public void run() { reset(); } }; Timer timer = new Timer(); Calendar calendar = Calendar.getInstance(); calendar.set(Calendar.HOUR_OF_DAY, 0); calendar.set(Calendar.MINUTE, 0); calendar.set(Calendar.SECOND, 0); Date firstResetTime = calendar.getTime(); if (firstResetTime.before(new Date())) { calendar.add(Calendar.DAY_OF_MONTH, 1); firstResetTime = calendar.getTime(); } timer.schedule(task, firstResetTime, TimeUnit.MILLISECONDS.convert(1, TimeUnit.DAYS)); } } ``` 在需要进行接口调用次数限制的接口方法,可以使用如下代码来限制每天最多调用5万次: ```java public void myApiMethod() { ApiCounter counter = ApiCounter.getInstance(); if (counter.getCount() >= 50000) { throw new RuntimeException("The daily limit of 50,000 calls has been reached."); } counter.increaseCount(); // Continue with the rest of the method logic. } ``` 这样,每次调用 myApiMethod 方法时,会先从 ApiCounter 实例获取当前的接口调用次数,如果超过了5万,则会返回一个错误提示信息,否则会增加计数器变量的值,并继续执行接口的逻辑。同时,定时任务会在每天的零点重置计数器变量,从而实现每天最多调用5万次的限制。 ### 回答2: 在Java的web项目,限制接口每天只允许调用5万次可以通过以下步骤实现: 1. 在项目引入一个计数器的功能,用于录每次接口调用次数。 2. 在项目定义一个计数器变量,用于录当天已经调用的次数。可以使用静态变量或者数据库的计数器表来存储这个值。 3. 在每次接口请求的地方,进行计数器的递增操作,即每次接口调用时将计数器变量加1。 4. 在接口调用之前,添加一个计数器检查功能,用于检查当前接口调用次数是否已经达到了上限。 5. 如果计数器检查结果显示接口调用次数已经达到上限(即达到5万次),则禁止继续调用接口,并返回相应的错误提示信息。 6. 如果计数器检查结果显示接口调用次数还未达到上限,那么就允许继续调用接口,继续执行接口的业务逻辑。 7. 在每天的第一个接口调用时,重置计数器变量,将计数器重置为0,重新开始计数。 通过上述步骤,就可以实现接口每天调用次数的限制,确保每天只允许调用5万次。当然,具体的实现方式还可以依据项目框架的不同而有所调整,例如可以使用拦截器、过滤器等技术来进行计数器的检查和控制。 ### 回答3: 在Java的web项目限制接口每天只允许调用5万次的方法有很多种,下面介绍几种常用的方法: 1. 使用计数器:通过在代码加入计数器,每次接口调用时,计数器加1。在数据库或缓存保存计数器的值,并定时清零。当计数器的值达到5万次时,后续的请求都会被拒绝。 2. 使用限流算法:可以使用像漏桶算法或令牌桶算法等限流算法,对接口调用进行限制。这些算法可以设置令牌生成速率,每次调用接口时,消耗令牌,当令牌数耗尽时拒绝后续请求。 3. 使用定时任务:可以通过定时任务来监控接口调用次数,并在每天的零点将计数器清零。通过定时任务可以定期重置接口调用次数,保证每天只允许调用5万次。 4. 使用限流框架:可以借助一些开源的限流框架来实现接口调用次数的限制,如Guava RateLimiter。这些框架提供了方便的接口限流功能,可以设置接口的访问速率,限制每天只允许调用5万次。 需要指出的是,以上方法只是对接口调用次数进行限制,并不能阻止恶意攻击或绕过限流的行为。如果需要更加安全可靠的限制,还需要考虑使用其他方式,如IP限制、验证码等手段来进一步加强接口的访问控制。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值