如何实现CSDN评论区粉丝幸运抽选功能:一场代码与运气的碰撞

文章目录

  • 前言
  • 抽选规则
  • 实现思路
  • 代码实现
    • 1.获取评论数据
    • 2.过滤符合抽选规则的评论者
    • 3.获取粉丝数据
    • 4.过滤符合抽选规则的粉丝
    • 5.增加公众号留言权重
    • 6.抽选粉丝
  • 完整的代码
  • 效果展示
  • 结语

前言

为了回馈粉丝们一直以来的的关注和支持,我近期开启了赠书活动,活动期间会在每一期的文章评论中随机抽选几位粉丝赠送本期书籍。为了确保绝对的公平,我借鉴了@东离与糖宝的想法,编写了一个评论区抽选粉丝程序。秉行公平、公正、公开原则,本文会将抽选粉丝程序的思路以及代码的实现展示给大家。

抽选规则

抽选评论区中的粉丝要符合以下条件:

  1. 关注了我的博客,即成为我的粉丝。
  2. 评论一次权重+1,最多3次,评论内容见当期参与方式。

同时,关注公众号并留言权重+3,留言内容见当期参与方式。

活动见赠书专栏

实现思路

  1. 首先需要获取该博客文章的所有评论者。
  2. 筛选符合条件的评论者。
    • 是否为我的粉丝。
    • 是否评论当期要求内容(可选)。
  3. 对于既是当期内容评论者又是公众号的粉丝,给予更高的权重值。
  4. 在符合条件的粉丝中随机抽选幸运读者。

代码实现

1.获取评论数据

虽然CSDN不提供直接获取文章评论者的接口,但这难不倒一个开发人员。我们可以在文章所在页面开启调试模式拿到这个接口:https://blog.csdn.net/phoenix/web/v1/comment/list/132678231?page=1&size=20,参数清晰明了,响应格式如下:

{"code":200,"message":"success","traceId":"b3e6eb62-396d-4289-a937-710767e68680","data":{"count":86,"pageCount":83,"floorCount":83,"foldCount":1,"list":[{"info":{"commentId":28613264,"articleId":132678231,"parentId":0,"postTime":"2023-09-09 16:16:10","content":"这里是评论内容","userName":"qq_28314431","digg":0,"diggArr":[],"loginUserDigg":false,"parentUserName":"qq_28314431","parentNickName":"叫我二蛋","avatar":"https://profile-avatar.csdnimg.cn/55e249ffc0b54ab287ecde8615968cca_qq_28314431.jpg!1","nickName":"这里是用户昵称","dateFormat":"前天 16:16","years":8,"vip":true,"vipIcon":"https://img-home.csdnimg.cn/images/20210121052537.png","vipUrl":"https://mall.csdn.net/vip","companyBlog":false,"companyBlogIcon":"","flag":false,"flagIcon":"","levelIcon":"https://csdnimg.cn/identity/blog6.png","commentFromTypeResult":{"index":1,"key":"pc","title":"PC"},"isTop":true,"isBlack":false,"region":"IP:陕西省","orderNo":"","redEnvelopeInfo":null},"sub":[],"pointCommentId":null}]}
}

这里基于Python直接发起POST请求获取评论数据,代码如下:

import requests
commentUrl = "https://blog.csdn.net/phoenix/web/v1/comment/list/132678231?page=1&size=20"
# 发送Post请求
headers = {"User-Agent": ""}
response = requests.get(commentUrl,headers=headers)
# 如果请求成功,接收的响应会是一个Response对象
if response.status_code == 200:# 使用json()方法将响应内容解析为JSONdata = response.json()print(data)
else:print("请求失败,状态码:", response.status_code)

2.过滤符合抽选规则的评论者

获取的评论数据中我们只需要评论内容和评论者昵称这两个数据,及字段content、nickName,接下来就可以根据条件筛选评论者,并且评论一次权重+1,代码如下:

commentList = data["data"]["list"]
commentUsers={}for item in commentList:nickName = item["info"]["nickName"]content = item["info"]["content"]if needcomment in content:commentCount=commentUsers.get(nickName)if commentCount is None:commentUsers[nickName]=1;else:commentUsers[nickName]=commentCount+1;

commentUsers最终会输出一个字典类型,key为昵称,value为权重值,格式如:{'EmotionFlying': 1, '小 王': 1}

符合条件的评论者筛选出来后要确定是否为我的粉丝,不是的话将其从commentUsers移除。

3.获取粉丝数据

同样,还是在页面中拿到“我的粉丝”接口:https://mp-action.csdn.net/interact/wrapper/pc/fans/v1/api/getFansOffsetList?pageSize=500&username=qq_28314431&fanId=,该接口中的fanId可以理解为‘page’页码,每次请求成功后会返回下一个fanId。获取粉丝数据代码如下:

import requests
fanUrl = "https://mp-action.csdn.net/interact/wrapper/pc/fans/v1/api/getFansOffsetList?pageSize=500&username=qq_28314431&fanId="
# 发送Post请求
headers = {"User-Agent": ""}
response = requests.get(fanUrl,headers=headers)
# 如果请求成功,接收的响应会是一个Response对象
if response.status_code == 200:# 使用json()方法将响应内容解析为JSONdata = response.json()print(data)
else:print("请求失败,状态码:", response.status_code)

4.过滤符合抽选规则的粉丝

在获取到粉丝后,将commentUsers中不在粉丝列表的数据移除就好,代码如下:

fansNickname = list(set([d['nickname'] for d in fansList]) & set(list(commentUsers.keys())))fans = {nickname: commentUsers.pop(nickname) for nickname in fansNickname}

fans 就是符合抽选规则的粉丝。

5.增加公众号留言权重

对于公众号留言的粉丝,需要手动将他们的昵称作为一个参数传入程序中,直接遍历fans,增加其权重,代码如下

gzhfans=["粉丝1","粉丝2"]
gzhfans = [fan for fan in gzhfans if fan in fans and isinstance(fans[fan], int)]
fans = fans | {fan: fans[fan] + 3 for fan in gzhfans}

fans 就是最终抽选粉丝的集合数据。

6.抽选粉丝

最后,就可以基于fans随机抽选粉丝了,这里用python 自带的random.choices随机方法,代码如下:

random.choices(list(fans.keys()), weights=list(fans.values()))[0]

完整的代码

由于接口中存在分页问题,同时为了提高代码的可读性和可用性,我对剩余的代码进行了完善,以下是完善后的代码:

import sys
import random
import requests'''
介绍- 该脚本用来随机抽选评论区粉丝
使用方法: - python3 selectFans.py- blogID: 博客ID- bloggerID: 博主ID- selectCount:抽选粉丝个数- needcomment: 符合的评论内容- gzhfans: 公众号留言的粉丝,添加权重
依赖:- python3- random- requests
'''# 评论数据接口
commentUrlT = "https://blog.csdn.net/phoenix/web/v1/comment/list/{}?page={}&size={}"
# 每次获取接口数据的数量,csdn的上限
pageSize = 500
# 粉丝数据接口
fansUrlT = "https://mp-action.csdn.net/interact/wrapper/pc/fans/v1/api/getFansOffsetList?pageSize={}&username={}&fanId={}"# 抽选粉丝
def select_fans():try:fans = satisfied_fans()assert fans, "没有符合抽选规则的粉丝"add_weight(fans, gzhfans);print("——————开始抽选粉丝——————")for i in range(1, selectCount + 1):luckfans = weighted_random_choice(list(fans.keys()), list(fans.values()))# 移除该粉丝fans.pop(luckfans)print("恭喜第{}为幸运粉丝:{}".format(i, luckfans))except Exception as e:print(e)# 添加公众号留言粉丝的权重
def add_weight(fans, gzhfans):print("——————添加公众号留言粉丝的权重——————")if gzhfans:gzhfans = [fan for fan in gzhfans if fan in fans and isinstance(fans[fan], int)]fans = fans | {fan: fans[fan] + 3 for fan in gzhfans}print("权重增加后的粉丝:{}".format(fans))else:print("没有公众号留言的粉丝")# 符合抽选规则的粉丝
def satisfied_fans():satisfiedFans = {}commentUsers = get_comment_users()assert commentUsers, "没有符合条件的评论者"fanId = ""one = True;# 如果是第一次或者 fanId 有值 ,并且评论者未被移除完,继续筛选。while one or (fanId is not None and commentUsers):one = FalsefanId, fans = get_satisfied_fans(commentUsers, fanId)satisfiedFans = satisfiedFans | fansprint("符合抽选规则的粉丝:{}".format(satisfiedFans))return satisfiedFans# 获取符合抽选规则的粉丝
def get_satisfied_fans(commentUsers, fanId):fanId, fansList = get_fans(fanId)# 在粉丝列表中的评论者fansNickname = list(set([d['nickname'] for d in fansList]) & set(list(commentUsers.keys())))# 移除并返回commentUsers的粉丝return fanId, {nickname: commentUsers.pop(nickname) for nickname in fansNickname}# 获取符合条件的评论者
def get_comment_users():commentUsers = {}for item in get_comments():# 解析评论者IDuserName = item["info"]["userName"]if userName == bloggerID:continue# 解析评论者nickName = item["info"]["nickName"]# 解析评论内容content = item["info"]["content"]# 判断评论内容是否符合要求if len(needcomment) == 0 or needcomment in content:commentCount = commentUsers.get(nickName)# 组装评论者及其权重if commentCount is None:commentUsers[nickName] = 1else:if commentUsers[nickName] < 3:commentUsers[nickName] = commentCount + 1print("符合条件的评论者{}".format(commentUsers))return commentUsers# 获取到所有的评论
def get_comments():commentUrl = commentUrlT.format(1, pageSize)data = request(commentUrl)# 解析到总页数total_pages = data["data"]["pageCount"]# 解析到评论数据commentList = data["data"]["list"]assert len(commentList) > 0, "该博客没有评论数据"for page in range(2, total_pages + 1):commentUrl = commentUrlT.format(page, pageSize)_data = request(commentUrl)_commentList = _data["data"]["list"]# 将所有评论数据合并commentList = commentList + _commentListreturn commentList# 获取到所有的粉丝
def get_fans(fanId):fansUrl = fansUrlT.format(fanId)data = request(fansUrl)# 解析到下一fanIdfanId = data["data"]["fanId"]# 解析到粉丝数据fansList = data["data"]["list"]return fanId, fansList# 调用接口获取数据
def request(url):headers = {"User-Agent": ""}response = requests.get(url, headers=headers)# 如果请求成功,接收的响应会是一个Response对象if response.status_code == 200:# 使用json()方法将响应内容解析为JSONreturn response.json()else:print("请求失败,状态码:", response.status_code)def weighted_random_choice(choices, weights):return random.choices(choices, weights=weights)[0]if __name__ == '__main__':# 博客IDblogID = input("博客ID:")# 博主IDbloggerID = input("博主ID:")# 抽选粉丝个数selectCount = int(input("抽选粉丝个数:"))# 符合的评论内容needcomment = input("符合的评论内容(不填写则不对评论内容有要求):")# 公众号留言的粉丝gzhfans = input("公众号留言的粉丝,可以为空,以','分隔:").split()# 评论数据接口commentUrlT = commentUrlT.format(blogID, "{}", "{}")# 粉丝数据接口fansUrlT = fansUrlT.format(pageSize, bloggerID, "{}")select_fans()

效果展示

执行代码中的使用示例后结果如下图,如果粉丝过于多,可能需要等待数秒,因为粉丝接口一次最多只能读500条,希望官方看到后可以为我们开个口子😁。

在这里插入图片描述

结语

为了方便大家的使用,我已将程序打包成可执行文件,支持Windows和Mac系统,需要的可以联系博主获取。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/76971.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【网络编程·数据链路层】MAC帧/以太网协议/ARP协议/RARP协议

需要云服务器等云产品来学习Linux的同学可以移步/-->腾讯云<--/-->阿里云<--/-->华为云<--/官网&#xff0c;轻量型云服务器低至112元/年&#xff0c;新用户首次下单享超低折扣。 目录 一、MAC帧 1、IP地址和MAC地址的区别 2、MAC帧协议 3、MTU对IP协议的…

腾讯云2核4G服务器5M带宽 218元一年 优惠价格明细表

腾讯云2核4G服务器5M带宽可以选择轻量应用服务器或云服务器ECS&#xff0c;轻量2核4G5M带宽服务器218元一年&#xff1a; 腾讯云2核4G服务器5M带宽收费 腾讯云2核4G服务器可以选择轻量应用服务器或者ECS云服务器&#xff0c;云服务器ECS是专业级云服务器&#xff0c;大多数使用…

【K8S系列】深入解析k8s网络插件—Canal

序言 做一件事并不难&#xff0c;难的是在于坚持。坚持一下也不难&#xff0c;难的是坚持到底。 文章标记颜色说明&#xff1a; 黄色&#xff1a;重要标题红色&#xff1a;用来标记结论绿色&#xff1a;用来标记论点蓝色&#xff1a;用来标记论点 在现代容器化应用程序的世界中…

docker-compose安装redis

基于docker-compose快速安装redis 目录 一、目录结构 1、docker-compose.yml 2、redis.conf 二、连接使用 一、目录结构 1、docker-compose.yml version: 3 services:redis:image: registry.cn-hangzhou.aliyuncs.com/zhengqing/redis:6.0.8 # 镜像red…

【性能测试】Jmeter —— jmeter计数器

jmeter计数器 如果需要引用的数据量较大&#xff0c;且要求不能重复或者需要递增&#xff0c;那么可以使用计数器来实现 如&#xff1a;新增功能&#xff0c;要求名称不能重复 1&#xff0c;新增计数器 计数器&#xff1a;允许用户创建一个在线程组之内都可以被引用的计数器…

van-list 下拉刷新 触底分页 触底分页事件只加载一次

我是 头部是筛选的条件&#xff0c;&#xff0c;更换不同的状态&#xff0c;显示不同的列表数据&#xff0c;比如 审批中数据是 对的&#xff0c;触底分页也是对的&#xff0c;如果我切换一个状态的话&#xff0c;总共是 15条数据&#xff0c;但是 切换了状态只显示第一页的数据…

8个免费的AI和LLM游乐场

推荐&#xff1a;使用 NSDT场景编辑器 快速搭建3D应用场景 在本文中&#xff0c;我们的目标是通过引入八个用户友好的平台来弥合这一差距&#xff0c;这些平台使任何人都可以免费测试和比较开源AI模型。此外&#xff0c;它们还提供多种更新型号&#xff0c;确保您及时了解最新进…

无涯教程-JavaScript - SLN函数

描述 SLN函数返回资产在一个期间内的直线折旧。 语法 SLN (cost, salvage, life)争论 Argument描述Required/OptionalCostThe initial cost of the asset.RequiredSalvage The value at the end of the depreciation (sometimes called the salvage value of the asset).Re…

win11本地连接没了怎么办

很多用户在使用win11系统时发现自己的网络连接没有了&#xff0c;遇到这种情况的话&#xff0c;我们应该怎么处理呢&#xff1f;我们可以尝试打开网络图标&#xff0c;下面就是小编整理出的教程&#xff0c;大家一起看看吧。 win11本地连接没了怎么办 方法一&#xff1a; 1、…

数据分享|R语言武汉流动人口趋势预测:灰色模型GM(1,1)、ARIMA时间序列、logistic逻辑回归模型...

全文链接&#xff1a;http://tecdat.cn/?p32496 人口流动与迁移&#xff0c;作为人类产生以来就存在的一种社会现象&#xff0c;伴随着人类文明的不断进步从未间断&#xff08;点击文末“阅读原文”获取完整代码数据&#xff09;。 相关视频 人力资源是社会文明进步、人民富裕…

Redis I/O多路复用机制

一、基础回顾 1.1 多路复用要解决什么问题 并发多客户端连接场景&#xff0c;在多路复用之前最简单和典型的方案就是同步阻塞网络IO模型。 这种模式的特点就是用一个进程来处理一个网络连接(一个用户请求),比如一段典型的示例代码如下。 直接调用 recv 函数从一个 socket 上…

去除pdf/word的水印艺术字

对于pdf中的水印如果无法去除水印&#xff0c;则先另存为word&#xff0c;然后再按下面办法处理即可&#xff1a; 查看宏&#xff0c;创建&#xff1a;删除艺术字 添加内容&#xff1a; Sub 删除艺术字()Dim sh As ShapeFor Each sh In ActiveDocument.ShapesIf sh.Type msoT…

HTML的段落中怎么样显示出标签要使用的尖括号<>?

很简单&#xff1a; 符号 < 用 < 替代&#xff1b; 符号 > 用 > 替代。 示例代码如下&#xff1a; <!DOCTYPE html> <html> <head><meta charset"UTF-8"><title>HTML中怎样打出尖括号</title> </head> <b…

Android Aidl跨进程通讯(三)--进阶使用

学更好的别人&#xff0c; 做更好的自己。 ——《微卡智享》 本文长度为2478字&#xff0c;预计阅读6分钟 前言 Android的AIDL使用和异常报错都已经介绍过了&#xff0c;今天这篇还是在原来的Demo基础上加入几个AIDL的进阶使用方法。 】 AIDL进阶使用 微卡智享 in,out,inout的使…

2022年全国研究生数学建模竞赛华为杯A题移动场景超分辨定位问题求解全过程文档及程序

2022年全国研究生数学建模竞赛华为杯 A题 移动场景超分辨定位问题 原题再现&#xff1a; 在日常家庭生活中&#xff0c;人们可能需要花费大量时间去寻找随意摆放在家中某些角落里的小物品。但如果给某些重要物品贴上电路标签&#xff0c;再利用诸如扫地机器人的全屋覆盖能力&…

docker入门

弱小和无知不是生存的障碍&#xff0c;傲慢才是。 Docker 学习 Docker概述 Docker为什么出现&#xff1f; 一款产品&#xff1a; 开发–上线 两套环境&#xff01;应用环境&#xff0c;应用配置&#xff01; 开发 — 运维。 问题&#xff1a;我在我的电脑上可以允许&#…

Vuex -访问(modules)模块中的 state mutations actions getters

文章目录 四大核心复习一、获取模块内的state数据1.目标&#xff1a;2.使用模块中的数据3.代码示例 二、获取模块内的getters数据1.目标&#xff1a;2.语法&#xff1a;3.代码演示 三、获取模块内的mutations方法1.目标&#xff1a;2.注意&#xff1a;3.调用方式&#xff1a;4.…

【实例项目:基于多设计模式下的日志系统(同步异步)】

本项目涉及的到所有源码见以下链接&#xff1a; https://gitee.com/ace-zhe/wz_log 一、项目简介 1.日志的概念&#xff08;白话版&#xff09; 日志类似于日记&#xff0c;通常是指对完成某件事情的过程中状态等的记录&#xff0c;而计算机中的日志是指日志数据&#xff0c…

考研英语笔记:程序员是否勤奋就看他的英语好不好

一位大佬朋友圈写道&#xff1a;看程序员是否勤奋就看他的英语好不好&#xff0c;智商高不高就看他算法好不好。 这句话我当时看到了很触动&#xff0c;默默的记在了心底。 对我来说&#xff0c;算法就免了&#xff0c;但学英语我一直在坚持。我不敢说我是优秀的程序员&#xf…

​消费盲返模式:一种让消费者和商家都受益的新型消费返利模式

您是否想过&#xff0c;如果您的消费能够带来意想不到的回报&#xff0c;您会不会更愿意购买商品或服务呢&#xff1f;您是否想过&#xff0c;如果您的商品或服务能够吸引更多的消费者&#xff0c;并让他们成为您的忠实客户&#xff0c;您会不会更有动力经营您的业务呢&#xf…