python-3.x - 从随机生成的数字中添加或减去数字
问题描述
我正在尝试在我的程序中实现一个功能,用户可以使用特定的双面模具添加或减去随机生成的数字。我的代码的基础发布在这里:
import discord
import random
DND_1d20 = range(1, 21)
# Roll d20
if message.content == ";roll 1d20":
response = random.choice(DND_1d20)
response_str = "You rolled {0}".format(response)
if response_str == "You rolled 20":
await message.channel.send("**Critical Hit!**\n You rolled 20")
if response_str == "You rolled 1":
await message.channel.send("**Critical Fail!**\n You rolled 1")
我希望用户能够指定掷骰子 ";1d20" 但也能够从生成的骰子中添加 ";1d20+(x)" 或减去 ";1d20-(x)" 任何数字 (x)卷。逻辑看起来像这样
-user ";1d20+2" 假设生成的随机数是 6。由于用户想在我们生成的随机数上加 2,结果将是 8。
-bot“你掷出 8”
# Roll d20
if message.content == ";roll 1d20":
response = random.choice(DND_1d20)
response_str = "You rolled {0}".format(response)
if response_str == "You rolled 20":
await message.channel.send("**Critical Hit!**\n You rolled 20")
if response_str == "You rolled 1":
await message.channel.send("**Critical Fail!**\n You rolled 1")
else:
if message.content == "-":
我该怎么做呢?我真的很困惑从哪里开始。我不认为上面的代码是正确的,因为消息必须完全是一个“-”。另外,我将如何合并值 (x),因为它可能是大量数字,或者来自用户输入的 +/- 符号?
任何帮助表示赞赏!
解决方案
这是一个更高级的解决方案,它使用一个库lark
来定义这些骰子表达式的语法(摘自this question),将这些表达式解析为语法树,然后评估这些树。dice_grammar.py
创建一个以此字符串命名的文件:
grammar="""
start: _expr
_expr: add
| subtract
| roll
| NUMBER
add: _expr "+" _expr
subtract: _expr "-" _expr
roll: [NUMBER] ("d"|"D") (NUMBER|PERCENT)
NUMBER: ("0".."9")+
PERCENT: "%"
%ignore " "
"""
如果您不熟悉这样的语法,请不要惊慌。所有这一切都表明我们可以掷骰子、加法和减法。然后我们可以dice_transformer.py
使用解析器将生成的树:
from lark import Transformer, v_args
from random import randint
class DiceTransformer(Transformer):
PERCENT = lambda self, percent: 100
NUMBER = int
def __init__(self):
super().__init__(visit_tokens=True)
@v_args(inline=True)
def start(self, expr):
return expr
@v_args(inline=True)
def add(self, left, right):
return left + right
@v_args(inline=True)
def subtract(self, left, right):
return left - right
@v_args(inline=True)
def roll(self, qty, size):
qty = qty or 1
return sum(randint(1, size) for _ in range(qty))
和 adice_bot.py
使用这些来评估用户的骰子表达式:
from discord.ext import commands
from lark import Lark
from lark.exceptions import LarkError
from dice_grammar import grammar
from dice_transformer import DiceTransformer
bot = commands.Bot(";")
parser = Lark(grammar, maybe_placeholders=True)
transformer = DiceTransformer()
@bot.command()
async def roll(ctx, *, expression):
try:
tree = parser.parse(expression)
except LarkError:
await ctx.send("Bad Expression")
return
print(tree.pretty()) # Log the roll
result = transformer.transform(tree)
await ctx.send(f"You rolled: {result}")
bot.run("token")
这使我们可以要求计算更复杂的卷,例如
;roll 2d6 +7 + d% - 3d4
根据 Erez 在评论中的建议,我将答案更改为 use lark.Transformer
。您应该能够在此答案的编辑历史记录中看到我的原始代码。
推荐阅读
- spring-boot - 如何在 Spring Webflux 中获取 FilePart 的大小
- android - 如何在android中显示视频、获取和管理流视频的声道?
- spring - Spring集成收件人列表路由器问题
- angularjs - AngularJS $scope.Itemlist 未定义
- django - 无法再次在 Django 应用程序中正确提供静态文件
- jquery - 移动版切换菜单:vue with pug
- angular - *ngIf 根据数据库值显示项目
- awk - AWK:使用内联 if-then-else 的稳健方式
- apostrophe-cms - 在 Apostrophe CMS 中使用模式创建动态表单
- powershell - 获取 vm PowerCLI 的快照信息