程序员羽化之路--假如需要一百万个对象

点击上方蓝字关注我们

菜菜哥,救命呀

又被产品经理砍了?

这次搞不好真要被砍了,线上一个用户系统内存溢出了,占用内存太高了

用户基数大,内存占用高正常

高的不太正常了,我觉得可能和我的设计有关

那说说你的用户设计

设计背景

每个平台都会有用户这种基础数据的设计,作为最基础的用户,每个用户都有很多属性,比如性别,姓名,手机号等,每个用户还可以有类似经验值这样的荣誉系统,根据不同的经验值来对应不同的等级,不同的等级对应不同的荣誉UI,比如一级用户可能只显示一个星星,二级用户显示两颗星星,以此类推,类似于QQ等级的星星月亮太阳,这样的荣誉系统随着平台的不断壮大,可能会衍生出很多类型。那么问题来了,用户登录的时候就需要初始化用户的这些荣誉值,以星星数为例,类似于以下代码 

public class Star
{//等级public int Level{get ;set ;}//对应的星星数目public int StarNumber{get ;set ;}//对应的星星颜色public int Color{get ;set ;}... 其他属性
}
//用户信息
public class User{   public Star StarInfo{get ;set ;}//...用户的其他属性
}//初始化用户信息
User u=new User(){ StarInfo=new Star(){ Level=1, StarNumber=1,Color=1}};

每一个登录用户都会初始化一个Star属性来表示当前用户的Star信息,当有100万用户甚至更多用户同时在线的时候,内存中就实例化了同样数量的Star对象,以及其他类似的属性对象。这么多重复的对象难道不能优化吗?当然不是!!

问题分析

一个业务出现问题,首先要分析问题的所在。根据以上所说,问题的根本在于产生了大量的对象,首先每个用户对象都有自己独特的状态,这个基本上不可能分解优化,但是类似Star这样的属性就有优化途径了,这些荣誉属性一个最大的共同点就是不可变,换句话说,等级1的用户对应的Star信息是永远不会变的,永远是level=1,starnumber=1,color=1 等。基于这个不变性,我们可以把这个Star抽离出来,供所有等级1的用户使用,假设原来有10万等级1的用户,原来需要10万个对象,现在只需要一个对象,这可是天壤之别。

解决问题

基于以上问题分析,我们需要做的是把对象重复使用,只要是对象重复问题,基本上可以利用一个对象出口来解决问题,类似于以下的对象初始化工厂,但是要注意线程安全问题,因为同时请求并初始化对象的线程会有多个。

public class UserStarFac{static object objLock = new object();static Dictionary<int, Star> UserStarMap = new Dictionary<int, Star>();public static Star GetUserStar(int level){//利用锁来防止实例化多次,当然这里可以优化lock (objLock){Star info = null; ;if(!UserStarMap.TryGetValue(level, out info)){info = new Star() { Color = 1, Level = 1, StarNumber = 1 };UserStarMap.Add(level,info);}return info;}}}

编写简单测试程序

static void Main(string[] args){int i = 0;List<User> userList = new List<User>();while (i < 100000){// userList.Add(new User() {  StarInfo=new Star() {  Color=1, Level=1, StarNumber=1} });userList.Add(new User() {  StarInfo= UserStarFac .GetUserStar(1)});i++; }Console.WriteLine("初始化完成");Console.Read();}

内存的测试结果:

不执行任何程序:占用内存:2.8 M  

无优化初始化10万对象:占用内存:11 M

优化之后初始化10万对象:占用内存:7 M

居然一个小小的优化就减少了4M内存,不要小看这小小的4M,你要看的是比例,居然减少了将近 50%,真实业务中,可以进行这种优化的地方数不胜数,不知道你是否在乎呢?

这种大量重复对象的问题尤其是在游戏编程中经常存在,比如五子棋游戏,棋子的初始化,一个游戏大厅存在成千上百万对局,如果每个局中的棋子都初始化一个对象,那内存使用是相当可怕的,这种需要把通用的对象属性,不变的对象属性抽离出来,做共享是有必要的。

据说这种优化有一个学名:享元模式,没有必要记住名字,但需要记住原理和场景,必须要提一句:注意不变的对象才可以哦

●程序员过关斩将-- 喷一喷坑爹的面向UI编程

●程序员过关斩将--redis做消息队列,香吗?

●程序员修神之路--有了容器为什么kubernetes还需要Pod?

●程序员修神之路--为什么我会了SOA,你们还要逼我学微服务?

●程序员过关斩将--要想获取我的用户信息,就得按照规矩来

●程序员过关斩将--更加优雅的Token认证方式JWT

●程序员过关斩将--cookie和session的关系其实很简单

●程序员修神之路--用NOSql给高并发系统加速

●程序员修神之路--高并发系统设计负载均衡架构

●程序员过关斩将--你为什么还在用存储过程?

●程序员修神之路--问世间异步为何物?

●程序员修神之路--提高网站的吞吐

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/310927.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

iis php打开空白页,windows+IIS+php 访问显示空白页 php版本信息访问正常

10:48:46 A session had already been started - ignoring session_start() in E:\zentaopms\module\common\model.php on line 24 when visiting10:48:46 ERROR: 鎮ㄨ闂殑鍩熷悕 172.25.8.82:8080 娌℃湁瀵瑰簲鐨勫叕鍙搞€&#xfffd; in E:\zentaopms\module\common\mo…

《C++ Primer》8.2.1节练习(部分)

练习8.4: #include <iostream> #include <fstream> #include <string> #include <vector> using namespace std;int main() {//打开文件ifstream in("D:\\CCCCCCCCCCCCCCCCCCC\\yyhaif.txt");if (!in) {cerr << "无法打开输入文…

WTM 3.5发布,VUE来了!

千呼万唤中&#xff0c;WTM的Vue前后端分离版本终于和大家见面了&#xff0c;我曾经跟群里1000多位用户保证过Vue版本会在春天到来&#xff0c;吹过的牛逼总算是圆上了。卧槽&#xff0c;NB啊!我等到花都谢了风太大&#xff0c;吹瞎了朕的双眼我是谁&#xff0c;我在哪儿&#…

《C++ Primer》8.3.1节练习

练习8.9: #include <iostream> #include <sstream> #include <string> #include <stdexcept>using namespace std;istream &f(istream &in) {string v;while (in >> v, !in.eof())//直到遇到文件结束符才停止读取{if (in.bad())throw r…

java 第三方序列化,11.既然有第三方的序列化方式,说明java官方提供的序列化方式应该有一些很明显或者很致命的缺点……...

序列化是什么&#xff1a;把一个java对象转化为二进制对象&#xff0c;并保存到硬盘&#xff0c;或在网络上传输。反序列化就是把序列化的二进制对象读到内存中。 作用&#xff1a;1、减少内存占用或网络传输。比如web容器中的session&#xff0c;当session数量过大比如10W连接…

2020年,我来盘点下微服务架构技术栈

2020年了&#xff0c;很多小伙伴儿对微服务还比较陌生&#xff0c;说起来很多人可能不敢相信&#xff0c;其实微服务这个概念早在2012年就提出来了&#xff0c;经过了这些年的发展&#xff0c;现在已经成为企业非常主流的架构选项了。今天&#xff0c;我就来带大家一起探讨下微…

2.5w字长文爆肝 C++动态内存与智能指针一篇搞懂!太顶了!!!

动态内存与智能指针1.动态内存与智能指针2.shared_ptr类2.1.make_shared函数2.2.shared_ptr的拷贝和赋值2.3.shared_ptr自动销毁所管理的对象2.4. shared_ptr会自动释放相关联的内存2.5.使用了动态生存期的资源的类2.6.定义StrBlob类2.7. StrBlob构造函数2.8.元素访问成员函数2…

ASP.NET Core应用的7种依赖注入方式

ASP.NET Core框架中的很多核心对象都是通过依赖注入方式提供的&#xff0c;如用来对应用进行初始化的Startup对象、中间件对象&#xff0c;以及ASP.NET Core MVC应用中的Controller对象和View对象等&#xff0c;所以我们可以在定义它们的时候采用注入的形式来消费已经注册的服务…

javascript php 传值,js 传值到 PHP 有关问题

js 传值到 PHP 问题说一下我的需求&#xff1a;通过点击button的时候&#xff0c;去通过php查找指定日期的数据库当中的数据&#xff0c;function load_point(date) {// Change this depending on the name of your PHP filedownloadUrl("phpsqlajax_genxml2.php", f…

ASP.NET Core 3.x - 为什么采用新的 Endpoint Routing 路由系统

Endpoint Routing 路由系统ASP.NET Core 3.x 使用了一套叫做 Endpoint Routing 的路由系统。这套路由系统在ASP.NET Core 2.2 的时候就开始露面了。这套Endpoint Routing路由系统提供了更强大的功能和灵活性&#xff0c;以便能更好的处理请求。早期ASP.NET Core的路由系统我们先…

php 接口缓存,php写的一个缓存接口demo,兼容redis和memcache

/*** 工厂方法模式* -------------* author zhangqian* version v1.0*///缓存接口interface cache {public function init($conf);public function setVal($key , $val);public function getVal($key);public function delVal($key);public function autoIncreament($key);}//m…

《C++ Primer》10.1节练习

练习10.1: #include <iostream> #include <vector> #include <algorithm> using namespace std;int main() {vector<int>vi;int val;vi.push_back(45);vi.push_back(45);vi.push_back(45);vi.push_back(45);for (int i 1; i < 45; i) {vi.push_ba…

ueditor编辑器php上传配置,php版本UEditor编辑器图片上传设置

"savePath" > "../../upload/" , //保存路径注意&#xff0c;这里远程抓取的图片保存在了upload根目录下&#xff0c;要想保存在upload的子目录下&#xff0c;例如每个月份的文件夹下&#xff0c;则需要找到以下代码$savePath $config[ sav…

Asp.Net Core AuthorizeAttribute 和AuthorizeFilter 跟进及源码解读

一、前言IdentityServer4已经分享了一些应用实战的文章&#xff0c;从架构到授权中心的落地应用&#xff0c;也伴随着对IdentityServer4掌握了一些使用规则&#xff0c;但是很多原理性东西还是一知半解&#xff0c;故我这里持续性来带大家一起来解读它的相关源代码&#xff0c;…

日期计算算法训练

知识点: 闰年共有366天&#xff08;1月~12月分别为31天、29天、31天、30天、31天、30天、31天、31天、30天、31天、30天、31天&#xff09;。 题目: 蓝桥杯2015初赛-星系炸弹[蓝桥杯2018初赛]第几天-日期计算(水题)[蓝桥杯2018初赛]星期一-日期计算

1张手稿图讲明白 Kubernetes 是怎么运行的

注意&#xff1a;如果您已经知道Kubernetes的工作原理&#xff0c;那么您可能会对我之前的博文感兴趣&#xff0c;请停止使用具有管理员权限的kubeconfigKubernetes是最初由Google设计的开源工具&#xff0c;现在由 Cloud Native Computing Foundation&#xff08;CNCF&#xf…

java继承层次结构,在状态模式中实现继承层次结构 - java

我有一个与此非常相似的设计&#xff1a;这里的NewOrder&#xff0c;Registered&#xff06;Granted都有通用方法AddOrderline()和Cancel()&#xff0c;因此将这两种方法重构为父类很容易。当我要Cancel一条Shipped行(当前未在图中显示)时&#xff0c;会出现问题。由于发运行不…

枚举算法训练

知识点: 素数-试除法和埃式筛选法模板不要让数组越界&#xff0c;一越界&#xff0c;答案总容易错。 题目: 蓝桥杯2015初赛-方程整数解-枚举 蓝桥杯2015初赛-奇妙的数字 蓝桥杯2015初赛-饮料换购-枚举 蓝桥杯2015初赛-奖券数目-枚举 蓝桥杯2015初赛-三羊献瑞-枚举 蓝桥杯…

就喜欢用vSphere部署K8s集群,不全是因为自动化!

通过努力&#xff0c;我们有了一个完整配置的工作负载域&#xff0c;其中包括一个NSX-T Edge部署。现在&#xff0c;我们准备继续使用Kubernetes 部署vSphere。通过VMware Cloud Foundation 4.0中的SDDC Manager&#xff0c;我们确保NSX-T Edge可用&#xff0c;并且还确保Workl…

模拟算法训练

知识点: C stringstream输入方式C关于getline()和getchar()的小点Csubstr()用法toupper小写变大写 tolower大写变小写有些题目记得开longlong 6.蛇型方向向量写法: int dx[] {0,1,0,-1},dy[] {1,0,-1,0}; int d 0; d (d1)%4;题目: 蓝桥杯2016初赛-有奖猜谜-模拟 蓝桥杯…