redis 怎么使自增长主键调过一段继续生成
发布网友
发布时间:2022-04-11 13:59
我来回答
共2个回答
懂视网
时间:2022-04-11 18:21
INCR key
将 key 中储存的数字值增一。
如果 key 不存在,那么 key 的值会先被初始化为 0 ,然后再执行 INCR 操作。
如果值包含错误的类型,或字符串类型的值不能表示为数字,那么返回一个错误。
本操作的值限制在 64 位(bit)有符号数字表示之内。
这是一个针对字符串的操作,因为 Redis 没有专用的整数类型,所以 key 内储存的字符串被解释为十进制 64 位有符号整数来执行 INCR 操作。
可用版本:
>= 1.0.0
时间复杂度:
O(1)
返回值:
执行 INCR 命令之后 key 的值。
redis> SET page_view 20
OK
redis> INCR page_view
(integer) 21
redis> GET page_view # 数字值在 Redis 中以字符串的形式保存
"21"
计数器是 Redis 的原子性自增操作可实现的最直观的模式了,它的想法相当简单:每当某个操作发生时,向 Redis 发送一个 INCR 命令。
比如在一个 web 应用程序中,如果想知道用户在一年中每天的点击量,那么只要将用户 ID 以及相关的日期信息作为键,并在每次用户点击页面时,执行一次自增操作即可。
比如用户名是 peter ,点击时间是 2012 年 3 月 22 日,那么执行命令 INCR peter::2012.3.22 。
可以用以下几种方式扩展这个简单的模式:
可以通过组合使用 INCR 和 EXPIRE ,来达到只在规定的生存时间内进行计数(counting)的目的。
客户端可以通过使用 GETSET 命令原子性地获取计数器的当前值并将计数器清零,更多信息请参考 GETSET 命令。
使用其他自增/自减操作,比如 DECR 和 INCRBY ,用户可以通过执行不同的操作增加或减少计数器的值,比如在游戏中的记分器就可能用到这些命令。
更多redis知识请关注redis入门教程栏目。
热心网友
时间:2022-04-11 15:29
数据库自增什么的麻烦死了,尤其是往后还需要考虑到分布式处理,然后偷了个懒,直接redis来搞起...
下面上代码
先定义个主键生成策略接口,往后方便扩展
[java] view plain copy
/**
* 定义主键生成策略接口,以便修改扩展
* @author LeiYong
*
*/
public interface KeyGenerate {
/**
* 生成String类型主键
* @param em
* @return
*/
public String generateStringKey(KeyGenerateEnum em);
/**
* 生成long类型主键
* @param em
* @return
*/
public Long generateLongKey(KeyGenerateEnum em);
}
接下来提供一个替补队员,基础的主键生成方案,采用随机数(Long)或UUID(String)方式生成
[java] view plain copy
/**
* 基础主键生成策略,采用随机数或UUID+随机数
* @author LeiYong
*
*/
public class BaseKeyGenerate implements KeyGenerate{
@Override
public String generateStringKey(KeyGenerateEnum em) {
return StringUtil.getUUID()+StringUtil.getNonceStr(6);
}
@Override
public Long generateLongKey(KeyGenerateEnum em) {
return System.currentTimeMillis()*1000000+NumberUtil.random(6);
}
}
主角Redis的缓存主键类登场
[java] view plain copy
/**
* 采用Redis的方式产生自增长主键
* @author LeiYong
*
*/
public class RedisKeyGenerate implements KeyGenerate{
/**
* Logger for this class
*/
private final Logger logger = LoggerFactory.getLogger(RedisKeyGenerate.class);
private final String SEQUENCE_KEY = "SEQ";
@Autowired
@Qualifier("redisClientSupport")
private RedisClientSupport redisClientSupport;
@Override
public String generateStringKey(KeyGenerateEnum em) {
Long key = generateLongKey(em);
if (key!=null) {
return key.toString();
}
//为了防止redis挂了程序能够继续运行
return new BaseKeyGenerate().generateStringKey(em);
}
@Override
public Long generateLongKey(KeyGenerateEnum em) {
synchronized (em) {
try {
redisClientSupport.selectDB(SystemConfig.REDIS_INDEX_DB);
Long key = redisClientSupport.incrementHash(SEQUENCE_KEY, em.name(), 1l);
if (key != null) {
return key;
}
} catch (CacheAccessException e) {
logger.warn("generateStringKey(KeyGenerateEnum) - exception ignored", e); //$NON-NLS-1$
}
}
//为了防止redis挂了程序能够继续运行
return new BaseKeyGenerate().generateLongKey(em);
}
}