当前位置: 首页 > news >正文

如何建设和优化网站东莞seo全网营销

如何建设和优化网站,东莞seo全网营销,佛山营销型网站搭建,网站的营销与推广场景#xff1a;限制请求后端接口的频率#xff0c;例如1秒钟只能请求次数不能超过10次#xff0c;通常的写法是#xff1a; 1.先去从redis里面拿到当前请求次数 2.判断当前次数是否大于或等于限制次数 3.当前请求次数小于限制次数时进行自增 这三步在请求不是很密集的时…场景限制请求后端接口的频率例如1秒钟只能请求次数不能超过10次通常的写法是 1.先去从redis里面拿到当前请求次数 2.判断当前次数是否大于或等于限制次数 3.当前请求次数小于限制次数时进行自增 这三步在请求不是很密集的时候程序执行很快可能不会产生问题如果两个请求几乎在同一时刻到来我们第1步和第2步的判断是无法保证原子性的。 改进方式使用redis的lua脚本将读取值、判断大小、自增放到redis的一次操作中redis底层所有的操作请求都是串行的也就是一个请求执行完才会执行下一个请求。 自增的lua脚本如下 /*** 自增过期时间的原子性脚本*/private String maxCountScriptText() {return local key KEYS[1]\n local count tonumber(ARGV[1])\n local time tonumber(ARGV[2])\n local current redis.call(get, key);\n if current and tonumber(current) count then\n return tonumber(current);\n end\n current redis.call(incr, key)\n if tonumber(current) 1 then\n redis.call(expire, key, time)\n end\n return tonumber(current);;} 将接口限流功能封装成一个注解RateLimiter在接口方法上面加上RateLimiter就可以实现限流 redis工具类 package com.zhou.redis.util;import com.zhou.redis.dto.MyRedisMessage; import com.zhou.redis.exception.LockException; import com.zhou.redis.script.MaxCountQueryScript; import com.zhou.redis.script.MaxCountScript; import lombok.extern.slf4j.Slf4j; import org.springframework.context.annotation.Configuration; import org.springframework.data.redis.core.HashOperations; import org.springframework.data.redis.core.RedisTemplate; import org.springframework.data.redis.core.ValueOperations;import java.time.Duration; import java.util.Collections; import java.util.List; import java.util.concurrent.TimeUnit;Configuration Slf4j public class RedisUtil {public RedisTemplateString, Object redisTemplate;private MaxCountScript maxCountScript;private MaxCountQueryScript maxCountQueryScript;public RedisUtil(RedisTemplate redisTemplate, MaxCountScript maxCountScript, MaxCountQueryScript maxCountQueryScript) {this.redisTemplate redisTemplate;this.maxCountScript maxCountScript;this.maxCountQueryScript maxCountQueryScript;}/*** 尝试加锁返回加锁成功或者失败* param time 秒**/public boolean tryLock(String key,Object value,Long time){if(time null || time 0){time 30L;}Boolean b redisTemplate.opsForValue().setIfAbsent(key, value, Duration.ofSeconds(time));return b null ? false : b;}/*** 释放锁拿到锁之后才能调用释放锁**/public boolean unLock(String key){Boolean b redisTemplate.delete(key);return b null ? false : b;}/*** 对key进行自增1* param maxCount 最大值* param time 增加次数* return 自增后的值*/public Long incr(String key,int maxCount, int time){ListString keys Collections.singletonList(key);return redisTemplate.execute(maxCountScript, keys, maxCount, time);}/*** 获得当前值*/public Long incrNow(String key){ListString keys Collections.singletonList(key);return redisTemplate.execute(maxCountQueryScript, keys);} } redis配置类 package com.zhou.redis.config;import com.fasterxml.jackson.annotation.JsonAutoDetect; import com.fasterxml.jackson.annotation.PropertyAccessor; import com.fasterxml.jackson.databind.ObjectMapper; import com.zhou.redis.listener.MyRedisListener; import com.zhou.redis.script.MaxCountQueryScript; import com.zhou.redis.script.MaxCountScript; import com.zhou.redis.util.RedisTopic; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.data.redis.connection.RedisConnectionFactory; import org.springframework.data.redis.core.RedisTemplate; import org.springframework.data.redis.listener.PatternTopic; import org.springframework.data.redis.listener.RedisMessageListenerContainer; import org.springframework.data.redis.listener.Topic; import org.springframework.data.redis.listener.adapter.MessageListenerAdapter; import org.springframework.data.redis.serializer.Jackson2JsonRedisSerializer; import org.springframework.data.redis.serializer.StringRedisSerializer;import java.util.Arrays; import java.util.List;Configuration public class RedisConfig {SuppressWarnings(all)Beanpublic RedisTemplateString, Object redisTemplate(RedisConnectionFactory factory) {RedisTemplateString, Object template new RedisTemplate();template.setConnectionFactory(factory);//Json序列化配置Jackson2JsonRedisSerializerObject jackson2JsonRedisSerializer new Jackson2JsonRedisSerializer(Object.class);ObjectMapper om new ObjectMapper();om.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);jackson2JsonRedisSerializer.setObjectMapper(om);//String的序列化StringRedisSerializer stringRedisSerializer new StringRedisSerializer();//key采用string的序列化template.setKeySerializer(stringRedisSerializer);//hash的key采用string的序列化template.setHashKeySerializer(stringRedisSerializer);//value序列化采用jacksontemplate.setValueSerializer(jackson2JsonRedisSerializer);//hash的value序列化方式采用jacksontemplate.setHashValueSerializer(jackson2JsonRedisSerializer);template.afterPropertiesSet();return template;}/*** Redis消息监听器容器* 这个容器加载了RedisConnectionFactory和消息监听器* 可以添加多个监听不同话题的redis监听器只需要把消息监听器和相应的消息订阅处理器绑定该消息监听器* 通过反射技术调用消息订阅处理器的相关方法进行一些业务处理** param redisConnectionFactory 连接工厂* param adapter 适配器* return redis消息监听容器*/BeanSuppressWarnings(all)public RedisMessageListenerContainer container(RedisConnectionFactory redisConnectionFactory,FuncUpdateListener listener,MessageListenerAdapter adapter) {RedisMessageListenerContainer container new RedisMessageListenerContainer();// 监听所有库的key过期事件container.setConnectionFactory(redisConnectionFactory);// 所有的订阅消息都需要在这里进行注册绑定,new PatternTopic(TOPIC_NAME1)表示发布的主题信息// 可以添加多个 messageListener配置不同的通道ListTopic topicList Arrays.asList(new PatternTopic(RedisTopic.TOPIC1),new PatternTopic(RedisTopic.TOPIC2));container.addMessageListener(listener, topicList);/*** 设置序列化对象* 特别注意1. 发布的时候需要设置序列化订阅方也需要设置序列化* 2. 设置序列化对象必须放在[加入消息监听器]这一步后面否则会导致接收器接收不到消息*/Jackson2JsonRedisSerializer seria new Jackson2JsonRedisSerializer(Object.class);ObjectMapper objectMapper new ObjectMapper();objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);seria.setObjectMapper(objectMapper);container.setTopicSerializer(seria);return container;}/*** 这个地方是给messageListenerAdapter 传入一个消息接受的处理器利用反射的方法调用“receiveMessage”* 也有好几个重载方法这边默认调用处理器的方法 叫OnMessage*/SuppressWarnings(all)Beanpublic MessageListenerAdapter listenerAdapter() {//MessageListenerAdapter receiveMessage new MessageListenerAdapter(printMessageReceiver, receiveMessage);MessageListenerAdapter receiveMessage new MessageListenerAdapter();Jackson2JsonRedisSerializer seria new Jackson2JsonRedisSerializer(Object.class);ObjectMapper objectMapper new ObjectMapper();objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);seria.setObjectMapper(objectMapper);receiveMessage.setSerializer(seria);return receiveMessage;}Beanpublic MaxCountScript maxCountScript() {return new MaxCountScript(maxCountScriptText());}Beanpublic MaxCountQueryScript maxCountQueryScript() {return new MaxCountQueryScript(maxCountQueryScriptText());}/*** 自增过期时间的原子性脚本*/private String maxCountScriptText() {return local key KEYS[1]\n local count tonumber(ARGV[1])\n local time tonumber(ARGV[2])\n local current redis.call(get, key);\n if current and tonumber(current) count then\n return tonumber(current);\n end\n current redis.call(incr, key)\n if tonumber(current) 1 then\n redis.call(expire, key, time)\n end\n return tonumber(current);;/*return local limitMaxCount tonumber(ARGV[1])\n local limitSecond tonumber(ARGV[2])\n local num tonumber(redis.call(get, KEYS[1]) or -1)\n if limitMaxCount then\n return -1\n end\n if num -1 then\n redis.call(incr, KEYS[1])\n redis.call(expire, KEYS[1], limitSecond)\n return 1\n else\n if num limitMaxCount then\n return 0\n else\n redis.call(incr, KEYS[1])\n return 1\n end\n end;*/}/*** 查询当前值脚本*/private String maxCountQueryScriptText() {return local key KEYS[1]\n local current redis.call(get, key);\n if current then\n return tonumber(current);\n else\n return current\n end\n;} } 拦截模式枚举类根据ip拦截或者方法拦截 package com.zhou.aop;/*** author lang.zhou* since 2023/1/31 17:56*/ public enum LimitType {IP,DEFAULT } 封装自定义注解RateLimiter package com.zhou.aop;import java.lang.annotation.*;/*** author lang.zhou* since 2023/1/31 17:49*/ Target(ElementType.METHOD) Retention(RetentionPolicy.RUNTIME) Documented public interface RateLimiter {/*** 限流key*/String key() default RateLimiter;/*** 限流时间,单位秒*/int time() default 60;/*** 限流次数*/int count() default 100;/*** 限流类型*/LimitType limitType() default LimitType.DEFAULT;/*** 限流后返回的文字*/String limitMsg() default 访问过于频繁请稍候再试; }注解的切面逻辑 package com.zhou.aop;import com.zhou.redis.util.RedisUtil; import com.zhou.common.utils.IpUtil; import lombok.extern.slf4j.Slf4j; import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.aspectj.lang.reflect.MethodSignature; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Component; import org.springframework.web.context.request.RequestContextHolder; import org.springframework.web.context.request.ServletRequestAttributes;import javax.servlet.http.HttpServletRequest; import java.lang.reflect.Method;/*** 接口限流切面* author lang.zhou* since 2023/1/31 17:50*/ Aspect Slf4j Component public class RateLimiterAspect {Autowiredprivate RedisUtil redisUtils;Before(annotation(rateLimiter))public void doBefore(JoinPoint point, RateLimiter rateLimiter) {int time rateLimiter.time();int count rateLimiter.count();String combineKey getCombineKey(rateLimiter, point);try {Long number redisUtils.incr(combineKey, count, time);if (number null || number.intValue() count){log.info(请求【{}】被拦截{}秒内请求次数{},combineKey,time,number);throw new RuntimeException(rateLimiter.limitMsg());}} catch (ServiceRuntimeException e) {throw e;} catch (Exception e) {throw new RuntimeException(网络繁忙请稍候再试);}}/*** 获取限流key*/public String getCombineKey(RateLimiter rateLimiter, JoinPoint point) {StringBuilder s new StringBuilder(rateLimiter.key());ServletRequestAttributes requestAttributes (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();if(requestAttributes ! null){HttpServletRequest request requestAttributes.getRequest();if (rateLimiter.limitType() LimitType.IP) {s.append(IpUtil.getIpAddr(request)).append(-);}}MethodSignature signature (MethodSignature) point.getSignature();Method method signature.getMethod();Class? targetClass method.getDeclaringClass();s.append(targetClass.getName()).append(.).append(method.getName());return s.toString();}} lua自增脚本类 package com.zhou.redis.script;import org.springframework.data.redis.core.script.DefaultRedisScript;/*** author lang.zhou* since 2023/2/25*/ public class MaxCountScript extends DefaultRedisScriptLong {public MaxCountScript(String script) {super(script,Long.class);} } lua查询当前值的脚本类 package com.zhou.redis.script;import org.springframework.data.redis.core.script.DefaultRedisScript;/*** author lang.zhou* since 2023/2/25*/ public class MaxCountQueryScript extends DefaultRedisScriptLong {public MaxCountQueryScript(String script) {super(script,Long.class);} } 订阅消息通道的枚举 package com.zhou.redis.util;public class RedisTopic {public static final String TOPIC1 TOPIC1;public static final String TOPIC2 TOPIC2; }消息实体类  package com.zhou.redis.dto;import lombok.Data;import java.io.Serializable;/*** redis订阅消息实体* since 2022/11/11 17:34*/ Data public class MyRedisMessage implements Serializable {private String msg; } 订阅消息监听器  package com.zhou.redis.listener;import com.zhou.redis.dto.MyRedisMessage; import com.zhou.redis.util.RedisTopic; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.data.redis.connection.Message; import org.springframework.data.redis.connection.MessageListener; import org.springframework.data.redis.core.RedisTemplate; import org.springframework.stereotype.Component;import javax.script.ScriptException;/*** author lang.zhou*/ Slf4j Component public class MyRedisListener implements MessageListener {Autowiredprivate RedisTemplateString,Object redisTemplate;Overridepublic void onMessage(Message message, byte[] pattern) {String topic new String(pattern);// 接收的topiclog.info(channel:{} , topic);if(RedisTopic.TOPIC1.equals(topic)){//}else if(RedisTopic.TOPIC2.equals(topic)){//序列化对象特别注意发布的时候需要设置序列化订阅方也需要设置序列化MyRedisMessage msg (MyRedisMessage) redisTemplate.getValueSerializer().deserialize(message.getBody());log.info(message:{},msg);}} } 注解使用方式1秒内一个ip最多只能请求10次 RestController RequestMapping(/test/api) public class CheckController{PostMapping(/limit)RateLimiter(time 1, count 10, limitType LimitType.IP, limitMsg 请求过于频繁请稍后重试)public void limit(HttpServletRequest request){//执行业务代码}}
http://www.hkea.cn/news/14407862/

相关文章:

  • vps里面设置了一下读取和写入网站无法显示了鞍山市建设工程安全生产监督管理站网站
  • 怎样做好手机网站建设济源建网站
  • 岳阳网站开发收费如何注册电商平台
  • 天津知名网站建设公司呼市网站制作招聘
  • 营销网站制作软件响应式网站怎样做
  • 西部数码网站模板试论述外贸网站建设应注意的问题
  • 网站搭建有免费的吗茂名网站建设维护
  • 新闻类的网站如何做优化平面设计公司平面图
  • 备案个人网站网站设计特色
  • 腾讯网站建设推广宁波市建设网
  • 注册服务器网站哪个好雅安建设局网站
  • 成都建设网站 scgckj网站建设个人网站
  • 做网站对象存储网站制作com
  • 网站开发汇报pc微信二维码永久入口
  • 做旅游网站公司自问自答网站怎么做
  • 专门做海产品的网站企业形象设计报价
  • CP网站开发制作H5找工程包工平台
  • 建小网站多少钱视频信号无线传输设备
  • 网站开发的代码合肥网站建设王道下拉強
  • 企业网站建设技网站升级建设费用吗
  • 赤峰网站制作公司铜仁建设厅官方网站
  • 做教育行业营销类型的网站免费发帖平台
  • 网站一般用什么语言写访问网站的过程
  • 网站建设可信赖北京朝阳区一小区现疑似病例
  • 24小时24元网站建设运维网站制作
  • 张掖专业做网站的公司建设行官方网站
  • 家教辅导培训网站建设万网域名注册官网gname
  • 爱站网综合查询网站建设推广者怎样找到客户
  • 肇东网站建设wordpress用ip访问不了
  • 宁波手机网站开发trellis wordpress