直接贴出代码,实现执行lua脚本的方法,用到的第三方类库是 StackExchange.Redis(nuget上有)
注:下面的代码是简化后的,实际使用要修改,
using System;
using System.Collections.Generic;
using System.Linq;
using System.Net;
using System.Threading.Tasks;
using StackExchange.Redis; namespace TL.Cloud.KV
{
public class Kv:IKv
{ private ConfigurationOptions _redisConfig;// = new ConfigurationOptions
private IDatabaseAsync _db;//=Connection.GetDatabase();
private IServer _server;// = GetServer();
private string _keyPrefix;// 区分前缀 /// <summary>
/// 每个微服务一个kv云实例,或共享云实例中(无独立前缀,服务间Key取名需要自行防止冲突)
/// </summary>
/// <param name="hostServiceName">宿主微服务</param>
public Kv(PublicCloudKvConfig config)
{
Init(config);
} private void Init(PublicCloudKvConfig config)
{
//独特的区分dev/100前缀,online可以没有
_keyPrefix = string.IsNullOrWhiteSpace(config.KeyPrefix)
?null
:config.KeyPrefix.Trim().ToLower()+':';//小写带冒号分割
//ali kv
var kvUrl = config.KvUrl; _redisConfig = ConfigurationOptions.Parse(kvUrl); _redisConfig.Password = config.KvPassword;
_redisConfig.SetDefaultPorts(); //自动填充默认端口
var connection = ConnectionMultiplexer.Connect(_redisConfig);
_db = connection.GetDatabase();
_server = GetServer(); } private IServer GetServer()
{
var config = new ConfigurationOptions
{
KeepAlive = ,
EndPoints = { _redisConfig.EndPoints[]},
AbortOnConnectFail = false,
AllowAdmin = true
};
var conn = ConnectionMultiplexer.Connect(config);
return conn.GetServer(config.EndPoints[]);
} public Task<RedisResult> EvalLua(string lua, IList<RedisKey> keys, IList<RedisValue> values)
{
if (_keyPrefix != null)
keys = keys.Select(p => p.Prepend(_keyPrefix)).ToList();//加上前缀
return _db.ScriptEvaluateAsync(lua, keys.ToArray(), values.ToArray());
} public async Task<RedisResult> EvalLua(byte[] luaSha1, IList<RedisKey> keys, IList<RedisValue> values)
{
if (_keyPrefix != null)
keys = keys.Select(p => p.Prepend(_keyPrefix)).ToList();//加上前缀
return await _db.ScriptEvaluateAsync(luaSha1, keys.ToArray(), values.ToArray());
} public async Task<byte[]> LoadLuaToServerAsync(string lua)
{
var sha1 = lua.CalcLuaSha1();//本地计算
if(!await _server.ScriptExistsAsync(sha1))//服务器上不存在
sha1 = await _server.ScriptLoadAsync(lua);//应该和计算的相同 return sha1;
}
}
}

下面是测试代码段

//下面是测试代码
[Test]
public async Task LoadLuaToServerAsync()
{
var key = "TestEvalLua10010";
var fieldContent = "testlua10011";
const string lua =
"redis.call('SET', KEYS[1], ARGV[1])\n" +
"return redis.call('GET', KEYS[1])\n"; var sha2 = lua.CalcLuaSha1();
var sha1 = await _kv.LoadLuaToServerAsync(lua);
Assert.AreEqual(, sha1.Length);
for (var i = ; i < ; i++)
Assert.AreEqual(sha1[i],sha2[i]); var keys = new List<RedisKey> { key };
var values = new List<RedisValue> { fieldContent };
var result = await _kv.EvalLua(sha1, keys, values);
Assert.AreEqual(fieldContent, (string)result);
}

计算sha1用到的方法

//计算lua的sha1结果,作为执行lua的参数
public static byte[] CalcLuaSha1(this string lua)
{
SHA1 sha1 = new SHA1CryptoServiceProvider();
var bytesSha1In = Encoding.Default.GetBytes(lua);
return sha1.ComputeHash(bytesSha1In);
}

c#中用lua脚本执行redis命令的更多相关文章

  1. Redis进阶之使用Lua脚本自定义Redis命令

    [本文版权归微信公众号"代码艺术"(ID:onblog)所有,若是转载请务必保留本段原创声明,违者必究.若是文章有不足之处,欢迎关注微信公众号私信与我进行交流!] 1.在Redis ...

  2. Lua脚本在redis分布式锁场景的运用

    目录 锁和分布式锁 锁是什么? 为什么需要锁? Java中的锁 分布式锁 redis 如何实现加锁 锁超时 retry redis 如何释放锁 不该释放的锁 通过Lua脚本实现锁释放 用redis做分 ...

  3. Lua脚本在Redis事务中的应用实践

    使用过Redis事务的应该清楚,Redis事务实现是通过打包多条命令,单独的隔离操作,事务中的所有命令都会按顺序地执行.事务在执行的过程中,不会被其他客户端发送来的命令请求所打断.事务中的命令要么全部 ...

  4. 运维实践-最新Nginx二进制构建编译lua-nginx-module动态链接Lua脚本访问Redis数据库读取静态资源隐式展现

    关注「WeiyiGeek」公众号 设为「特别关注」每天带你玩转网络安全运维.应用开发.物联网IOT学习! 希望各位看友[关注.点赞.评论.收藏.投币],助力每一个梦想. 本章目录 目录 0x0n 前言 ...

  5. nginx插入lua脚本访问redis

    目标:收集用户日志 流程: 浏览器端get方法将数据传到nginx服务 nginx收集到数据,执行内嵌lua脚本,访问redis,根据token获得用户id 将日志信息存入文件 1.nginx安装,参 ...

  6. 使用Lua 脚本实现redis 分布式锁,报错:ERR Error running script (call to f_8ea1e266485534d17ddba5af05c1b61273c30467): @user_script:10: @user_script: 10: Lua redis() command arguments must be strings or integers .

    在使用SpringBoot开发时,使用RedisTemplate执行 redisTemplate.execute(lockScript, redisList); 发现报错: ERR Error run ...

  7. redis如何执行redis命令

    Redis 命令 Redis 命令用于在 redis 服务上执行操作.所以我们必须要启动Redis服务程序,也就是redis安装目录下的redis-server.exe,你可以双击执行,也可以打开cm ...

  8. shell基础之脚本执行,命令别名以及快捷键等

    脚本执行方式 比如我们在/root/下编写了一个脚本,名字为hello.sh.那么怎么调用执行它呢?有两种办法: (1)直接通过bash,如下: bash  hello.sh 注:采用bash执行脚本 ...

  9. 脚本批量执行Redis命令

    1.将命令写在文件中 数据量比较大的话,建议用程序去生成文件.例如: List<String> planIdList = planDao.findAll().parallelStream( ...

随机推荐

  1. 成功安装的Sublime Text3

    安装指南 1:下载官网:https://www.sublimetext.com/3 2:安装指南:https://jingyan.baidu.com/article/b0b63dbfe1b8ff4a4 ...

  2. 安装tesserocr错误(未解决)

    在win10下使用pip install tesserocr安装时,始终报错,未解决问题 解压tesserocr-2.2.2.tar.gz该文件夹后,查看setup.py文件,发现似乎model只能再 ...

  3. Github链接及git学习心得总结

    众所周知GitHub已经是当下非常流行的代码托管库了,全世界有无数的程序员把他们的代码放在GitHub里.那比起云盘之类的工具,用GitHub有什么好处呢:1. 以后在帖子里只需要扔一个链接,大家就能 ...

  4. iOS中单例创建时不严格造成的问题和解决方法

    这次项目中遇到了一个单例创建不严格造成了的问题.简单说来就是在有的地方使用了alloc创建了多个实例,当然如果严格按照接口的方法调用是不会有问题的,但是如果项目碰到有不太熟悉的人使用时在处理时就会出现 ...

  5. PAT L2-027 名人堂与代金券

    https://pintia.cn/problem-sets/994805046380707840/problems/994805055176163328 对于在中国大学MOOC(http://www ...

  6. PDF文档打印太慢怎么办

    如下图,用Adobe Acrobat打开PDF文件,然后[高级]-打勾[作为图像打印]即可

  7. 使用kindeditor来替换ecshop的fckeditor编辑器,让ecshop可以批量上传图片

    老杨原创 kindeditor此编辑器可以让ecshop批量上传图片,可以插入代码,可以全屏编辑,可以插入地图.视频,进行更多word操作,设置字体. 步骤一:进入kindeditor的官网,http ...

  8. Activiti动态设置办理人扩展

    关键词:Assignee.Candidate users.Candidate groups:setAssignee.taskCandidateUser.taskCandidateGroup 主要解决问 ...

  9. React learn path

    React learn path The Road to learn React https://github.com/the-road-to-learn-react https://roadtore ...

  10. GUI and Usability Test Scenarios

    1 all fields on page (e.g. text box ,radio options, dropdown lists) should be aligned properly2 Nume ...