kaggle入门题Titanic

时间:2022-12-06 05:16:51

集成开发环境: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 # 接下来的过程和上边差不多,分别是判断了年龄、上船地点、船上亲属人数

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

kaggle入门题Titanic

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)

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

kaggle入门题Titanic