对象池服务可以减少从头创建每个对象的系统开销。在激活对象时,它从池中提取。在停用对象时,它放回池中,等待下一个请求。
我们来看下主线程中,如何与对象池打交道:

static void Main(string[] args)
{
InstancePoolResolver.Register<OrderQueryServiceInterface, OrderQueryService>(); while (true)
{
Thread.Sleep();
Console.Clear(); for (int i = ; i < ;i++ )
{
ThreadPool.QueueUserWorkItem(new WaitCallback(ConsumeObject));
}
}
} private static void ConsumeObject(object state)
{
OrderQueryServiceInterface srv = null;
try
{
using (srv = InstancePoolResolver.Resolve<OrderQueryServiceInterface>()) //从对象池中取得对象,没有可用对象则throw exception
{
Console.WriteLine("Object ID--->" + srv.GetHashCode());
Thread.Sleep(); //故意长时间占用对象
}
}
catch (Exception ex)
{
Console.WriteLine(ex.Message);
}
finally
{
if (srv != null)
srv.Dispose();
}
}

运行效果:

最多只有2个instance,那在哪里设置这个数量呢?请看

是通过Attribute打tag上去的

下面来看看最核心的InstancePoolResolver类

public sealed class InstancePoolResolver
{
private static Dictionary<Type, Type> typeMappers = new Dictionary<Type, Type>();
private static Dictionary<Type, int> typeMappersMaxInstanceCount = new Dictionary<Type, int>();
private static Dictionary<Type, List<PoolableObject>> typeInstances = new Dictionary<Type, List<PoolableObject>>(); private static object o4lock = new object();
public static void Register<T, TProvider>()
where TProvider : class, new()
{
if (typeMappers.ContainsKey(typeof(T)))
throw new Exception("Key existed"); lock (o4lock)
{
Type t = typeof(T);
typeMappers.Add(t, typeof(TProvider));
typeInstances.Add(t, new List<PoolableObject>()); InstanceSettingAttribute setting = GetInstanceSettingAttribute(typeof(TProvider));
typeMappersMaxInstanceCount.Add(t, setting.MaxInstanceGlobal);
}
} public static T Resolve<T>()
where T: PoolableObject
{
Type t = typeof(T);
if (!typeMappers.ContainsKey(t) || !typeInstances.ContainsKey(t))
throw new Exception("Key empty, register please"); lock (o4lock)
{
List<PoolableObject> instances = typeInstances[t];
if (instances == null)
{
instances = new List<PoolableObject>();
typeInstances[t] = instances;
}
foreach (PoolableObject o in instances)//是否已经存在已有闲置对象
{
if (o.IsInPool)
{
o.IsInPool = false;
return (T)o;
}
}
if (instances.Count < typeMappersMaxInstanceCount[t])//new新对象到对象池中
{
Type type = typeMappers[t];
PoolableObject obj = (PoolableObject)Activator.CreateInstance(type);
instances.Add(obj);
obj.IsInPool = false;
return (T)obj;
}
}
throw new Exception("Object Pool fulled!"); //没有多余的资源
} private static InstanceSettingAttribute GetInstanceSettingAttribute(Type type)
{
object[] attrs = type.GetCustomAttributes(typeof(InstanceSettingAttribute), false);
if (attrs == null || attrs.Count() == )
return new InstanceSettingAttribute() { MaxInstanceGlobal=}; return (InstanceSettingAttribute)attrs[];
}
}

其实很简单,只是多了个获取Attribute的函数

C# 对象池的实现(能限制最大实例数量,类似于WCF的MaxInstanceCount功能)的更多相关文章

  1. 深度剖析C++对象池自动回收技术实现

    http://www.tuicool.com/articles/mQBfQfN 对象池可以显著提高性能,如果一个对象的创建非常耗时或非常昂贵,频繁去创建的话会非常低效.对象池通过对象复用的方式来避免重 ...

  2. Apache common pool2 对象池

    对象池的容器:包含一个指定数量的对象.从池中取出一个对象时,它就不存在池中,直到它被放回.在池中的对象有生命周期:创建,验证,销毁,对象池有助于更好地管理可用资源,防止JVM内部大量临时小对象,频繁触 ...

  3. Java之对象池

    单例模式是限制了一个类只能有一个实例,对象池模式则是限制一个类实例的个数.对象池类就像是一个对象管理员,它以Static列表(也就是装对象的池子)的形式存存储某个实例数受限的类的实例,每一个实例还要加 ...

  4. Java对象池

    单例模式是限制了一个类只能有一个实例,对象池模式则是限制一个类实例的个数.对象池类就像是一个对象管理员,它以Static列表(也就是装对象的池子)的形式存存储某个实例数受限的类的实例,每一个实例还要加 ...

  5. Java对象池示例

    单例模式是限制了一个类只能有一个实例,对象池模式则是限制一个类实例的个数.对象池类就像是一个对象管理员,它以Static列表(也就是装对象的池子)的形式存存储某个实例数受限的类的实例,每一个实例还要加 ...

  6. 对象池在 .NET (Core)中的应用[1]: 编程体验

    借助于有效的自动化垃圾回收机制,.NET让开发人员不在关心对象的生命周期,但实际上很多性能问题都来源于GC.并不说.NET的GC有什么问题,而是对象生命周期的跟踪和管理本身是需要成本的,不论交给应用还 ...

  7. 对象池在 .NET (Core)中的应用[2]: 设计篇

    <编程篇>已经涉及到了对象池模型的大部分核心接口和类型.对象池模型其实是很简单的,不过其中有一些为了提升性能而刻意为之的实现细节倒是值得我们关注.总的来说,对象池模型由三个核心对象构成,它 ...

  8. 设计模式之美:Object Pool(对象池)

    索引 意图 结构 参与者 适用性 效果 相关模式 实现 实现方式(一):实现 DatabaseConnectionPool 类. 实现方式(二):使用对象构造方法和预分配方式实现 ObjectPool ...

  9. Egret中的对象池ObjectPool

    为了可以让对象复用,防止大量重复创建对象,导致资源浪费,使用对象池来管理. 对象池具体含义作用,自行百度. 一 对象池A 二 对象池B 三 字符串key和对象key的效率 一 对象池A /** * 对 ...

随机推荐

  1. Testlink1.9.17使用方法(第十三章 使用中遇到的问题)

    第十三章 使用中遇到的问题 一. 登录Testlink后,新建一个项目后,会出现如下提示: 解决办法:打开Testlink安装文件夹下的config.inc.php文件, 原来:$tlCfg-> ...

  2. (后端)SQL SERVER 字符串按数字排序

    应用于B1-1,B1-2,B10-1,B11-1 sqlserver肯定不能按照字符串进行排序,需要进行处理一番: select CONVERT(varchar, LEFT(code,1)),conv ...

  3. [katalon] 页面切换

    UI自动化测试过程中会涉及到需要切换多个页面, 如点击一个按钮之后跳转到新的页面, 后者A站点提交信息后,B站点审核. Katalon虽然不支持控制多个浏览器,但是支持处理tab切换. 核心方法是使用 ...

  4. windows 获取用户的Sid的方法

    正常获取: whoami /user 如果要获取其他用户的SID就显得力不从心了,我们可以使用微软提供的系统工具 Sysinternals Suite 下载地址:https://docs.micros ...

  5. RHEL 5.7 使用rpm安装XtraBackup问题总结

    在Red Hat Enterprise Linux Server release 5.7 (Tikanga)上使用RPM方式安装Percona Xtrabackup 2.4.6时遇到了一些问题,特意总 ...

  6. mssql sqlserver 禁止删除数据表中指定行数据(转自:http://www.maomao365.com/?p=5323)

    转自:http://www.maomao365.com/?p=5323 摘要:下文主要讲述,如何禁止删除数据表中指定行数据 最近收到用户一个需求,禁止所有人删除”表A”中,ID 为1.2.3.4.5的 ...

  7. SQL Server 2012 手动安装帮助文档+排错

    逆天SQL Server 2012装的不要不要的,最后发现...竟然没帮助文档...汗啊!原来它跟vs一样要自己装帮助文档...好吧,官网一下载,妹的...报错...然后就让我们还原这个安装过程以及逆 ...

  8. Flask中的CBV

    Flask中的CBV 第一种 class Index(views.MethodView): methods = ['GET', 'POST'] decorators = [] def get(self ...

  9. 【Linux基础】文件处理实例

    1.文件拆分 //每4000行拆分一个文件 epms_t_ep_fx_stl_xy_20190129.dat 2.行处理 //查找第二列为711611且第三列为711100记录,打印行号和整行数据 a ...

  10. IE9样式错乱,IE11无法正常加载v-loading等问题 引入了babel-polyfill插件,依然出现”polyfill-eventsource added missing EventSource to window”的奇怪问题(ie所有版本都有出现)

    第一步:安装babel-ployfill (已安装请跳过此步骤) yarn add babel-ployfill 修改webpack打包配置文件:webpack.bash.conf.js // 引入b ...