您好,欢迎访问一九零五行业门户网

Redis缓存服务器在Windows下的使用

一、redis服务端
首先下载redis服务器,点击前往下载.msi版本,双击安装redis服务端就有了,并以服务的形式随系统一起启动:
安装好redis服务器之后第一件事就是设置密码,进入安装目录:c:\program files\redis - 找到配置文件:redis.windows-service.conf - 找到:# requirepass foobared - 回车换行加上:requirepass 这里写自己的新密码(顶行写,前面不要留空格) - 到服务里重启redis服务,或者重启电脑
不设置密码的坏处,看看携程这哥们的遭遇就知道了:记一次redis被攻击的事件
二、redis客户端(命令行和可视化工具rdm)
命令行方式演示:启动redis客户端、读写redis服务器
上图命令解释:
cd c:\program files\redis:cd命令进入redis安装目录,相当于windows系统里双击进入redis的安装目录
redis-cli.exe:打开redis-cli客户端程序,相当于windows系统里双击运行一个exe程序(安装了上面的redis服务端程序,需要一个客户端程序连接这个服务端。连接本机redis服务器直接敲此命令,连接远程的需要加ip和端口,例:redis-cli.exe -h 111.11.11.111 -p 6379)
keys *:查看所有键值对(如果redis服务器设置了密码,这条命令会报错,需要先输入密码,执行此命令:auth 你的密码)
set blog oppoic.cnblogs.com:设置一个键值对,键是:blog,值是:oppoic.cnblogs.com(按目录存储:set 目录名:键 值)
get blog:获取键为blog对应的值
keys *:查看所有键值对
其他常用命令:
config get dir:获取redis安装目录
ping:返回pong表示redis服务器正常
redis-cli.exe:进入第一个数据库(默认),redis一共0到15共16个库,进入第三个库 redis-cli -n 2(已经进去了,select 0~15 随意切换)
quit:退出redis程序
exit:退出dos窗口
flushdb:删除当前选择数据库中的所有key
flushall:删除所有数据库中的数据库
更多命令:https://redis.io/commands
至此,一个运行在本机的redis缓存服务器已经搭建完成,并且可以读写了。但是命令行显然对小白用户不友好,可视化工具登场:redis desktop manager(https://redisdesktop.com/download)
左侧树显示已经有一个连接了,点击底部的connect to redis server再添加一个连接:
name:连接名称,随便起
host:主机地址,本机就是127.0.0.1,远程的输入对应ip
port:端口,redis服务器默认端口6379
auth:密码,设置了就输,没设置留空
连上redis服务器就可以看到,默认16个库(配置文件可改),索引从0开始。常见用法是一个项目一个库,项目下不同功能模块分不同目录存在这个库下。
有了可视化工具之后的操作就不用说了,双击,右键新建、删除。。。会用windows系统的都会用这个工具。相比于命令行,redis desktop manager这个可视化工具更友好,调试远程服务器上的数据也更方便,指哪打哪。
注:本机可以这样,连接远程服务器需要到服务器上的redis安装目录下,找到redis.windows-service.conf文件,找到bind 127.0.0.1 前面加#注释掉,然后到服务里右键重启redis服务,或者重启windows系统
三、c#操作redis服务器
以上都是命令行和可视化工具操作redis服务器,c#程序操作redis需要借助stackexchange.redis(https://github.com/stackexchange/stackexchange.redis),为了统一调用,封装了一个redishelper帮助类:
using newtonsoft.json; using stackexchange.redis; using system; using system.componentmodel; using system.configuration; using system.reflection; using system.text; namespace redis_demo { /// <summary> /// redis 帮助类 /// </summary> public static class redishelper { private static string _conn = configurationmanager.appsettings["redis_connection_string"] "127.0.0.1:6379"; private static string _pwd = configurationmanager.appsettings["redis_connection_pwd"] "123456"; static connectionmultiplexer _redis; static readonly object _locker = new object(); #region 单例模式 public static connectionmultiplexer manager { get { if (_redis == null) { lock (_locker) { if (_redis != null) return _redis; _redis = getmanager(); return _redis; } } return _redis; } } private static connectionmultiplexer getmanager(string connectionstring = null) { if (string.isnullorempty(connectionstring)) { connectionstring = _conn; } var options = configurationoptions.parse(connectionstring); options.password = _pwd; return connectionmultiplexer.connect(options); } #endregion /// <summary> /// 添加 /// </summary> /// <param name="folder">目录</param> /// <param name="key">键</param> /// <param name="value">值</param> /// <param name="expireminutes">过期时间,单位:分钟。默认600分钟</param> /// <param name="db">库,默认第一个。0~15共16个库</param> /// <returns></returns> public static bool stringset(cachefolderenum folder, string key, string value, int expireminutes = 600, int db = -1) { string fd = getdescription(folder); return manager.getdatabase(db).stringset(string.isnullorempty(fd) ? key : fd + ":" + key, value, timespan.fromminutes(expireminutes)); } /// <summary> /// 获取 /// </summary> /// <param name="folder">目录</param> /// <param name="key">键</param> /// <param name="db">库,默认第一个。0~15共16个库</param> /// <returns></returns> public static string stringget(cachefolderenum folder, string key, int db = -1) { try { string fd = getdescription(folder); return manager.getdatabase(db).stringget(string.isnullorempty(fd) ? key : fd + ":" + key); } catch (exception) { return null; } } /// <summary> /// 删除 /// </summary> /// <param name="folder">目录</param> /// <param name="key">键</param> /// <param name="db">库,默认第一个。0~15共16个库</param> /// <returns></returns> public static bool stringremove(cachefolderenum folder, string key, int db = -1) { try { string fd = getdescription(folder); return manager.getdatabase(db).keydelete(string.isnullorempty(fd) ? key : fd + ":" + key); } catch (exception) { return false; } } /// <summary> /// 是否存在 /// </summary> /// <param name="key">键</param> /// <param name="db">库,默认第一个。0~15共16个库</param> public static bool keyexists(cachefolderenum folder, string key, int db = -1) { try { string fd = getdescription(folder); return manager.getdatabase(db).keyexists(string.isnullorempty(fd) ? key : fd + ":" + key); } catch (exception) { return false; } } /// <summary> /// 延期 /// </summary> /// <param name="folder">目录</param> /// <param name="key">键</param> /// <param name="min">延长时间,单位:分钟,默认600分钟</param> /// <param name="db">库,默认第一个。0~15共16个库</param> public static bool addexpire(cachefolderenum folder, string key, int min = 600, int db = -1) { try { string fd = getdescription(folder); return manager.getdatabase(db).keyexpire(string.isnullorempty(fd) ? key : fd + ":" + key, datetime.now.addminutes(min)); } catch (exception) { return false; } } /// <summary> /// 添加实体 /// </summary> /// <param name="folder">目录</param> /// <param name="key">键</param> /// <param name="t">实体</param> /// <param name="ts">延长时间,单位:分钟,默认600分钟</param> /// <param name="db">库,默认第一个。0~15共16个库</param> public static bool set<t>(cachefolderenum folder, string key, t t, int expireminutes = 600, int db = -1) { string fd = getdescription(folder); var str = jsonconvert.serializeobject(t); return manager.getdatabase(db).stringset(string.isnullorempty(fd) ? key : fd + ":" + key, str, timespan.fromminutes(expireminutes)); } /// <summary> /// 获取实体 /// </summary> /// <param name="folder">目录</param> /// <param name="key">键</param> /// <param name="db">库,默认第一个。0~15共16个库</param> public static t get<t>(cachefolderenum folder, string key, int db = -1) where t : class { string fd = getdescription(folder); var strvalue = manager.getdatabase(db).stringget(string.isnullorempty(fd) ? key : fd + ":" + key); return string.isnullorempty(strvalue) ? null : jsonconvert.deserializeobject<t>(strvalue); } /// <summary> /// 获得枚举的description /// </summary> /// <param name="value">枚举值</param> /// <param name="nameinstead">当枚举值没有定义descriptionattribute,是否使用枚举名代替,默认是使用</param> /// <returns>枚举的description</returns> private static string getdescription(this enum value, boolean nameinstead = true) { type type = value.gettype(); string name = enum.getname(type, value); if (name == null) { return null; } fieldinfo field = type.getfield(name); descriptionattribute attribute = attribute.getcustomattribute(field, typeof(descriptionattribute)) as descriptionattribute; if (attribute == null && nameinstead == true) { return name; } return attribute == null ? null : attribute.description; } } }
向redis服务器第一个库的fd1目录里,添加一个键为name,值为wangjie的记录:
redishelper.stringset(cachefolderenum.folder1, "name", "wangjie");
获取这条记录:
string key = redishelper.stringget(cachefolderenum.folder1, "name"); console.writeline("键为name的记录对应的值:" + key);
删除这条记录:
bool result = redishelper.stringremove(cachefolderenum.folder1, "name"); if (result) { console.writeline("键为name的记录删除成功"); } else { console.writeline("键为name的记录删除失败"); }
查询这条记录是否存在:
bool ifexist = redishelper.keyexists(cachefolderenum.folder1, "name"); if (ifexist) { console.writeline("键为name的记录存在"); } else { console.writeline("键为name的记录不存在"); }
向redis服务器第二个库的fd2目录里,添加一个键为sd1,值为一个对象的记录:
student student = new student() { id = 1, name = "张三", class = "三年二班" }; redishelper.set<student>(cachefolderenum.folder2, "sd1", student, 10, 1);
获取这个对象:
student sdget = redishelper.get<student>(cachefolderenum.folder2, "sd1", 1); if (sdget != null) { console.writeline("id:" + sdget.id + " name:" + sdget.name + " class:" + sdget.class); } else { console.writeline("找不到键为sd1的记录"); }
源码:(http://files.cnblogs.com/files/oppoic/redis_demo.zip)
环境是vs 2013,如果跑不起来自行把cs里的代码拷出来编译运行
四、其他
msopentech开发redis缓存服务器自带持久化,写入之后重启电脑键值对还存在,一般写入键值对要设置过期时间,否则一直占用内存不会被释放。redis存储方式不光有键对应字符串,还有对应list,hashtable等,当然redis更多高阶的用法还是在linux下。
更多redis缓存服务器在windows下的使用。
其它类似信息

推荐信息