摘要:關于數(shù)據(jù)來源本項目寫于年七月初,主要使用爬取網(wǎng)貸之家以及人人貸的數(shù)據(jù)進行分析。注這是現(xiàn)在網(wǎng)貸之家的請求后臺的接口,爬蟲編寫的時候與數(shù)據(jù)接口與如今的請求接口不一樣,所以網(wǎng)貸之家的數(shù)據(jù)爬蟲部分已無效。
關于數(shù)據(jù)來源
本項目寫于2017年七月初,主要使用Python爬取網(wǎng)貸之家以及人人貸的數(shù)據(jù)進行分析。
網(wǎng)貸之家是國內最大的P2P數(shù)據(jù)平臺,人人貸國內排名前二十的P2P平臺。
源碼地址
抓包工具主要使用chrome的開發(fā)者工具 網(wǎng)絡一欄,網(wǎng)貸之家的數(shù)據(jù)全部是ajax返回json數(shù)據(jù),而人人貸既有ajax返回數(shù)據(jù)也有html頁面直接生成數(shù)據(jù)。
請求實例
從數(shù)據(jù)中可以看到請求數(shù)據(jù)的方式(GET或者POST),請求頭以及請求參數(shù)。
從請求數(shù)據(jù)中可以看到返回數(shù)據(jù)的格式(此例中為json)、數(shù)據(jù)結構以及具體數(shù)據(jù)。
注:這是現(xiàn)在網(wǎng)貸之家的API請求后臺的接口,爬蟲編寫的時候與數(shù)據(jù)接口與如今的請求接口不一樣,所以網(wǎng)貸之家的數(shù)據(jù)爬蟲部分已無效。
根據(jù)抓包分析得到的結果,構造請求。在本項目中,使用Python的 requests庫模擬http請求
具體代碼:
import requests class SessionUtil(): def __init__(self,headers=None,cookie=None): self.session=requests.Session() if headers is None: headersStr={"Accept":"application/json, text/javascript, */*; q=0.01", "X-Requested-With":"XMLHttpRequest", "User-Agent":"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/57.0.2987.133 Safari/537.36", "Accept-Encoding":"gzip, deflate, sdch, br", "Accept-Language":"zh-CN,zh;q=0.8" } self.headers=headersStr else: self.headers=headers self.cookie=cookie //發(fā)送get請求 def getReq(self,url): return self.session.get(url,headers=self.headers).text def addCookie(self,cookie): self.headers["cookie"]=cookie //發(fā)送post請求 def postReq(self,url,param): return self.session.post(url, param).text
在設置請求頭的時候,關鍵字段只設置了"User-Agent",網(wǎng)貸之家和人人貸的沒有反爬措施,甚至不用設置"Referer"字段來防止跨域錯誤。
爬蟲實例以下是一個爬蟲實例
import json import time from databaseUtil import DatabaseUtil from sessionUtil import SessionUtil from dictUtil import DictUtil from logUtil import LogUtil import traceback def handleData(returnStr): jsonData=json.loads(returnStr) platData=jsonData.get("data").get("platOuterVo") return platData def storeData(jsonOne,conn,cur,platId): actualCapital=jsonOne.get("actualCapital") aliasName=jsonOne.get("aliasName") association=jsonOne.get("association") associationDetail=jsonOne.get("associationDetail") autoBid=jsonOne.get("autoBid") autoBidCode=jsonOne.get("autoBidCode") bankCapital=jsonOne.get("bankCapital") bankFunds=jsonOne.get("bankFunds") bidSecurity=jsonOne.get("bidSecurity") bindingFlag=jsonOne.get("bindingFlag") businessType=jsonOne.get("businessType") companyName=jsonOne.get("companyName") credit=jsonOne.get("credit") creditLevel=jsonOne.get("creditLevel") delayScore=jsonOne.get("delayScore") delayScoreDetail=jsonOne.get("delayScoreDetail") displayFlg=jsonOne.get("displayFlg") drawScore=jsonOne.get("drawScore") drawScoreDetail=jsonOne.get("drawScoreDetail") equityVoList=jsonOne.get("equityVoList") experienceScore=jsonOne.get("experienceScore") experienceScoreDetail=jsonOne.get("experienceScoreDetail") fundCapital=jsonOne.get("fundCapital") gjlhhFlag=jsonOne.get("gjlhhFlag") gjlhhTime=jsonOne.get("gjlhhTime") gruarantee=jsonOne.get("gruarantee") inspection=jsonOne.get("inspection") juridicalPerson=jsonOne.get("juridicalPerson") locationArea=jsonOne.get("locationArea") locationAreaName=jsonOne.get("locationAreaName") locationCity=jsonOne.get("locationCity") locationCityName=jsonOne.get("locationCityName") manageExpense=jsonOne.get("manageExpense") manageExpenseDetail=jsonOne.get("manageExpenseDetail") newTrustCreditor=jsonOne.get("newTrustCreditor") newTrustCreditorCode=jsonOne.get("newTrustCreditorCode") officeAddress=jsonOne.get("officeAddress") onlineDate=jsonOne.get("onlineDate") payment=jsonOne.get("payment") paymode=jsonOne.get("paymode") platBackground=jsonOne.get("platBackground") platBackgroundDetail=jsonOne.get("platBackgroundDetail") platBackgroundDetailExpand=jsonOne.get("platBackgroundDetailExpand") platBackgroundExpand=jsonOne.get("platBackgroundExpand") platEarnings=jsonOne.get("platEarnings") platEarningsCode=jsonOne.get("platEarningsCode") platName=jsonOne.get("platName") platStatus=jsonOne.get("platStatus") platUrl=jsonOne.get("platUrl") problem=jsonOne.get("problem") problemTime=jsonOne.get("problemTime") recordId=jsonOne.get("recordId") recordLicId=jsonOne.get("recordLicId") registeredCapital=jsonOne.get("registeredCapital") riskCapital=jsonOne.get("riskCapital") riskFunds=jsonOne.get("riskFunds") riskReserve=jsonOne.get("riskReserve") riskcontrol=jsonOne.get("riskcontrol") securityModel=jsonOne.get("securityModel") securityModelCode=jsonOne.get("securityModelCode") securityModelOther=jsonOne.get("securityModelOther") serviceScore=jsonOne.get("serviceScore") serviceScoreDetail=jsonOne.get("serviceScoreDetail") startInvestmentAmout=jsonOne.get("startInvestmentAmout") term=jsonOne.get("term") termCodes=jsonOne.get("termCodes") termWeight=jsonOne.get("termWeight") transferExpense=jsonOne.get("transferExpense") transferExpenseDetail=jsonOne.get("transferExpenseDetail") trustCapital=jsonOne.get("trustCapital") trustCreditor=jsonOne.get("trustCreditor") trustCreditorMonth=jsonOne.get("trustCreditorMonth") trustFunds=jsonOne.get("trustFunds") tzjPj=jsonOne.get("tzjPj") vipExpense=jsonOne.get("vipExpense") withTzj=jsonOne.get("withTzj") withdrawExpense=jsonOne.get("withdrawExpense") sql="insert into problemPlatDetail (actualCapital,aliasName,association,associationDetail,autoBid,autoBidCode,bankCapital,bankFunds,bidSecurity,bindingFlag,businessType,companyName,credit,creditLevel,delayScore,delayScoreDetail,displayFlg,drawScore,drawScoreDetail,equityVoList,experienceScore,experienceScoreDetail,fundCapital,gjlhhFlag,gjlhhTime,gruarantee,inspection,juridicalPerson,locationArea,locationAreaName,locationCity,locationCityName,manageExpense,manageExpenseDetail,newTrustCreditor,newTrustCreditorCode,officeAddress,onlineDate,payment,paymode,platBackground,platBackgroundDetail,platBackgroundDetailExpand,platBackgroundExpand,platEarnings,platEarningsCode,platName,platStatus,platUrl,problem,problemTime,recordId,recordLicId,registeredCapital,riskCapital,riskFunds,riskReserve,riskcontrol,securityModel,securityModelCode,securityModelOther,serviceScore,serviceScoreDetail,startInvestmentAmout,term,termCodes,termWeight,transferExpense,transferExpenseDetail,trustCapital,trustCreditor,trustCreditorMonth,trustFunds,tzjPj,vipExpense,withTzj,withdrawExpense,platId) values (""+actualCapital+"",""+aliasName+"",""+association+"",""+associationDetail+"",""+autoBid+"",""+autoBidCode+"",""+bankCapital+"",""+bankFunds+"",""+bidSecurity+"",""+bindingFlag+"",""+businessType+"",""+companyName+"",""+credit+"",""+creditLevel+"",""+delayScore+"",""+delayScoreDetail+"",""+displayFlg+"",""+drawScore+"",""+drawScoreDetail+"",""+equityVoList+"",""+experienceScore+"",""+experienceScoreDetail+"",""+fundCapital+"",""+gjlhhFlag+"",""+gjlhhTime+"",""+gruarantee+"",""+inspection+"",""+juridicalPerson+"",""+locationArea+"",""+locationAreaName+"",""+locationCity+"",""+locationCityName+"",""+manageExpense+"",""+manageExpenseDetail+"",""+newTrustCreditor+"",""+newTrustCreditorCode+"",""+officeAddress+"",""+onlineDate+"",""+payment+"",""+paymode+"",""+platBackground+"",""+platBackgroundDetail+"",""+platBackgroundDetailExpand+"",""+platBackgroundExpand+"",""+platEarnings+"",""+platEarningsCode+"",""+platName+"",""+platStatus+"",""+platUrl+"",""+problem+"",""+problemTime+"",""+recordId+"",""+recordLicId+"",""+registeredCapital+"",""+riskCapital+"",""+riskFunds+"",""+riskReserve+"",""+riskcontrol+"",""+securityModel+"",""+securityModelCode+"",""+securityModelOther+"",""+serviceScore+"",""+serviceScoreDetail+"",""+startInvestmentAmout+"",""+term+"",""+termCodes+"",""+termWeight+"",""+transferExpense+"",""+transferExpenseDetail+"",""+trustCapital+"",""+trustCreditor+"",""+trustCreditorMonth+"",""+trustFunds+"",""+tzjPj+"",""+vipExpense+"",""+withTzj+"",""+withdrawExpense+"",""+platId+"")" cur.execute(sql) conn.commit() conn,cur=DatabaseUtil().getConn() session=SessionUtil() logUtil=LogUtil("problemPlatDetail.log") cur.execute("select platId from problemPlat") data=cur.fetchall() print(data) mylist=list() print(data) for i in range(0,len(data)): platId=str(data[i].get("platId")) mylist.append(platId) print mylist for i in mylist: url=""+i try: data=session.getReq(url) platData=handleData(data) dictObject=DictUtil(platData) storeData(dictObject,conn,cur,i) except Exception,e: traceback.print_exc() cur.close() conn.close
整個過程中 我們 構造請求,然后把解析每個請求的響應,其中json返回值使用json庫進行解析,html頁面使用BeautifulSoup庫進行解析(結構復雜的html的頁面推薦使用lxml庫進行解析),解析到的結果存儲到mysql數(shù)據(jù)庫中。
爬蟲代碼爬蟲代碼地址(注:爬蟲使用代碼Python2與python3都可運行,本人把爬蟲代碼部署在阿里云服務器上,使用Python2 運行)
數(shù)據(jù)分析數(shù)據(jù)分析主要使用Python的numpy、pandas、matplotlib進行數(shù)據(jù)分析,同時輔以海致BDP。
時間序列分析 數(shù)據(jù)讀取一般采取把數(shù)據(jù)讀取pandas的DataFrame中進行分析。
以下就是讀取問題平臺的數(shù)據(jù)的例子
problemPlat=pd.read_csv("problemPlat.csv",parse_dates=True)#問題平臺
數(shù)據(jù)結構
eg 問題平臺數(shù)量隨時間變化
problemPlat["id"]["2012":"2017"].resample("M",how="count").plot(title="P2P發(fā)生問題")#發(fā)生問題P2P平臺數(shù)量 隨時間變化趨勢
圖形化展示
使用海致BDP完成(Python繪制地圖分布輪子比較復雜,當時還未學習)
各省問題平臺數(shù)量 各省平臺成交額 規(guī)模分布分析eg 全國六月平臺成交額分布
代碼
juneData["amount"].hist(normed=True) juneData["amount"].plot(kind="kde",style="k--")#六月份交易量概率分布
核密度圖形展示
成交額取對數(shù)核密度分布
np.log10(juneData["amount"]).hist(normed=True) np.log10(juneData["amount"]).plot(kind="kde",style="k--")#取 10 對數(shù)的 概率分布
圖形化展示
可看出取10的對數(shù)后分布更符合正常的金字塔形。
lujinData=platVolume[platVolume["wdzjPlatId"]==59] corr=pd.rolling_corr(lujinData["amount"],allPlatDayData["amount"],50,min_periods=50).plot(title="陸金所交易額與所有平臺交易額的相關系數(shù)變化趨勢")
圖形化展示
車貸平臺與全平臺成交額數(shù)據(jù)對比
carFinanceDayData=carFinanceData.resample("D").sum()["amount"] fig,axes=plt.subplots(nrows=1,ncols=2,sharey=True,figsize=(14,7)) carFinanceDayData.plot(ax=axes[0],title="車貸平臺交易額") allPlatDayData["amount"].plot(ax=axes[1],title="所有p2p平臺交易額")趨勢預測 eg預測陸金所成交量趨勢(使用Facebook Prophet庫完成)
lujinAmount=platVolume[platVolume["wdzjPlatId"]==59] lujinAmount["y"]=lujinAmount["amount"] lujinAmount["ds"]=lujinAmount["date"] m=Prophet(yearly_seasonality=True) m.fit(lujinAmount) future=m.make_future_dataframe(periods=365) forecast=m.predict(future) m.plot(forecast)
趨勢預測圖形化展示
數(shù)據(jù)分析代碼地址(注:數(shù)據(jù)分析代碼智能運行在Python3 環(huán)境下)
代碼運行后樣例(無需安裝Python環(huán)境 也可查看具體代碼解圖形化展示)
這是本人從 Java web轉向數(shù)據(jù)方向后自己寫的第一項目,也是自己的第一個Python項目,在整個過程中,也沒遇到多少坑,整體來說,爬蟲和數(shù)據(jù)分析以及Python這門語言門檻都是非常低的。
如果想入門Python爬蟲,推薦《Python網(wǎng)絡數(shù)據(jù)采集》
如果想入門Python數(shù)據(jù)分析,推薦 《利用Python進行數(shù)據(jù)分析》
文章版權歸作者所有,未經(jīng)允許請勿轉載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉載請注明本文地址:http://systransis.cn/yun/41378.html
摘要:利用這一業(yè)務邏輯,惡意爬蟲通過各類社工庫拿到一批手機號后可以在短時內驗證這批號碼是否為某一網(wǎng)站的注冊用戶。事前的甄別預防才是關鍵惡意爬蟲在給網(wǎng)站帶來可觀訪問量的同時,也帶來了難以估量的威脅和損失。 整個互聯(lián)網(wǎng)的流量中,真人占比有多少? 80% ??60% ??50% ? showImg(https://segmentfault.com/img/bVGSra?w=350&h=346); ...
摘要:之前寫了一個電商爬蟲系列的文章,簡單的給大家展示了一下爬蟲從入門到進階的路徑,但是作為一個永遠走在時代前沿的科技工作者,我們從來都不能停止。金融數(shù)據(jù)實在是價值大,維度多,來源廣。由于也是一種,因此通常來說,在中抽取某個元素是通過來做的。 相關教程: 手把手教你寫電商爬蟲-第一課 找個軟柿子捏捏 手把手教你寫電商爬蟲-第二課 實戰(zhàn)尚妝網(wǎng)分頁商品采集爬蟲 手把手教你寫電商爬蟲-第三課 實戰(zhàn)...
摘要:之前寫了一個電商爬蟲系列的文章,簡單的給大家展示了一下爬蟲從入門到進階的路徑,但是作為一個永遠走在時代前沿的科技工作者,我們從來都不能停止。金融數(shù)據(jù)實在是價值大,維度多,來源廣。由于也是一種,因此通常來說,在中抽取某個元素是通過來做的。 相關教程: 手把手教你寫電商爬蟲-第一課 找個軟柿子捏捏 手把手教你寫電商爬蟲-第二課 實戰(zhàn)尚妝網(wǎng)分頁商品采集爬蟲 手把手教你寫電商爬蟲-第三課 實戰(zhàn)...
摘要:的關鍵技術主要有內容存儲和分發(fā)技術。分發(fā)本身是和存儲密不可分的存儲和分發(fā)的實質都是數(shù)據(jù)的讀取和使用,兩者是不可能分割的。只是存儲場景和分發(fā)場景,設計有些不同,服務質量的要求也不一樣。根據(jù)區(qū)域和時段的不同,存儲的價格也會有不同。 showImg(https://segmentfault.com/img/remote/1460000019478027); PPIO 是為開發(fā)者打造的去中心化...
閱讀 3778·2021-09-02 09:53
閱讀 2758·2021-07-30 14:57
閱讀 3504·2019-08-30 13:09
閱讀 1206·2019-08-29 13:25
閱讀 817·2019-08-29 12:28
閱讀 1463·2019-08-29 12:26
閱讀 1136·2019-08-28 17:58
閱讀 3315·2019-08-26 13:28