我手上有一个Windows表单应用程序,其中明确分离了GUI和业务逻辑。解决方案很大(40个项目,数千个类,大约2000个数据库表)。我的任务是找到一个关于如何重用业务逻辑以构建Web应用程序作为前端的解决方案。
我认为主要问题是我们的静态Session类,它在解决方案中的每个其他类都使用:
public static class Session
{
public static string CurrentUser { get; set; }
public static string CurrentDatabase { get; set; }
public static string CurrentCompanyProfile { get; set; }
public static string CurrentLanguage { get; set; }
}
这个类在ASP.NET中不起作用,所有用户都会共享它,因为它被大量使用所以无法替换它。
我在想我需要从这个类中删除所有信息并保留接口。通过调用属性代码的get方法,可以某种方式将此调用重定向到HttpContext.Current.Session,在那里我将存储一个包含所有这些信息的类。
我不确定如何正确地做到这一点,因为Session位于一个程序集中,该程序集不会引用System.Web并且考虑到我不能破坏我们的WinForms实现。
感谢。
答案 0 :(得分:2)
假设您可以更新Session类,您可以在Session类中抽象属性的存储。在当前的应用程序中,您可以创建一个简单的内存存储,在Web应用程序中,您可以创建一个Web会话存储。
首先定义一个session-value-provider,在当前的静态Session类中使用它。
public interface ISessionValueProvider {
string CurrentUser { get; set; }
string CurrentDatabase { get; set; }
string CurrentCompanyProfile { get; set; }
string CurrentLanguage { get; set; }
}
public static class Session {
private static ISessionValueProvider _sessionValueProvider;
public static void SetSessionValueProvider(ISessionValueProvider provider) {
_sessionValueProvider = provider;
}
public static string CurrentUser {
get { return _sessionValueProvider.CurrentUser; }
set { _sessionValueProvider.CurrentUser = value; }
}
// Etc for the other props
}
与当前应用程序相比,定义一个sesion值提供程序,它使用内存来存储值。
public class MemorySessionValueProvider: ISessionValueProvider {
public string CurrentUser {get; set; }
// Etc for the other props
}
要使用它,请创建一个实例并将其提供给静态会话类。例如,您可以将其添加到Main方法中。
Session.SetSessionValueProvider(new MemorySessionValueProvider());
现在,对于您的Web应用程序,创建一个使用Web会话的会话值提供程序。
public class WebSessionValueProvider: ISessionValueProvider {
private const string CURRENTUSERKEY = "CurrentUser"; // TODO: Change this if necessary
public string CurrentUser {
get { return (string)HttpContext.Current.Session[CURRENTUSERKEY]; }
set { HttpContext.Current.Session[CURRENTUSERKEY] = value; }
}
// Etc for the other props
}
再次,将此web-session-value-provider的实例提供给静态会话类。例如,在global.asax。
中Session.SetSessionValueProvider(new WebSessionValueProvider);
答案 1 :(得分:1)
我会使用IoC容器(例如Castle Windsor)。声明一个接口,例如IStateManager
:
public interface IStateManager
{
T GetItem<T>(string key);
SetItem<T>(string key, T value);
}
然后,Session
类的实现将发生如下变化:
public static class Session
{
private static IStateManager _manager;
private static IStateManager Manager
{
get
{
if (_manager == null)
{
IStateManager m = null; // Get instance using IoC container
Interlocked.CompareExchange(ref _manager, m, null);
}
return _manager;
}
}
public static string CurrentUser
{
get { return Manager.GetItem<string>("CurrentUser"); }
set { Manager.SetItem<string>("CurrentUser", value); }
}
// The rest is similar
}
对于WinForms,实现将只在静态实例上运行,对于ASP.NET,您有一个不同的实现来维护会话中的数据。好处是你将为不同的环境提供不同的安装程序,而你的代码根本不需要了解它们(因此,逻辑不依赖于WinForms或ASP。 NET)。
此外,这种方法是可测试的(例如,使用Moq模拟IStateManager
非常容易。