B站小UP主抽奖简易解决方案
B站小UP主抽奖简易解决方案
大家好,我是b站UP主 一团雷丘,这是对我2020年12月30日视频《轻松解决小UP主抽奖难题》的项目分析,视频地址:https://www.bilibili.com/video/BV12A411p76j
B站有关API如何查找与使用
在开始制作该项目的时候,我的思路是使用requests爬虫进行爬取,再使用beautifulsoup进行分析,问题在于b站的前端是动态渲染的,不存在页面切换,不好使用爬虫(除非使用selenium,但是这样就不适合作为后端了)。在苦恼的时候我意外发现有总结b站API的文章,而在自己查看b站网站结构后我发现b站的前端信息几乎都是API获得的,例如粉丝列表,在点击时就有了如图所示的API访问信息。
主要功能实现
该项目主要由python编写,因此选择requests库发起API请求,json库处理json数据,flask作为网页后端,服务端方面使用Nginx反向代理从80端口代理到内网5000端口。
1.BV号反查AV号
def BV2AV(bv_number):
url = f"https://api.bilibili.com/x/web-interface/view?bvid={bv_number}"
req = requests.get(url)
return json.loads(req.text)["data"]["aid"]
2.UID查用户昵称
def mid2username(mid):
url = f"https://api.bilibili.com/x/space/acc/info?mid={mid}&jsonp=jsonp"
req = requests.get(url)
name = json.loads(req.text)["data"]["name"]
return name
3.获得全部评论列表
def video_floor(bv_number):
av_number = BV2AV(bv_number)
page = 1
url = f"https://api.bilibili.com/x/v2/reply?&type=1&pn={page}&oid={av_number}"
req = requests.get(url)
json_file = json.loads(req.text)
all_comment = []
# 总评论条数
comment_number = json_file["data"]["page"]["count"]
# 计算评论页数
comment_page = comment_number // 20 + 1
for page in range(1, comment_page + 1):
url = f"https://api.bilibili.com/x/v2/reply?&type=1&pn={page}&oid={av_number}"
req = requests.get(url)
json_file = json.loads(req.text)
comments = json_file["data"]["replies"]
all_comment = all_comment + comments
return all_comment
4.检查UID防止重复评论
def check_mid(all_comment):
MID_list = []
for each_comment in all_comment:
mid_tmp = each_comment["member"]["mid"]
if(MID_list.count(mid_tmp) == 0):
MID_list.append(mid_tmp)
return MID_list
5.检查某人是否关注了UP主(该功能尚未完成)
该功能没有完成的原因是B站并不允许无限制的访问关注列表的API,只能访问前五页,在访问了几次之后更是出现了封禁IP的行为,后续看看b站是否有其它可以代替的API,否则该功能只能是半成品。
def check_subscibe(mid, up_mid):
print(f"正在检查{mid}是否关注了{up_mid}")
subscribe_list = []
flag = False
page = 1
url = f"https://api.bilibili.com/x/relation/followings?vmid={mid}&pn={page}&ps=20"
req = requests.get(url)
subscribe_number = json.loads(req.text)["data"]["total"]
page_number = subscribe_number // 50 + 1
for page in range(1, min(page_number + 1, 5)):
url = f"https://api.bilibili.com/x/relation/followings?vmid={mid}&pn={page}&ps=50"
req = requests.get(url)
json_tmp = json.loads(req.text)
if(json_tmp["data"]["list"] != None):
subscribe_list_tmp = json_tmp["data"]["list"]
subscribe_list += subscribe_list_tmp
for each_subscribe in subscribe_list:
if(each_subscribe["mid"] == up_mid):
flag = True
return flag
return flag
项目整体情况
想查看整个项目请移步github,
项目地址:https://github.com/Tessgraymane/bilibili_lottery/