分布式锁

  • 数据库

    数据库是使用唯一索引不允许重复的特性(或自定义实现如乐观锁). 但持有锁的进程如果释放锁时异常则容易导致死锁.

  • zookeeper

    使用临时节点, watcher可以获得节点被删除的通知, 当客户端连接失效后, 临时节点清除, 所以这种情况下不会有死锁发生.

  • redis

    setNX实现

// 为了简单就不用配置文件了
@Configuration
public class RedisManager { private JedisPool jedisPool ; public RedisManager(){
JedisPoolConfig jedisPoolConfig = new JedisPoolConfig();
jedisPoolConfig.setMaxTotal(20);
jedisPoolConfig.setMaxIdle(10); this.jedisPool = new JedisPool(jedisPoolConfig, "127.0.0.1",6379);
} @Bean
public Jedis getJedis(){
if (this.jedisPool==null){
return null;
}
return jedisPool.getResource();
} }
// lock, unlock 方法这里没有用异常处理.
@Component
public class RedisLock { @Autowired
private Jedis jedis; public String lock(String key, int timeout){
String uuid = UUID.randomUUID().toString();
long lockTime = System.currentTimeMillis();
while(System.currentTimeMillis()-lockTime<timeout*1000) {
long rs = jedis.setnx(key, uuid);
if (rs==1){
jedis.expire(key, (int) timeout);
return uuid;
} // 保证key被设置了超时时间
if (jedis.ttl(key)==-1){
jedis.expire(key, (int) timeout);
}
}
return uuid;
} public boolean unlock(String key, String value){
while(true) {
jedis.watch(key);
String ret = jedis.get(key);
if (value.equals(ret)) {
Transaction transaction = jedis.multi();
transaction.del(key);
List<Object> transResult = transaction.exec();
if (transResult == null) {
continue;
}
jedis.unwatch();
return true;
}
jedis.unwatch();
break;
}
return false; } }

lua语言

redis可以使用lua脚本. lua是动态类型语言.

  • 减少网络开销

  • 原子操作

  • 代码可以复用

  • 轻量级

安装

tar -zxvf ...

make linux

make install

出现fatal error: readline/readline.h: No such file or directory 则需要运行

sudo apt-get install libreadline-dev

命令

  • 全局变量, 局部变量
a=1
print(a) local b=2 print(b) //局部变量,在一条语句 local array = {"a","b"} //数组定义
  • 逻辑表达式
      • / %

不等于使用~=

  • 逻辑运算

not (a and/or B)

  • 字符串连接

str1..str2

  • 字符串长度用#

#"hello"

  • 分支

if expression then elseif expression then end

  • 循环

while expression do ..... end

for i=1,100 do .... end

for i,v in iparis(array) do ..... end

  • 注释

单行使用-- ; 范围使用--[[ ..... ]]

  • 函数
function(params...)
--......
return a;
end

内置对象: String , Table tonumber()

  • lua 可以调用redis命令

redis.call() --需要引入库支持, redis内置的lua, 可以在redis内部执行

lua在Redis中的使用

  • 执行 eval 'lua script' keynumber key...

  • KEYS[] ARGV[] 入参, 必须大写

  • 执行lua脚本文件./redis-cli --eval luafile.lua keyparam1 , valueparam2 vlaueparam3

程序中使用lua

String lua = "local num = redis.call('incr',KEYS[1]) ... ";
jedis.eval();

可以使用摘要执行脚本, 这对大脚本文件比较有用.

jedis.evalsha(jedis.scriptLoad(lua),key,value);

分布式系列十一: Redis进阶的更多相关文章

  1. 分布式系列十: Redis安装和命令

    redis是一个开源的, 内存数据结构存储, 一般用来作为数据库,缓存和消息代理. Redis的优势 多种数据结构 字符类型String 散列类型Hash 列表类型List 集合类型Set 有序集合类 ...

  2. 分布式缓存技术redis学习系列

    分布式缓存技术redis学习系列(一)--redis简介以及linux上的安装以及操作redis问题整理 分布式缓存技术redis学习系列(二)--详细讲解redis数据结构(内存模型)以及常用命令 ...

  3. 【分布式缓存系列】Redis实现分布式锁的正确姿势

    一.前言 在我们日常工作中,除了Spring和Mybatis外,用到最多无外乎分布式缓存框架——Redis.但是很多工作很多年的朋友对Redis还处于一个最基础的使用和认识.所以我就像把自己对分布式缓 ...

  4. Redis进阶实践之十三 Redis的Redis-trib.rb文件详解

    一.简介     事先说明一下,本篇文章不涉及对redis-trib.rb源代码的分析,只是从使用的角度来阐述一下,对第一次使用的人来说很重要.redis-trib.rb是redis官方推出的管理re ...

  5. Redis系列(一):Redis的简介与安装

    原文链接(转载请注明出处):Redis系列(一):Redis的简介与安装 什么是 Redis Redis 是一个使用ANSI C 编写的开源.支持网络协议.基于内存.可选持久性的键值对数据库,它是一个 ...

  6. Redis进阶实践之十三 Redis的Redis-trib.rb脚本文件使用详解

    转载来源:http://www.cnblogs.com/PatrickLiu/p/8484784.html 一.简介 事先说明一下,本篇文章不涉及对redis-trib.rb源代码的分析,只是从使用的 ...

  7. Redis进阶实践之七Redis和Lua初步整合使用(转载 7)

    Redis进阶实践之七Redis和Lua初步整合使用 一.引言 Redis学了一段时间了,基本的东西都没问题了.从今天开始讲写一些redis和lua脚本的相关的东西,lua这个脚本是一个好东西,可以运 ...

  8. Redis进阶实践之四Redis的基本数据类型(转载4)

    Redis进阶实践之四Redis的基本数据类型 一.引言 今天正式开始了Redis的学习,如果要想学好Redis,必须先学好Redis的数据类型.Redis为什么会比以前的Memchaed等内存缓存软 ...

  9. Java分布式:分布式锁之Redis实现

    Java分布式:分布式锁之Redis实现 分布式锁系列教程重点分享锁实现原理 Redis锁原理 核心命令 Redis分布式锁的原理是基于其SETNX命令,我们来看SETNX的解释. 实现过程 使用SE ...

随机推荐

  1. PHP字符串函数、常量、数组排序

    PHP字符串函数.常量.数组排序 strlen() 说明:strlen(),可以统计字符串长度 用途:strlen() 常用于循环和其他函数,在确定字符串何时结束很重要时.(例如,在循环中,我们也许需 ...

  2. 用div画三角/矩形/圆

    1. 画三角 <!DOCTYPE html> <html> <head> <title></title> <meta charset= ...

  3. 关于Math.round()方法

    先上结论: 1.参数的小数点后第一位<5,运算结果为参数整数部分. 2.参数的小数点后第一位>5,运算结果为参数整数部分绝对值+1,符号(+ or -)不变. 3.参数的小数点后第一位=5 ...

  4. spring boot中配置日志log和热部署

    Java的日志有很多 个人强烈不推荐log4j ,推荐log4j2和logback 在高并发,多线程的环境下log4j1 的性能和log4j2相比可以用junk来形容  对就是junk.log4j2的 ...

  5. codeforces#1139E. Maximize Mex(逆处理,二分匹配)

    题目链接: http://codeforces.com/contest/1139/problem/E 题意: 开始有$n$个同学和$m$,每个同学有一个天赋$p_{i}$和一个俱乐部$c_{i}$,然 ...

  6. 证明与计算(2): 离散对数问题(Discrete logarithm Problem, DLP)

    离散对数问题,英文是Discrete logarithm Problem,有时候简写为Discrete log,该问题是十几个开放数学问题(Open Problems in Mathematics, ...

  7. PyCharm中Django项目主urls导入应用中views的红线问题

    PyCharm中Django项目主urls导入应用中views的红线问题 使用PyCharm学习Django框架,从项目的主urls中导入app中的views的时候,导入的包中下面有红线报错,但是却能 ...

  8. open live writer 测试

    什么是Hash表 1.定义 Hash(散列/哈希),就是把任意长度的输入(又叫做预映射, pre-image),通过散列算法,变换成固定长度的输出,该输出就是散列值.这种转换是一种压缩映射,也就是,散 ...

  9. 从JS的深拷贝与浅拷贝到jq的$.extend()方法

    一.堆内存与栈内存 堆和栈都是内存中划分出来的用来存储的区域,栈为自动分配的内存空间,它由系统自动释放,堆为动态分配的内存,大小不定也不会自动释放. 二.js基本数据类型与引用类型的不同 基本数据类型 ...

  10. Shodan的http.favicon.hash语法详解与使用技巧

    在Shodan搜索中有一个关于网站icon图标的搜索语法,http.favicon.hash,我们可以使用这个语法来搜索出使用了同一icon图标的网站,不知道怎么用的朋友请参考我上一篇文章. 通过上一 ...