zoukankan      html  css  js  c++  java
  • Python 北京二手房成交数据分析过程

    此为之前偶尔在社区看到的优秀作业“链家2011-2016北京二手房成交数据分析”,在此为了工作简历上的项目巩固复习练习一次。

    环境准备

    import numpy as np
    import pandas as pd
    import matplotlib.pyplot as plt
    %matplotlib inline

    插入数据

    #数据读取
    f=open(r'D:DocumentsTencent Files2698968530FileRecv日月光华链家成交数据lianjia1.csv') data=pd.read_csv(f)

    观察数据

    data.head()

    合并

    数据源一共是7个csv文件,文件名是“lianjia+1到7”,可以使用循环语句将七个文件写入到一个列表dataR中

    dataR=[]
    for i in range(1,8):
        f=open(r'D:DocumentsTencent Files2698968530FileRecv日月光华链家成交数据lianjia{}.csv'.format(i))
        data=pd.read_csv(f)
        dataR.append(data)

    出现了error 

    UnicodeDecodeError: 'gbk' codec can't decode byte 0x97 in position 154: illegal multibyte sequence
    这说明7个文件的编码类型还不一样,部分数据不能用gbk类型解码。那我们就需要使用try except方法,先try gbk编码,不行就except使用我们默认的utf-8编码
    dataY=[]
    for i in range(1,8):
        try:
            f=open(r'D:DocumentsTencent Files2698968530FileRecv日月光华链家成交数据lianjia{}.csv'.format(i),encoding='gbk')
            data=pd.read_csv(f)
        except:
            f=open(r'D:DocumentsTencent Files2698968530FileRecv日月光华链家成交数据lianjia{}.csv'.format(i),encoding='utf-8')
            data=pd.read_csv(f)
        dataY.append(data)

    输出列表长度看是否是7

    len(dataY)

    随便取一个的几行看看

    dataY[2].tail()

    数据正常显示,说明并没有问题。但是这是一个list,我们怎么把7个数据合并到一起呢?这就需要使用到pandas包里面的concat函数。

    data=pd.concat(dataY)

    对他进行描述性统计分析

    data.shape

    data.describe()

    data.info()

    data.head()

    这个数据集一共有141140条数据,14个属性值,只有“套数”和“总价”是数值类型,其他的都是字符串类型,再取他的前3行进行观察,发现成交单价是字符串类型,因为它写的是xxx元/平。后期还需要再对成交单价进行数据的处理操作。

    接下来我们要对数据进行预处理了,但是首先要想到的是,数据有没有缺失值,通过对结果是否报错来判断是否有缺失值

    data.isnull()

    ...

    对这些布尔值进行sum运算,可以得出有多少缺失值

    (data.isnull()).sum()

     结果得出版块(bankuai)数据缺失值为1321,门店(mendian)缺失值为13条,其他的数据缺失值都是1条。

    data[data.cjdanjia.isnull()]
    data.dropna(how='all',inplace=True)
    data.isnull().sum()

    发现第57119条数据缺失是很多属性一起缺失的,使用drop_duplicates把这条数据删除,在此之前,因为这个函数是删除重复的下一条数据,因此需要将数据按照地区排序,将空值放在后位进行删除,最后进行检查。

    用duplicated的subset参数指定重复的列,查找出来这些列重复的数据。然后再排序

    (data.duplicated(subset=['cjdanjia','cjxiaoqu','cjlouceng','bankuai'])).sum()
    data.sort_values(by='bankuai',inplace=True)

    再使用drop_duplicates函数,就可以去掉这些重复值了,同时能保留板块的有效信息

    data.drop_duplicates(subset=['cjdanjia','cjxiaoqu','cjlouceng'],inplace=True)

    数据类型转换,异常值处理,数据离散化

    我们想对成交单价进行分析,这列数据非常重要。但是它是字符串形式,我们要把单价和'元/平'分开来。首先我们先看一下是不是所有数据包含了'元/平'。波浪号~放在语句前面表示否定。

    data.head()
    (~data.cjdanjia.str.contains('元/平')).sum()

    得出结果为0。 得出不包含'元/平'的数据数量为0 ,则就是都有'元/平'。那我们定义一个lambda x函数,把这里数据进行转换,把'元/平'替换为空字符串

    data.cjdanjia.map(lambda x:round(float(x.replace('元/平',''))/10000,2))

    把元/平变成了空字符串,那么数据就只留下了单价数值。然后我们把这个单价从字符串object类型,astype变成float类型,便于后面的计算。然后除以10000,用round函数保留2位小数点。这样得出来的结果就是3.45万,5.31万的类型。

    看成交单价的最大最小值

    data.cjdanjia.min()
    data.cjdanjia.max()

    发现最小值为0,去掉0的数据,再看最小值

    data=data[data.cjdanjia>0]
    data.cjdanjia.min()

    此时最小值为0.01,还是不正常,为了处理这样的异常值,我们需要设置一个范围,比如5000元一平,往上的数据才算有效数据

    data=data[data.cjdanjia>0.5]
    data.cjdanjia.min()

    此时最小值为0..51,单价数据是我们想要的数据类型了,我们想把这些数据进行离散化,分成多个区间,看成交单价的分布,这个时候就需要使用到bins和cut函数

    bins=[0,1,2,3,4,5,7,9,11,13,15]
    pd.cut(data.cjdanjia,bins)

    再对这份数据进行value_counts,看看落在各个区间上的数据都有多少

    pd.cut(data.cjdanjia,bins).value_counts()

    画出点图

    pd.cut(data.cjdanjia,bins).value_counts().plot()

    然后直接画个柱状图看看。rot是让x轴标签倾斜20度,不然会挤在一起。

    pd.cut(data.cjdanjia,bins).value_counts().plot.bar(rot=20)

    也可以画出饼图

    pd.cut(data.cjdanjia,bins).value_counts().plot.pie(figsize=(8,8))

    字符串的处理

    首先,我们看看是不是所有数据都包含这三个数据,也就是用/分开之后,是不是都是三个数据,以免套用函数报错

    (data.cjlouceng.str.split('/').map(len)!=3).sum()

        0

    data.cjlouceng

    可以把朝向这个数据单独取出来之后,单独给原表增加一列'chaoxiang

    data.cjlouceng.map(lambda x:x.split('/')[0])
    data['chaoxiang']=data.cjlouceng.map(lambda x:x.split('/')[0])

    楼层这列也这样处理

    data['louceng']=data.cjlouceng.map(lambda x:x.split('/')[1])
    data

    对楼层取unique,可以看出还有未知这个数据,我们把未知这类数据去掉。(原始数据还有空字符串' ',之前处理的时候已经查找出来了,但是没有记录在此)

    data.louceng.unique()

    还有未知和空字符串的部分数据也需要处理

    data[data.louceng=='']
    data[data.louceng=='未知']
    data=data[(data.louceng!='未知')&(data.louceng!='')]
    data

    pd.get_dummies(data.louceng)

    然后我们可以使用get_dummies对楼层的这几个类别进行one-hot处理,这样就能非常方便离散化处理,然后得出各个类别的counts。

    然后再使用join函数,把这个结果直接插入到原表后面去

    data.join(pd.get_dummies(data.louceng))

    再进行sum,得出各个类别的数量

    pd.get_dummies(data.louceng).sum()

    他的柱形图

    pd.get_dummies(data.louceng).sum().plot.bar()

    把数据导出成csv文件。为了防止index变成乱码,添加用utf_8_sig编码的参数。

    (pd.get_dummies(data.louceng).sum()).to_csv('loucengfenbu3.csv',encoding='utf_8_sig')

    分组运算、布尔过滤和数据透视

    首先,对于成交时间进行处理,仅取出中间的年

    data['cjshijian']=data.cjshijian.map(lambda x:x.split('')[1])

    先进行分隔,取后面的时间,再再按照-进行分隔,取年份

    data['year']=data.cjshijian.map(lambda x:x.split('-')[0])
    data.groupby(['year','xingming'])['xingming'].value_counts()

    分析每一年的经纪人数量。按照年份,经纪人姓名分组

    data.groupby(['year','xingming'])['xingming'].count()

    研究成交总价大于1亿的经纪人的工作年限。我们可以先分组,然后再sum,查出大于1亿的数据

    data_group=data.groupby(['xingming','congyenianxian'])['cjzongjia'].sum()
    data_group[data_group>10000]

     

    等等也可以研究其他的问题。

  • 相关阅读:
    Java LinkList遍历方式
    Java LinkedList的实现原理
    ArrayList 原理(2)
    ArrayList 原理(1)
    Java中HashMap的实现原理
    Java HashMap两种遍历方式
    【转】Unity3d:读取FBX中的动画
    C#与U3D中字符串尾0
    U3D中的又一个坑
    MaxScript 学习笔记【有转载】
  • 原文地址:https://www.cnblogs.com/RR-99/p/10370884.html
Copyright © 2011-2022 走看看