flask url_for() 将参数视为查询字符串

mic*_*y06 3 python flask

因为 'code' 是下面 MethodView 类 C 的 get 方法的参数,所以当我调用 url_for('c', code='O7A') 时,我希望得到的 url 是:/c/O7A

我看到的不是预期值,而是:/c/?code=O7A

from flask import Flask, url_for
from flask.views import MethodView

app = Flask(__name__)


class B(MethodView):

    def get(self):
        return 'ok'

    def post(self):
        print url_for('c', code='O7A')
        return 'ok'

app.add_url_rule('/b',
                 view_func=B.as_view('b'),
                 methods=['GET', 'POST'])

class C(MethodView):

    def get(self, code):
        return 'ok'

    def post(self):
        return 'ok'

app.add_url_rule('/c/<code>',
                 view_func=C.as_view('c'),
                 methods=['GET'])
app.add_url_rule('/c/',
                 view_func=C.as_view('c'),
                 methods=['POST'])

print app.url_map

if __name__ == "__main__":
    app.run()
Run Code Online (Sandbox Code Playgroud)

Mar*_*eth 5

你犯了一个错误,新版本的 Flask 实际上会抛出一个错误让你知道它。当我使用 Flask 10.1 运行此程序时,出现以下错误:

Traceback (most recent call last):
  File "flask_app.py", line 33, in <module>
    methods=['POST'])
  File "/home/mark/temp/test/local/lib/python2.7/site-packages/flask/app.py", line 62, in wrapper_func
    return f(self, *args, **kwargs)
  File "/home/mark/temp/test/local/lib/python2.7/site-packages/flask/app.py", line 984, in add_url_rule
    'existing endpoint function: %s' % endpoint)
AssertionError: View function mapping is overwriting an existing endpoint function: c
Run Code Online (Sandbox Code Playgroud)

问题在于端点,特别是您如何命名它们。当您这样做时url_for('c', code='O7A'),您提供的第一个参数是“端点”。基本上,它是一个唯一标识 URL 规则的字符串。

大多数情况下,您实际上并没有指定端点,因为 Flask 会神奇地为您指定;如果您不提供端点,Flask 将使用传递给view_func参数的函数的名称。在您的情况下,您正在使用MethodView帮助程序,因此实际上,端点是您传递给as_view. 因此,当你做..

app.add_url_rule('/c/<code>',
                 view_func=C.as_view('c'),
                 methods=['GET'])
Run Code Online (Sandbox Code Playgroud)

...你是说你想定义一个/c/<code>具有指定视图函数的路由,并且只允许使用 GET 方法对该路由进行请求。因为您没有指定端点,Flask 为其分配了一个端点( 的参数as_view) of c

接下来你有这个......

app.add_url_rule('/c/',
                 view_func=C.as_view('c'),
                 methods=['POST'])
Run Code Online (Sandbox Code Playgroud)

...这几乎完全相同。因此,它还尝试定义此路由以具有端点c. 这意味着新端点将覆盖旧端点!

要解决此问题,您应该选择不同的端点名称,如下所示...

app.add_url_rule('/c/<code>',
                 view_func=C.as_view('c_GET'),
                 methods=['GET'])
app.add_url_rule('/c/',
                 view_func=C.as_view('c_POST'),
                 methods=['POST'])
Run Code Online (Sandbox Code Playgroud)

然后,当你url_for打电话时,你可以做...

print url_for('c_GET', code='O7A')
Run Code Online (Sandbox Code Playgroud)