内容简介:(本文所使用的Python库和版本号: Python 3.5, Numpy 1.14, scikit-learn 0.19, matplotlib 2.2 )我们经常看到办公大楼中人来人往,进进出出,在平时没有什么活动的时候,进出大楼的人数会非常少,而一旦举办有大型商业活动,则人山人海,熙熙攘攘,所以很明显,大楼进出的人数和大楼是否举办活动有很明显的关联,那么,是否可以构建一个模型,通过大楼进出人数来预测该大楼是否在举办某种活动了?答案是肯定的,且听炼丹老顽童娓娓道来。
(本文所使用的 Python 库和版本号: Python 3.5, Numpy 1.14, scikit-learn 0.19, matplotlib 2.2 )
我们经常看到办公大楼中人来人往,进进出出,在平时没有什么活动的时候,进出大楼的人数会非常少,而一旦举办有大型商业活动,则人山人海,熙熙攘攘,所以很明显,大楼进出的人数和大楼是否举办活动有很明显的关联,那么,是否可以构建一个模型,通过大楼进出人数来预测该大楼是否在举办某种活动了?
答案是肯定的,且听炼丹老顽童娓娓道来。
1. 准备数据集
本项目案例所使用到的原始数据集来源于:UCI大学数据集,该数据集是公开的,读者可以自行下载该数据集到自己的本地电脑上。
1.1 了解数据集
从该数据集的官方网站上,我们可以看到该数据集的基本介绍:
可以看出,该数据集一共有10080个样本,没有缺失数据,一共有四个基本属性,集四个features,可以用于分类模型和时序模型,此处我们只是用来进行“是否举办活动”的预测,很明显,是一个多分类问题,具体而言是一个二分类问题。
在我们下载该数据后,通过查看比对,发现该有效数据主要存放在两个文件中(CalIt2.data和CalIt2.events),其中CalIt2.data中存储了10080条数据记录,每一条数据包含有四列,每一列的信息说明如下表所示。而CalIt2.events包括有30条数据记录,也包括有四列,其中的第一列是日期,第二列是活动开始时间,第三列是活动结束时间,第四列表示有活动。下是我总结的本项目案例数据集的基本信息。
其中表格中前面四行表示数据集的四个features,这些信息位于CalIt2.data中,最后一行表示数据集的Label,位于CalIt2.events中。
1.2 数据规整
由于本项目的数据集位于两个不同的文件中,同时两个文件的样本格式也不是一一对应,故而在构建模型之前,需要我们对数据进行规整,组成我们所需要的数据类型。
此处所用到的数据规整至少包括有三个方面:将同一个时间段内进出大楼的人数统一到一行,将CalIt2.events中的标记数据加载到CalIt2.data中组成完整的数据集,将数据集中的日期转换为星期数,作为一个特征向量。
1.2.1 进出大楼人员统一到一行
由于原始数据集CalIt2.data中,进大楼的位于一行,最前面的代号为9,出大楼的位于另外一行,最前面的代号为7,故而我们需要将这两行数据统一到一行,下面是实现代码,所采用的核心思想是:用DataFrame的布尔索引来获取进出大楼的单独一个DataFrame,然后将这两个DF整合到一个DataFrame中。
# 将feature_set中进楼和出楼的人员统计到一行。 # 目前数据集中出楼人员(代号7)位于偶数行,进楼人员(代号9)位于奇数行。 # 可以使用for in 方式依次取出各行的人员数,但此处我更愿意使用布尔型索引 code_7=feature_set[0]==7 code_7_data=feature_set[code_7].iloc[:,3].reset_index(drop=True) code_9=feature_set[0]==9 code_9_data=feature_set[code_9].reset_index(drop=True) # print(code_9_data)# OK feature_set2=code_9_data feature_set2[4]=code_7_data feature_set2.drop([0],axis=1,inplace=True) # 删除第0列 # print(feature_set2) # col3 表示in,col4表示out,打印没有问题 # feature_set2.to_csv('d:/feature_set2.csv') # 保存以便查看是否有误 复制代码
打印结果可以参考我的原始代码( 我的github ),此处由于打印出来后结果太长,我没有贴上来。
1.2.2 将label添加到Feature_set中组成完整的数据集
由于原始数据集的features特征向量放置在CalIt2.data文件中,且Label向量放置在CalIt2.events中,故而有必要将这两部分整合到一起。但是,整合过程并不是简单的将两个DataFrame连接起来,而要考虑时间范围。在CalIt2.events文件中列举了有活动的日期和起止时间,故而我们需要对Features中的日期和时间拿出来和CalIt2.events进行逐一比对,如果日期和时间落在events文件中,则表示这个时间段有活动,需要作出特殊标记。有很多种方法可以是实现这种逐一比对,下面我还是采用DataFrame的切片和索引来完成,我认为,这种方式算是速度比较快的一种方式。
# 下面是如何将feature_set2和label_set整合到一个DataFrame中来 # 要判断时间,如果feature_set2中的日期和时间都落在了label_set对应的时间内, # 则表示有event发生,用1表示,如果没有,用0表示。 # 比较日期时间的方法有很多,此处我采用比较简单的方法 feature_set2[5]=0 # 表示是否有event的列都初始化为0 def calc_mins(time_str): nums=time_str.split(':') return 60*int(nums[0])+int(nums[1]) # 将时间转换为分钟数,此处不用考虑秒 for row_id,date in enumerate(label_set[0]): # 先取出label中的日期 temp_df=feature_set2[feature_set2[1]==date] if temp_df is None: continue start_min=calc_mins(label_set.iloc[row_id,1]) stop_min=calc_mins(label_set.iloc[row_id,2]) for row in temp_df[2]: # 在逐一判断时间是否位于label中时间之间 feature_min=calc_mins(row) if feature_min>=start_min and feature_min<=stop_min: feature_row=temp_df[temp_df[2]==row].index.tolist() feature_set2.loc[feature_row,5]=1 # feature_set2.to_csv('d:/feature_set2_withLabel.csv') # 保存后打印查看没有问题 复制代码
1.2.3 将日期转换为星期数
这个相对于前面两个数据规整方面,要简单得多,直接贴代码。
# 进一步处理,由于日期在以后的日子里不可重复,作为feature并不合适,而可以用星期数来代替, feature_set2[0]=pd.to_datetime(feature_set2[1]) # print(feature_set2.tail()) feature_set2[0]=feature_set2[0].map(lambda x: x.strftime('%w')) # 将日期转换为星期数 feature_set2=feature_set2.reindex(columns=range(6)) print(feature_set2.tail()) # 查看转换没有问题 feature_set2.to_csv('E:\PyProjects\DataSet\BuildingInOut/Sorted_Set.txt') # 将整理好的数据集保存,下次可以直接读取 复制代码
-----------------------输---------出--------------------------------
0 1 2 3 4 5 5035 6 11/05/05 21:30:00 0 0 0 5036 6 11/05/05 22:00:00 0 3 0 5037 6 11/05/05 22:30:00 0 0 0 5038 6 11/05/05 23:00:00 0 0 0 5039 6 11/05/05 23:30:00 0 1 0
-------------------------完-------------------------------------
处理完成之后的feature_set2可以直接保存一下,这样下次可以直接调用这个数据规整之后的文件,读取里面的数据集来训练和测试即可。
1.3 数据编码
很明显,从上面的数据集中可以看到第1列是日期,不适合做特征向量,需要删除,而第2列是字符串类型,没法用于机器学习,故而我们需要对第2列进行编码,如下代码:
# 由于第1列只是包含日期,作为特征向量并不合适,故而需要删除 feature_set2.drop([1],axis=1,inplace=True) # 而第2列明显是字符串类型,里面的内容对机器学习而言如同天书,故需要编码 from sklearn import preprocessing time_encoder=preprocessing.LabelEncoder() feature_set2[2]=time_encoder.fit_transform(feature_set2[2]) print(feature_set2.tail()) 复制代码
------------------输---------出--------------------------------
0 2 3 4 5 5035 6 43 0 0 0 5036 6 44 0 3 0 5037 6 45 0 0 0 5038 6 46 0 0 0 5039 6 47 0 1 0
---------------------完-------------------------------------
所以,可以看到,第2列已经变成了数字编码,这部分内容在我以前的文章中用到过很多次。如 【火炉炼AI】机器学习013-用朴素贝叶斯分类器估算个人收入阶层
########################小**********结#################
1. 在网络上,我找了好久都没有找到如何处理这个数据集,好多都是直接给出处理后的结果,所以此处我就自己对这个数据集进行了规整,并将规整的代码放出来。
2. 数据集的规整和预处理往往会花掉机器学习的大部分时间,此处因为已经有了现成数据,仅仅是对数据进行规整,故而耗时相对较少。
3. Pandas和Numpy基本上是数据预处理,数据清洗,数据规整的有力神器,一定要熟练掌握。
#####################################################
2. 使用SVM构建分类器
SVM分类器的构建和其他项目案例中并没有太大差别,可以直接参考我的其他文章: 【火炉炼AI】机器学习014-用SVM构建非线性分类模型 .下面直接贴出代码。
# 下面是使用SVM构建分类器 from sklearn.svm import SVC classifier=SVC(kernel='rbf',probability=True,class_weight='balanced') classifier.fit(train_X,train_y) 复制代码
-------------------输---------出--------------------------------
SVC(C=1.0, cache_size=200, class_weight='balanced', coef0=0.0, decision_function_shape='ovr', degree=3, gamma='auto', kernel='rbf', max_iter=-1, probability=True, random_state=None, shrinking=True, tol=0.001, verbose=False)
-------------------------完-------------------------------------
# 用交叉验证来检验模型的准确性,只是在test set上验证准确性 from sklearn.cross_validation import cross_val_score num_validations=5 accuracy=cross_val_score(classifier,test_X,test_y, scoring='accuracy',cv=num_validations) print('准确率:{:.2f}%'.format(accuracy.mean()*100)) precision=cross_val_score(classifier,test_X,test_y, scoring='precision_weighted',cv=num_validations) print('精确度:{:.2f}%'.format(precision.mean()*100)) recall=cross_val_score(classifier,test_X,test_y, scoring='recall_weighted',cv=num_validations) print('召回率:{:.2f}%'.format(recall.mean()*100)) f1=cross_val_score(classifier,test_X,test_y, scoring='f1_weighted',cv=num_validations) print('F1 值:{:.2f}%'.format(f1.mean()*100)) 复制代码
-----------------------输---------出--------------------------------
准确率:93.78%
精确度:92.96%
召回率:93.78%
F1 值:92.96%
----------------------------完-------------------------------------
从交叉验证的结果可以看出,在测试集上,该SVM分类器仍然可以得到各项指标都达到90%以上的效果,可以认为,该SVM分类器的效果比较好。当然,如果还想继续提升该分类效果,可以使用GridSearch方法来搜索最佳参数组合,GridSearch的使用方法可以参考我的文章 【火炉炼AI】机器学习017-使用GridSearch搜索最佳参数组合
3. 使用该分类器对新样本进行预测
首先我们需要新样本数据,此处我自己构建了一些新样本,并不一定准确,但是可以用于预测。对新样本的预测也很简单,直接使用classifier.predict()函数即可。
# 看起来该模型的预测效果很不错 # 那么用它来对新样本数据进行预测,会是什么样了? new_samples=np.array([[2,'09:30:00',20,12], # 即某个星期二,上午9点-9点半时间段,进大楼20人,出大楼12人 [2,'11:30:00',26,9], [6,'12:30:00',4,22], [0,'05:00:00',1,0]]) transformed=time_encoder.transform(new_samples[:,1]) # print(transformed) # 检查OK new_samples[:,1]=transformed print(new_samples) # 使用classifier进行预测 output_class = classifier.predict(new_samples) print('has events? {}'.format(output_class)) 复制代码
---------------------输---------出--------------------------------
[['2' '19' '20' '12']
['2' '23' '26' '9']
['6' '25' '4' '22']
['0' '10' '1' '0']]
has events? [0 1 0 0]
----------------------------完-------------------------------------
从输出结果可以看出,只有第二个样本有活动,其他样本都没有活动,可以想象的出来,第一个样本9点-9点半有很多人进楼,但是少量人出楼,会不会是正常的上班族?而第二个样本11点到11点半之间有很多人进来,却很少有了出去,这个时间段应该就是会议期间正常的人员流动,而第三个样本12点到12点半,有很多人出楼,这大概是人家出去吃午饭吧。而第四个样本,周日的凌晨五点,估计只有鬼才能在大楼门口晃悠了。。。。
########################小**********结#################
1. 这个项目的难点不在于SVM模型的构建和分类训练上,而在于前期数据集的规整上。
2. 后面使用SVM进行模型构建和训练,对新样本进行预测,在我前面的文章中讲了很多次了,至此已经没有什么新意了。
#####################################################
注:本部分代码已经全部上传到( 我的github )上,欢迎下载。
参考资料:
1, Python机器学习经典实例,Prateek Joshi著,陶俊杰,陈小莉译
以上所述就是小编给大家介绍的《【火炉炼AI】机器学习018-项目案例:根据大楼进出人数预测是否举办活动》,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对 码农网 的支持!
猜你喜欢:- 政府炸毁黑客团伙的大楼,黑客反击“一颗导弹警告”
- 如何攻击一幢大楼:智能建筑成为黑客新“游乐场”
- 美国开展网络安全检查,模拟黑客成功控制五角大楼武器系统
- 五角大楼证实有设立黑名单,拒绝中国和俄罗斯软件
- 硅谷巨头:五角大楼全知全能监控系统的创造者
- 谷歌CEO暗访五角大楼:不做战争生意,但还想和军方做好朋友?
本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们。
Spring技术内幕
计文柯 / 机械工业出版社 / 2010-1-1 / 55.00元
内容简介: 本书是Spring领域的问鼎之作,由业界拥有10余年开发经验的资深Java专家亲自执笔!Java开发者社区和Spring开发者社区一致强烈推荐。 国内第一本基于Spring3.0的著作,从源代码的角度对Spring的内核和各个主要功能模块的架构、设计和实现原理进行了深入剖析。你不仅能从木书中参透Spring框架的优秀架构和设计思想,而且还能从Spring优雅的实现源码中一窥......一起来看看 《Spring技术内幕》 这本书的介绍吧!