我正在编写一个 Web 应用程序,它允许用户浏览到网站中的多个网页,发出某些请求。用户输入的所有信息都将存储在我创建的对象中。问题是我需要从网站的任何部分访问这个对象,我真的不知道实现这一点的最佳方法。我知道一种解决方案是使用会话变量,但我不知道如何在 asp .net MVC 中使用它们。我将在哪里声明一个会话变量?还有其他方法吗?
10 回答
我认为您会想考虑事物是否真的属于会话状态。这是我发现自己时不时做的事情,它是一种很好的强类型方法,但是在将事情放在会话上下文中时应该小心。并非所有东西都应该存在,因为它属于某个用户。
在 global.asax 中钩住 OnSessionStart 事件
void OnSessionStart(...)
{
HttpContext.Current.Session.Add("__MySessionObject", new MySessionObject());
}
从代码中 HttpContext.Current 属性 != null 的任何位置,您都可以检索该对象。我使用扩展方法来做到这一点。
public static MySessionObject GetMySessionObject(this HttpContext current)
{
return current != null ? (MySessionObject)current.Session["__MySessionObject"] : null;
}
这样你就可以在代码中
void OnLoad(...)
{
var sessionObj = HttpContext.Current.GetMySessionObject();
// do something with 'sessionObj'
}
这里的答案是正确的,但是我很难在 ASP.NET MVC 3 应用程序中实现它。我想访问控制器中的 Session 对象,但不知道为什么我不断收到“实例未设置为对象错误的实例”。我注意到的是,当我尝试通过执行以下操作访问会话时,在控制器中,我不断收到该错误。这是因为 this.HttpContext 是 Controller 对象的一部分。
this.Session["blah"]
// or
this.HttpContext.Session["blah"]
但是,我想要的是作为 System.Web 命名空间一部分的 HttpContext,因为这是上面的答案建议在 Global.asax.cs 中使用的那个。所以我必须明确地执行以下操作:
System.Web.HttpContext.Current.Session["blah"]
这对我有帮助,不确定我是否做了任何不是 MO 的事情,但我希望它对某人有所帮助!
因为我不喜欢看到关于这个地方的“HTTPContext.Current.Session”,所以我使用单例模式来访问会话变量,它使您可以轻松访问强类型的数据包。
[Serializable]
public sealed class SessionSingleton
{
#region Singleton
private const string SESSION_SINGLETON_NAME = "Singleton_502E69E5-668B-E011-951F-00155DF26207";
private SessionSingleton()
{
}
public static SessionSingleton Current
{
get
{
if ( HttpContext.Current.Session[SESSION_SINGLETON_NAME] == null )
{
HttpContext.Current.Session[SESSION_SINGLETON_NAME] = new SessionSingleton();
}
return HttpContext.Current.Session[SESSION_SINGLETON_NAME] as SessionSingleton;
}
}
#endregion
public string SessionVariable { get; set; }
public string SessionVariable2 { get; set; }
// ...
然后您可以从任何地方访问您的数据:
SessionSingleton.Current.SessionVariable = "Hello, World!";
如果您使用的是 asp.net mvc,这里有一个访问会话的简单方法。
从控制器:
{Controller}.ControllerContext.HttpContext.Session["{name}"]
从一个视图:
<%=Session["{name}"] %>
这绝对不是访问会话变量的最佳方式,但它是一条直接路径。所以谨慎使用它(最好在快速原型设计期间),并在合适的时候使用 Wrapper/Container 和 OnSessionStart。
高温高压
好吧,恕我直言..
- 永远不要在您的视图/母版页中引用会话
- 尽量减少对 Session 的使用。MVC 为此提供了 TempData obj,它基本上是一个 Session,只为一次访问服务器而存在。
关于#1,我有一个强类型的主视图,它具有访问 Session 对象所代表的任何内容的属性......
ViewMasterPage<AdminViewModel>
AdminViewModel
{
SomeImportantObjectThatWasInSession ImportantObject
}
AdminViewModel<TModel> : AdminViewModel where TModel : class
{
TModel Content
}
进而...
ViewPage<AdminViewModel<U>>
虽然我不知道asp.net mvc,但这是我们在一个普通的.net网站中应该做的。它也应该适用于 asp.net mvc。
YourSessionClass obj=Session["key"] as YourSessionClass;
if(obj==null){
obj=new YourSessionClass();
Session["key"]=obj;
}
您可以将其放在一个方法中以便于访问。高温高压
有 3 种方法可以做到这一点。
可以直接访问
HttpContext.Current.Session
你可以模拟
HttpContextBase
为
HttpContextBase
我更喜欢第三种方式。这个链接是很好的参考。
在 BaseController 中获取/设置 HttpContext 会话方法与模拟 HttpContextBase 以创建获取/设置方法
我访问会话的方式是编写一个帮助类,它封装了各种字段名称及其类型。我希望这个例子有帮助:
using System;
using System.Collections.Generic;
using System.Web;
using System.Web.SessionState;
namespace dmkp
{
/// <summary>
/// Encapsulates the session state
/// </summary>
public sealed class LoginInfo
{
private HttpSessionState _session;
public LoginInfo(HttpSessionState session)
{
this._session = session;
}
public string Username
{
get { return (this._session["Username"] ?? string.Empty).ToString(); }
set { this._session["Username"] = value; }
}
public string FullName
{
get { return (this._session["FullName"] ?? string.Empty).ToString(); }
set { this._session["FullName"] = value; }
}
public int ID
{
get { return Convert.ToInt32((this._session["UID"] ?? -1)); }
set { this._session["UID"] = value; }
}
public UserAccess AccessLevel
{
get { return (UserAccess)(this._session["AccessLevel"]); }
set { this._session["AccessLevel"] = value; }
}
}
}
伙计们的回答很好,但我会提醒你不要总是依赖 Session。这样做既快速又容易,当然会起作用,但在所有情况下都不会很好。
例如,如果您遇到托管不允许会话使用的情况,或者您在 Web 场中,或者在共享 SharePoint 应用程序的示例中。
如果您想要一个不同的解决方案,您可以考虑使用IOC 容器,例如Castle Windsor,创建一个提供程序类作为包装器,然后根据您的要求使用每个请求或会话生活方式保留您的类的一个实例。
IOC 将确保每次返回相同的实例。
更复杂的是,如果您需要一个简单的解决方案,只需使用会话。
下面是一些出于兴趣的实现示例。
使用此方法,您可以按照以下方式创建提供程序类:
public class CustomClassProvider : ICustomClassProvider
{
public CustomClassProvider(CustomClass customClass)
{
CustomClass = customClass;
}
public string CustomClass { get; private set; }
}
并将其注册为:
public void Install(IWindsorContainer container, IConfigurationStore store)
{
container.Register(
Component.For<ICustomClassProvider>().UsingFactoryMethod(
() => new CustomClassProvider(new CustomClass())).LifestylePerWebRequest());
}
您可以使用 ViewModelBase 作为所有模型的基类,该类将负责从会话中提取数据
class ViewModelBase
{
public User CurrentUser
{
get { return System.Web.HttpContext.Current.Session["user"] as User };
set
{
System.Web.HttpContext.Current.Session["user"]=value;
}
}
}
你可以在 HttpContextBase 上写一个扩展方法来处理会话数据
T FromSession<T>(this HttpContextBase context ,string key,Action<T> getFromSource=null)
{
if(context.Session[key]!=null)
{
return (T) context.Session[key];
}
else if(getFromSource!=null)
{
var value = getFromSource();
context.Session[key]=value;
return value;
}
else
return null;
}
在控制器中使用如下
User userData = HttpContext.FromSession<User>("userdata",()=> { return user object from service/db });
第二个参数是可选的,当会话中不存在值时,它将用于填充该键的会话数据。