Redis とは
Redis は、ANSI C 言語で書かれたオープンソースのログタイプの Key-Value データベースで、ネットワークをサポートし、メモリベースで永続化できます。
#1. 他のユーザー状態保存ソリューションとの比較
一般的に、開発ではユーザー状態はセッションを使用します。または Cookie のどちらの方法にもさまざまな長所と短所があります。
セッション: InProc モードでは簡単に失われ、同時実行の問題が発生します。 SQLServer または SQLServer モードを使用してパフォーマンスを消費する場合
Cookie によって一部のユーザー情報が簡単に公開される可能性があり、暗号化と復号化によってもパフォーマンスが消費されます。
Redis は、このソリューションを使用していくつかの問題を解決します (
1)。Redis はアクセス速度が高速です。
2. ユーザー データは簡単には失われません。
3. ユーザーが多い場合、クラスターをサポートするのは簡単です。
4. オンライン ユーザーを表示できます。
5. ユーザーが 1 か所でログインできるようにします。
6. 永続性をサポートします。
2. 実装のアイデア
1. セッションは実際にセッション ID を Cookie に保存し、ユーザーは毎回そのセッション ID をサーバーに送信することがわかっています。彼が訪問すると、サーバーは ID を介してユーザーに対応するステータス データを検索します。
ここでの私のアプローチは、Cookie にセッション ID を定義することです。プログラムがユーザーのステータスを取得する必要がある場合、セッション ID は Redis で検索するためのキーとして使用されます。
2.同時に、セッションは、ユーザーが一定期間アクセスしない場合にセッションをリサイクルすることをサポートします。
Redis の Keys の有効期限機能を借用してこの機能をサポートしますが、更新の観点から、プログラムはリクエストをインターセプトし、このメソッドを独自に呼び出す必要があります (デモには例があります)
開始コードの説明の下
#3.Redis 呼び出しインターフェイス
<appSettings> <!--每台Redis之间用,分割.第一个必须为主机--> <add key="SessionRedis" value="127.0.0.1:6384,127.0.0.1:6384"/> </appSettings>
初期設定構成
static Managers() { string sessionRedis= ConfigurationManager.AppSettings["SessionRedis"]; string timeOut = ConfigurationManager.AppSettings["SessionRedisTimeOut"]; if (string.IsNullOrEmpty(sessionRedis)) { throw new Exception("web.config 缺少配置SessionRedis,每台Redis之间用,分割.第一个必须为主机"); } if (string.IsNullOrEmpty(timeOut)==false) { TimeOut = Convert.ToInt32(timeOut); } var host = sessionRedis.Split(char.Parse(",")); var writeHost = new string[] { host[0] }; var readHosts = host.Skip(1).ToArray(); ClientManagers = new PooledRedisClientManager(writeHost, readHosts, new RedisClientManagerConfig { MaxWritePoolSize = writeReadCount,//“写”链接池链接数 MaxReadPoolSize = writeReadCount,//“读”链接池链接数 AutoStart = true }); }制御の便宜のためにデリゲートを記述します
/// <summary> /// 写入 /// </summary> /// <typeparam name="F"></typeparam> /// <param name="doWrite"></param> /// <returns></returns> public F TryRedisWrite<F>(Func<IRedisClient, F> doWrite) { PooledRedisClientManager prcm = new Managers().GetClientManagers(); IRedisClient client = null; try { using (client = prcm.GetClient()) { return doWrite(client); } } catch (RedisException) { throw new Exception("Redis写入异常.Host:" + client.Host + ",Port:" + client.Port); } finally { if (client != null) { client.Dispose(); } } }例詳細についてはソース コードを参照してください
/// <summary> /// 以Key/Value的形式存储对象到缓存中 /// </summary> /// <typeparam name="T">对象类别</typeparam> /// <param name="value">要写入的集合</param> public void KSet(Dictionary<string, T> value) { Func<IRedisClient, bool> fun = (IRedisClient client) => { client.SetAll<T>(value); return true; }; TryRedisWrite(fun); }
4. セッションを実装します
上記のようにセッション ID を cookie に書き込みます
/// <summary> /// 用户状态管理 /// </summary> public class Session { /// <summary> /// 初始化 /// </summary> /// <param name="_context"></param> public Session(HttpContextBase _context) { var context = _context; var cookie = context.Request.Cookies.Get(SessionName); if (cookie == null || string.IsNullOrEmpty(cookie.Value)) { SessionId = NewGuid(); context.Response.Cookies.Add(new HttpCookie(SessionName, SessionId)); context.Request.Cookies.Add(new HttpCookie(SessionName, SessionId)); } else { SessionId = cookie.Value; } } }ユーザーへのアクセス方法
/// <summary> /// 获取当前用户信息 /// </summary> /// <typeparam name="T"></typeparam> /// <returns></returns> public object Get<T>() where T:class,new() { return new RedisClient<T>().KGet(SessionId); } /// <summary> /// 用户是否在线 /// </summary> /// <returns></returns> public bool IsLogin() { return new RedisClient<object>().KIsExist(SessionId); } /// <summary> /// 登录 /// </summary> /// <typeparam name="T"></typeparam> /// <param name="obj"></param> public void Login<T>(T obj) where T : class,new() { new RedisClient<T>().KSet(SessionId, obj, new TimeSpan(0, Managers.TimeOut, 0)); }
6. 更新
デフォルトのユーザーはログアウトされていませんログイン ステータス。そのため、ユーザーがアクセスするたびに、ユーザーのログアウト時刻を 30 分遅らせる必要があります。これには、Redis 更新メソッドを呼び出す必要があります/// <summary> /// 延期 /// </summary> /// <param name="key"></param> /// <param name="expiresTime"></param> public void KSetEntryIn(string key, TimeSpan expiresTime) { Func<IRedisClient, bool> fun = (IRedisClient client) => { client.ExpireEntryIn(key, expiresTime); return false; }; TryRedisWrite(fun); }
カプセル化後
/// <summary> /// 续期 /// </summary> public void Postpone() { new RedisClient<object>().KSetEntryIn(SessionId, new TimeSpan(0, Managers.TimeOut, 0)); }ここでは、MVC3 の ActionFilter を使用してすべてのユーザー リクエストをインターセプトしました
namespace Test { public class SessionFilterAttribute : ActionFilterAttribute { /// <summary> /// 每次请求都续期 /// </summary> /// <param name="filterContext"></param> public override void OnActionExecuting(ActionExecutingContext filterContext) { new Session(filterContext.HttpContext).Postpone(); } } }Global.asax
public static void RegisterGlobalFilters(GlobalFilterCollection filters) { filters.Add(new SessionFilterAttribute()); } protected void Application_Start() { RegisterGlobalFilters(GlobalFilters.Filters); }
5 に登録する必要があります。メソッドを呼び出す
呼び出しを容易にし、4.0 機能の新機能を借用し、Controllerpublic static class ExtSessions {public static Session SessionExt(this Controller controller) { return new Session(controller.HttpContext); } }
呼び出しメソッド
public class HomeController : Controller { public ActionResult Index() { this.SessionExt().IsLogin(); return View(); } }に拡張属性を追加します。
以上がRedis はセッションを置き換えますか?の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。