zoukankan      html  css  js  c++  java
  • kaggle入门题Titanic

    集成开发环境:Pycharm

    python版本:2.7(anaconda库)

    用到的库:科学计算库numpy,数据分析包pandas,画图包matplotlib,机器学习库sklearn

    大体步骤分为三步:

    1.数据分析

    2.交叉验证

    3.预测并输出结果

    导入库函数

    import numpy as np
    import pandas as pa
    import matplotlib.pyplot as pl
    from sklearn.linear_model import LogisticRegression
    from sklearn import model_selection
    # 最后一个库是原来cross_validation的替代,原来那个过时了,再用会有警告,其实里边的函数都没变

    第一步:数据分析

    1.1通过画图分析出数据中对预测影响很小的特征值,忽略这些特征可以减小工作量

     1 # pandas库里的read_csv可以把csv文件读取为dataFrame对象,这个对象就是很多列(数组)共享一个index,这个index 是真的存在的且可以指定,每一列都有name
     2 # 其形状大小是真实数据的形状,不包括index和name
     3 # 路径可以是绝对路径和相对路径,注意路径中的反斜线是两根,或者一根/
     4 train_obj = pa.read_csv('train.csv')
     5 test_obj = pa.read_csv('test.csv')
     6 
     7 # 首先把一看就知道对结果没有影响的数据去掉
     8 # DataFrame对象的drop方法:删除行或者列,可以多行,drop('行名或者列名',axis=0/1(0是行,1是列)),多个行名或者列名:['','']
     9 trainData = train_obj.drop(['Name', 'Ticket', 'Cabin'], axis=1)
    10 testData = test_obj.drop(['Name', 'Ticket', 'Cabin'], axis=1)
    11 
    12 # 下面用画图的形式对其他变量对最后结果的影响作出判断:
    13 # 首先考察船舱和性别,用groupby函数对的数据进行分类,groupby('列名'),根据给出的列,把具有相同数值的合为一组,组数为列值的种数
    14 # 如果groupby(['列名1','列名2']),则列1为分大类,大类中再根据列2进行分类
    15 train_group1 = trainData.groupby(['Sex', 'Pclass'])  # 返回类型是Groupby类型,里边有key1*key2组
    16 # sum()是将各组内每个特征的所有数据相加,返回也是key1*key2组,只不过每组里只有一行数据,是所有数据之和
    17 # count()是统计各组每个特征的数据量(由于有缺失数据,所以各个特征间是不一定一样的)
    18 rate = (train_group1.sum() / train_group1.count())['Survived']  # 有6组,每组都只有一个值,生存率
    19 
    20 # 下面开始画图,柱状图
    21 x = np.array([1, 2, 3])
    22 wi = 0.15
    23 # bar(left,height,width,color)用来化柱状图,参数分别是每个柱子的中心位置坐标,高度(一般就是数据),宽度,颜色,前三个一般都是集合
    24 pl.bar(x-wi, rate.female, wi*2, color='r')
    25 pl.bar(x+wi, rate.male, wi*2, color='b')
    26 pl.title("survived rate")
    27 # 坐标轴名称
    28 pl.xlabel("Pclass")
    29 pl.ylabel("rate")
    30 # 指定网格线格式
    31 pl.grid(True, linestyle='-', color='0.7')
    32 # 轴的刻度
    33 pl.xticks([1, 2, 3])
    34 # arange(起点,终点,间隔),创建一个等差序列
    35 pl.yticks(np.arange(0.0, 1.1, 0.1))
    36 # 对不同的柱做标记
    37 pl.legend(['female', 'male'])
    38 pl.show()
    39 # 接下来的过程和上边差不多,分别是判断了年龄、上船地点、船上亲属人数

    分析性别和上船地点的结果,结果显示两个特征的影响都需要保留

    1.2数据预处理,将特征中比较相似的直接相加作为一个,非数值类型转为数值类型,对缺失值进行处理

     1 # 船上的亲属人数就是把两个代表家属的变量相加
     2 trainData['family'] = trainData['SibSp'] + trainData['Parch']
     3 # pandas中的map()函数一般是用来进行数据转换的,参数是函数或者字典,一个series调用map(),会对照字典用的key,将series中的key全部改成相应的value,map()函数返回一个新的,对源对象不操作
     4 trainData['Sex'] = trainData['Sex'].map({'female': 0, 'male': 1})
     5 trainData['Embarked'] = trainData['Embarked'].map({'C': 0, 'Q': 1, 'S': 2})
     6 # numpy.isnan()函数的作用是返回一个Boolean数组,对应input的下标,若当前位置是nan那么就是true,否则false
     7 # 对一个数组挑选时可以用array[True,False,True...]True就是选择
     8 # 挑选出age为nan的,这个集合不用age这个特征,做四特征预测
     9 train_age_nan = trainData[np.isnan(trainData['Age'])]
    10 # 挑选出age不为空的数据,直接调用dropna(),这个方法会去掉输入数组中有nan特征的数据,挑选后的具备所有特征,所以可以用所有特征
    11 train_all = trainData.dropna()
    12 # 对于以上两个数据集分别挑选出需要的特征组成训练集,因为之前已经确认了有的是对结果没有影响的,所以可以只挑选有用的
    13 # dataframe选取列的格式是两个方括号,[['列名','列名']]
    14 train_five = train_all[['Sex', 'Embarked', 'family', 'Age', 'Pclass']]
    15 train_four = train_age_nan[['Sex', 'Embarked', 'family', 'Pclass']]
    16 
    17 # 两个数据集的标记数组
    18 train_five_label = train_all['Survived']
    19 train_four_label = train_age_nan['Survived']
    20 
    21 # 至此处理数据工作全部完成,接下来就是训练模型

    2.交叉验证

     1 # 使用分类器类
     2 classifier = LogisticRegression(C=10000)  # 这个作为5特征分类器,这里的c是正则化系数,防止系统由于过于复杂而产生过拟合
     3 # 调用cross_validation.cross_val_score(分类器,data,target,cv),这个函数返回每次交叉验证的准确率,方法是StratifiedKFold,具体看http://blog.sina.com.cn/s/blog_6a90ae320101a5rc.html
     4 # 函数返回一个数组
     5 score = model_selection.cross_val_score(classifier, train_five, train_five_label, cv=5)  # 交叉验证
     6 classifier2 = LogisticRegression()  # 这个作为4特征分类器
     7 score2 = model_selection.cross_val_score(classifier2, train_four, train_four_label, cv=5)  # 交叉验证
     8 # 查看平均值
     9 print "五特征分类器交叉验证结果:"
    10 print score.mean()
    11 print "四特征分类器交叉验证结果:"
    12 print score2.mean()

    3.输出结果

     1 # 下面开始输出结果
     2 
     3 # 首先处理测试集
     4 testData['Sex'] = testData['Sex'].map({'female': 0, 'male': 1})
     5 testData['Embarked'] = testData['Embarked'].map({'C': -1, 'Q': 0, 'S': 1})
     6 testData['family'] = testData['SibSp'] + testData['Parch']
     7 # 这里不能用两个数据集,因为如果将数据分开分别判断,那么无法和乘客号对应
     8 # 读取官方结果
     9 answer = pa.read_csv('gender_submission.csv')
    10 
    11 # 训练分类器
    12 classifier.fit(train_five, train_five_label)
    13 classifier2.fit(train_four, train_four_label)
    14 
    15 # 输出测试集结果
    16 passengers = pa.DataFrame(testData['PassengerId'])
    17 # 这里必须先创建一个列,直接赋值会出错
    18 passengers['Survived'] = ''
    19 # isnan()是numpy库里的函数
    20 #这里有个问题就是如果直接对passengers['Survived'][i]每行赋值的话会有SettingCopyWarning警告,正确的方法是先生成一个数组temp,然后把数组赋值给列,直接添加会有警告
    21 #这里创建数组不能用np.zeros(),因为用zeros生成的是float数组,最后结果要求int类型
    22 temp = [0] * passengers.shape[0]
    23 for i in range(len(testData)):
    24     if np.isnan(testData['Age'][i]):
    25         # 这里也有一个问题:由于需要一个一个做预测,但是predict函数要求输入二维数组,单行数据是一维的
    26         #所以要把数据转为numpy中的array,然后转为二维数组
    27         #reshape()函数可以利用原数组数据改变数组的尺寸,但是主要新旧数组数据量相同,而reshape(1,-1)可以将[xxx]转为[[xxx]]这种,就可以用了
    28         temp[i] = int(classifier2.predict(np.array( testData[['Pclass', 'Sex', 'Embarked', 'family']].ix[i].dropna()).reshape(1,-1)))
    29     else:
    30         temp[i] = int(classifier.predict(np.array(testData[['Sex', 'Embarked', 'family', 'Age', 'Pclass']].ix[i]).reshape(1,-1)))
    31 print temp
    32 passengers['Survived'] = temp
    33 # 结算准确率
    34 acc = 1 - float(sum(abs(answer['Survived'] - passengers['Survived']))) / len(passengers)
    35 print "准确率是:"
    36 print acc
    37 # 输出文件
    38 passengers.to_csv('result.csv',encoding='utf-8',index=False)

     最后的结果,准确率有点低,再找找原因

  • 相关阅读:
    PHP入门:在Windows中安装PHP工作环境
    Oracle数据库优化的经验总结
    引入js和css文件的总结
    PHP的加密解密字符串函数
    js+html5双人五子棋(源码下载)
    JAVA的网络编程基础概念
    asp.net的code-Behind技术
    10个调试Java的技巧
    Oracle基础 各种语句的定义格式
    CSS行高line-height的理解
  • 原文地址:https://www.cnblogs.com/stAr-1/p/8044338.html
Copyright © 2011-2022 走看看