教你用Python制作一款带有界面的NBA爬虫小程序

一、前言

有时将代码转成带有界面的程序,会极大的方便使用,虽然在网上有很多现成的GUI系统,但是套用别人的代码,心里难免有些尴尬,所以本文将用Python爬虫结合wxpython模块构造一个NBA爬虫小软件,演示效果如下

本文框架构造将分为二个部分讲解:

构建GUI界面

举例套用爬虫框架

主要涉及的Python模块有

requests

wx

pymysql

pandas

  二、GUI界面设计

首先介绍下流程:GUI界面设计讲解插入界面背景图片

设计GUI界面的代码思路其实很简单,首先导入wx库

#一、引用模块

importwx

这里引用的模块是wxpython模块,建立GUI的模块很多,常见的有PyQt、Tkinter等。这些模块各有各的优缺点,读者可以翻阅相关资料进行选择。

#二、定义全局变量(创建面板及布局)

classMyFrame(wx.Frame):

def__init__(self, parent, id):

wx.Frame.__init__(self, parent, id,'titlename',size=(400,300))

panel = wx.Panel(self)

self.bt_confirm = wx.Button(panel, label='name1')

self.bt_confirm.Bind(wx.EVT_BUTTON,self.OnclickSubmit)

self.bt_cancel = wx.Button(panel, label='name2')

self.bt_cancel.Bind(wx.EVT_BUTTON,self.OnclickCancel)

self.InitUI()

定义全局变量对于初级的GUI来说就是构建一个形式窗口+按钮布置,不需要自建一个模块。但对于高级的GUI诸如投资系统而言,全局变量是尤为重要的,换句话说全局变量需要放在一个py文件中初始化。

上述代码是创建部分的代码,个性化布局需要添加容器进行设置,稍后会在总代码中呈现。

#三、调用局部变量并绑定事件

defInitUI(self):

""" 点击InitUI,执行方法 """

defOnclickSubmit(self,event):

""" 绑定OnclickSubmit事件 """

简单来说就是绑定事件,该事件是你点击对应按钮产生的效果。这部是整个GUI的核心,如果你在做签到系统,那么你就要绑定一个导入员工名单txt文件的事件。

#四、GUI执行脚本

if__name__ =='__main__':

app = wx.App()# 初始化

frame = MyFrame(parent=None,id=-1)

frame.Show()

app.MainLoop()# 调用主循环

delapp

第四步的基本套路就是如此。

  三、举例实现

以一个简单的NBA爬虫系统为例,首先创建面板与布局👇

classMyFrame(wx.Frame):

def__init__(self, parent, id):

wx.Frame.__init__(self, parent, id,'NBA可视化',size=(400,300))

panel = wx.Panel(self)

self.bt_confirm = wx.Button(panel, label='合同信息')

self.bt_confirm.Bind(wx.EVT_BUTTON,self.OnclickSubmit)

self.bt_cancel = wx.Button(panel, label='清空')

self.bt_cancel.Bind(wx.EVT_BUTTON,self.OnclickCancel)

self.bt_imf = wx.Button(panel, label='可视化')

self.bt_imf.Bind(wx.EVT_BUTTON,self.Onclickvisual)

self.bt_team = wx.Button(panel, label='球队信息',pos=(280,20))

self.bt_team.Bind(wx.EVT_BUTTON,self.Onclickteam)

self.bt_obtain = wx.Button(panel, label='球员信息',pos=(20,20))

self.bt_obtain.Bind(wx.EVT_BUTTON,self.Onclickimfor)

self.bt_ml = wx.Button(panel, label='得分榜')

self.bt_ml.Bind(wx.EVT_BUTTON,self.Onclickmql)

self.title = wx.StaticText(panel, label="NBA可视化")

self.label_user = wx.StaticText(panel, label="球队名称")

self.text_user = wx.TextCtrl(panel, style=wx.TE_LEFT)

self.label_pwd = wx.StaticText(panel, label="球员名称")

self.text_pwd = wx.TextCtrl(panel, style=wx.TE_LEFT)

self.label_path = wx.StaticText(panel, label="储存路径")

self.text_pathword = wx.TextCtrl(panel, style=wx.TE_LEFT)

然后添加容器,横向排列

hsizer_user = wx.BoxSizer(wx.HORIZONTAL)

hsizer_user.Add(self.label_user, proportion=0, flag=wx.ALL, border=5)

hsizer_user.Add(self.text_user, proportion=1, flag=wx.ALL, border=5)

hsizer_pwd = wx.BoxSizer(wx.HORIZONTAL)

hsizer_pwd.Add(self.label_pwd, proportion=0, flag=wx.ALL, border=5)

hsizer_pwd.Add(self.text_pwd, proportion=1, flag=wx.ALL, border=5)

hsizer_path = wx.BoxSizer(wx.HORIZONTAL)

hsizer_path.Add(self.label_path, proportion=0, flag=wx.ALL, border=5)

hsizer_path.Add(self.text_pathword, proportion=1, flag=wx.ALL, border=5)

hsizer_button = wx.BoxSizer(wx.HORIZONTAL)

hsizer_button.Add(self.bt_confirm, proportion=0, flag=wx.ALIGN_CENTER, border=5)

hsizer_button.Add(self.bt_cancel, proportion=0, flag=wx.ALIGN_CENTER, border=5)

hsizer_button.Add(self.bt_imf, proportion=0, flag=wx.ALIGN_CENTER, border=5)

hsizer_button.Add(self.bt_ml, proportion=0, flag=wx.ALIGN_CENTER, border=5)

接着添加容器,纵向排列

vsizer_all = wx.BoxSizer(wx.VERTICAL)

vsizer_all.Add(self.title, proportion=0, flag=wx.BOTTOM | wx.TOP | wx.ALIGN_CENTER,

border=15)

vsizer_all.Add(hsizer_user, proportion=0, flag=wx.EXPAND | wx.LEFT | wx.RIGHT, border=45)

vsizer_all.Add(hsizer_pwd, proportion=0, flag=wx.EXPAND | wx.LEFT | wx.RIGHT, border=45)

vsizer_all.Add(hsizer_path, proportion=0, flag=wx.EXPAND | wx.LEFT | wx.RIGHT, border=45)

vsizer_all.Add(hsizer_button, proportion=0, flag=wx.ALIGN_CENTER | wx.TOP, border=15)

panel.SetSizer(vsizer_all)

self.InitUI()

下一步是事件绑定

defInitUI(self):

""" 点击使用说明按钮,执行方法 """

defOnQuit1(self,e):

""" 输入注意事项 """

defOnclickSubmit(self,event):

""" 点击合同信息按钮,执行方法 """

defOnclickvisual(self,event):

""" 点击可视化按钮,执行方法 """

defOnclickCancel(self,event):

""" 点击清空按钮,执行方法 """

defOnclickimfor(self,event):

""" 点击球员名称按钮,执行方法 """

defOnclickteam(self,event):

""" 点击球队名称按钮,执行方法 """

defOnclickmql(self,event):

""" 点击得分榜按钮,执行方法 """

这里的事件处理不是很难,读者可以自己尝试创新,最后执行脚本

if__name__ =='__main__':

app = wx.App()# 初始化

frame = MyFrame(parent=None,id=-1)# 实例MyFrame类,并传递参数 

frame.Show()# 显示窗口

app.MainLoop()# 调用主循环方法

效果如图👇

  补充:插入背景图片

想要构造一个个性化系统,最不能缺的就是将界面背景换成自己想要的。这里我选择用一张老科的图片。

相信有的读者会觉得一个独立的单机的GUI软件会更适合自己,我也恰恰如此,因此,在设置背景图片中于之后的GUI需要进行打包,故需要将指定的二进制图片base64化,转换后存入py文件后以import为媒介才能打包。二进制代码转换如下:

importbase64

withopen("name.jpg","rb")asf:

base64_str = base64.b64encode(f.read())

withopen('%s.py'% picture_name.replace('.','_'),'w+')asf1:

f1.write(base64_str)

f1.close()

此时可以得到有base64编码的py文件,而后在代码中进行引用。由于打包不能打包图片,故这里稍微复杂的实现“引用指定图片的base64编码——创建图片——插入背景图片”功能!

最后在再稍加修饰将文本底色改为透明。编写这段代码的框架非常固定,所以我借鉴了大神们的代码,基本代码框架如下:

#这里需要在主事件中插入两句话

panel.Bind(wx.EVT_ERASE_BACKGROUND,self.OnEraseBack)

self.Bind(wx.EVT_ERASE_BACKGROUND, self.OnEraseBack)

#引用编码并创建图片

frombg_pngimportimgasbg

defpic(picPath,picName):

tmp = open(picPath,'wb')

tmp.write(base64.b64decode(picName))

tmp.close()

pic('bg.png',bg)

#插入图片(子事件 有缩进)

defOnEraseBack(self,event):

'''加入图片背景'''

try:

dc = event.GetDC()

ifnotdc:

dc = wx.ClientDC(self)

rect = self.GetUpdateRegion().GetBox()

dc.SetClippingRect(rect)

dc.Clear()

bmp = wx.Bitmap(nowpath+r'\bg.png')

dc.DrawBitmap(bmp,-500,-100)

except:

pass

#将文本底色改为透明

#第一步:将主事件中wx.StaticText全部换成TransparentStaticText

#第二步:重现StaticText控件

classTransparentStaticText(wx.StaticText):

def__init__(self, parent, id=wx.ID_ANY, label='', pos=wx.DefaultPosition, size=wx.DefaultSize,

style=wx.TRANSPARENT_WINDOW, name='TransparentStaticText')

:

wx.StaticText.__init__(self, parent, id, label, pos, size, style, name)

self.Bind(wx.EVT_PAINT, self.OnPaint)

self.Bind(wx.EVT_ERASE_BACKGROUND,lambdaevent:None)

self.Bind(wx.EVT_SIZE, self.OnSize)

defOnPaint(self, event):

bdc = wx.PaintDC(self)

dc = wx.GCDC(bdc)

font_face = self.GetFont()

font_color = self.GetForegroundColour()

dc.SetFont(font_face)

dc.SetTextForeground(font_color)

dc.DrawText(self.GetLabel(),0,0)

defOnSize(self, event):

self.Refresh()

event.Skip()

最终效果如图:

注意如果你想打包的话,需要代码中中引入下面三个模块:

importsix

importpackaging

importpackaging.version

importpackaging.specifiers

importpackaging.requirements

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,039评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,426评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,417评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,868评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,892评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,692评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,416评论 3 419
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,326评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,782评论 1 316
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,957评论 3 337
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,102评论 1 350
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,790评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,442评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,996评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,113评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,332评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,044评论 2 355

推荐阅读更多精彩内容