Flask是一個基於Python開發而且依賴jinja2模板和Werkzeug WSGI服務的一個微型框架,對於Werkzeug本質是Socket服務端,其用於接收http請求並對請求進行預處理,而後觸發Flask框架,開發人員基於Flask框架提供的功能對請求進行相應的處理,並返回給用戶,若是要返回給用戶複雜的內容時,須要藉助jinja2模板來實現對模板的處理,即:將模板和數據進行渲染,將渲染後的字符串返回給用戶瀏覽器。html
「微」(micro) 並不表示你須要把整個 Web 應用塞進單個 Python 文件(雖然確實能夠 ),也不意味着 Flask 在功能上有所欠缺。微框架中的「微」意味着 Flask 旨在保持核心簡單而易於擴展。Flask 不會替你作出太多決策——好比使用何種數據庫。而那些 Flask 所選擇的——好比使用何種模板引擎——則很容易替換。除此以外的一切都由可由你掌握。如此,Flask 能夠與您珠聯璧合。前端
默認狀況下,Flask 不包含數據庫抽象層、表單驗證,或是其它任何已有多種庫能夠勝任的功能。然而,Flask 支持用擴展來給應用添加這些功能,如同是 Flask 自己實現的同樣。衆多的擴展提供了數據庫集成、表單驗證、上傳處理、各類各樣的開放認證技術等功能。Flask 也許是「微小」的,但它已準備好在需求繁雜的生產環境中投入使用。python
pip3 install flask
#Flask依賴一個實現wsgi協議的模塊:werkzeug from werkzeug.wrappers import Request, Response @Request.application def hello(request): return Response('Hello World!') if __name__ == '__main__': from werkzeug.serving import run_simple run_simple('localhost', 4000, hello)
flask依賴wsgi,實現wsgi模塊:wsgiref,werkzeug,uwsgimysql
Django:無socket,依賴第三方模塊wsgi,中間件,路由系統(CBV,FBV),視圖函數,ORM。cookie,session,Admin,Form,緩存,信號,序列化。。
Flask:無socket,中間件(須要擴展),路由系統,視圖(CBV)、第三方模塊(依賴jinja2),cookie,session弱爆了
from flask import Flask app = Flask(__name__) @app.route('/') def hello_world(): return 'Hello World!' if __name__ == '__main__': app.run()
app = Flask(__name__) # 這是實例化一個Flask對象,最基本的寫法 # 可是Flask中還有其餘參數,如下是可填的參數,及其默認值 def __init__(self, import_name, static_path=None, static_url_path=None, static_folder='static', template_folder='templates', instance_path=None, instance_relative_config=False, root_path=None):
template_folder:模板所在文件夾的名字正則表達式
root_path:能夠不用填,會自動找到,當前執行文件,所在目錄地址redis
在return render_template時會將上面兩個進行拼接,找到對應的模板地址sql
static_folder:靜態文件所在文件的名字,默認是static,能夠不用填數據庫
static_url_path:靜態文件的地址前綴,寫成什麼,訪問靜態文件時,就要在前面加上這個django
app = Flask(__name__,template_folder='templates',static_url_path='/xxxxxx')
如:在根目錄下建立目錄,templates和static,則return render_template時,能夠找到裏面的模板頁面;如在static文件夾裏存放11.png,在引用該圖片時,靜態文件地址爲:/xxxxxx/11.png編程
instance_path和instance_relative_config是配合來用的、
這兩個參數是用來找配置文件的,當用app.config.from_pyfile('settings.py')這種方式導入配置文件的時候會用到
instance_relative_config:默認爲False,當設置爲True時,from_pyfile會從instance_path指定的地址下查找文件。
instsnce_path:指定from_pyfile查詢文件的路徑,不設置時,默認尋找和app.run()的執行文件同級目錄下的instance文件夾;若是配置了instance_path(注意須要是絕對路徑),就會從指定的地址下里面的文件
#方式一 @app.route('/index.html',methods=['GET','POST'],endpoint='index') def index(): return 'Index' #方式二 def index(): return "Index" self.add_url_rule(rule='/index.html', endpoint="index", view_func=index, methods=["GET","POST"]) #endpoint是別名 or app.add_url_rule(rule='/index.html', endpoint="index", view_func=index, methods=["GET","POST"]) app.view_functions['index'] = index
添加路由關係的本質:將url和視圖函數封裝成一個Rule對象,添加到Flask的url_map字段中
from flask import Flask,render_template,request,redirect,session app = Flask(__name__) app.secret_key = "sdsfdsgdfgdfgfh" # 設置session時,必需要加鹽,不然報錯 def wrapper(func): def inner(*args,**kwargs): if not session.get("user_info"): return redirect("/login") ret = func(*args,**kwargs) return ret return inner @app.route("/login",methods=["GET","POST"]) # 指定該路由可接收的請求方式,默認爲GET def login(): if request.method=="GET": return render_template("login.html") else: # print(request.values) #這個裏面什麼都有,至關於body username = request.form.get("username") password = request.form.get("password") if username=="haiyan" and password=="123": session["user_info"] = username # session.pop("user_info") #刪除session return redirect("/index") else: # return render_template("login.html",**{"msg":"用戶名或密碼錯誤"}) return render_template("login.html",msg="用戶名或者密碼錯誤") @app.route("/index",methods=["GET","POST"]) @wrapper #本身定義裝飾器時,必須放在路由的裝飾器下面 def index(): # if not session.get("user_info"): # return redirect("/login") return render_template("index.html") if __name__ == '__main__': app.run(debug=True)
debug = True 是指進入調試模式,服務器會在 咱們的代碼修改後, 自動從新載入,有錯誤的話會提醒,每次修改代碼後就不須要再手動重啓
- request - request.form #POST請求的數據 - request.args #GET請求的數據,不是徹底意義上的字典,經過.to_dict能夠轉換成字典 - request.querystring #GET請求,bytes形式的 - response - return render_tempalte() - return redirect() - return "" v = make_response(返回值) #能夠把返回的值包在了這個函數裏面,而後再經過.set_cookie綁定cookie等 - session - 存在瀏覽器上,而且是加密的 - 依賴於:secret_key
GET請求:
URL爲: http://127.0.0.1:5000/login?name=%27%E8%83%A1%E5%86%B2%27&nid=2
from urllib.parse import urlencode,quote,unquote def login(): if request.method == 'GET': s1 = request.args s2 = request.args.to_dict() s3 = urlencode(s1) s4 = urlencode(s2) s5 = unquote(s3) s6 = unquote(s4) s7 = quote("胡衝") print('s1',s1) print('s2',s2) print('s3',s3) print('s4',s4) print('s5',s5) print('s6',s6) print('s7',s7) return render_template('login.html') #############結果以下#################### s1 ImmutableMultiDict([('name', "'胡衝'"), ('nid', '2')]) s2 {'name': "'胡衝'", 'nid': '2'} s3 name=%27%E8%83%A1%E5%86%B2%27&nid=2 s4 name=%27%E8%83%A1%E5%86%B2%27&nid=2 s5 name='胡衝'&nid=2 s6 name='胡衝'&nid=2 s7 %E8%83%A1%E5%86%B2
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
|
flask中的配置文件是一個flask.config.Config對象(繼承字典),默認配置爲:
{
'DEBUG'
: get_debug_flag(default
=
False
), 是否開啓Debug模式
'TESTING'
:
False
, 是否開啓測試模式
'PROPAGATE_EXCEPTIONS'
:
None
,
'PRESERVE_CONTEXT_ON_EXCEPTION'
:
None
,
'SECRET_KEY'
:
None
,
'PERMANENT_SESSION_LIFETIME'
: timedelta(days
=
31
),
'USE_X_SENDFILE'
:
False
,
'LOGGER_NAME'
:
None
,
'LOGGER_HANDLER_POLICY'
:
'always'
,
'SERVER_NAME'
:
None
,
'APPLICATION_ROOT'
:
None
,
'SESSION_COOKIE_NAME'
:
'session'
,
'SESSION_COOKIE_DOMAIN'
:
None
,
'SESSION_COOKIE_PATH'
:
None
,
'SESSION_COOKIE_HTTPONLY'
:
True
,
'SESSION_COOKIE_SECURE'
:
False
,
'SESSION_REFRESH_EACH_REQUEST'
:
True
,
'MAX_CONTENT_LENGTH'
:
None
,
'SEND_FILE_MAX_AGE_DEFAULT'
: timedelta(hours
=
12
),
'TRAP_BAD_REQUEST_ERRORS'
:
False
,
'TRAP_HTTP_EXCEPTIONS'
:
False
,
'EXPLAIN_TEMPLATE_LOADING'
:
False
,
'PREFERRED_URL_SCHEME'
:
'http'
,
'JSON_AS_ASCII'
:
True
,
'JSON_SORT_KEYS'
:
True
,
'JSONIFY_PRETTYPRINT_REGULAR'
:
True
,
'JSONIFY_MIMETYPE'
:
'application/json'
,
'TEMPLATES_AUTO_RELOAD'
:
None
,
}
方式一:
app.config[
'DEBUG'
]
=
True
PS: 因爲Config對象本質上是字典,因此還可使用app.config.update(...)
方式二:
app.config.from_pyfile(
"python文件名稱"
)
如:
settings.py
DEBUG
=
True
app.config.from_pyfile(
"settings.py"
)
app.config.from_envvar(
"環境變量名稱"
)
環境變量的值爲python文件名稱名稱,內部調用from_pyfile方法
app.config.from_json(
"json文件名稱"
)
JSON文件名稱,必須是json格式,由於內部會執行json.loads
app.config.from_mapping({
'DEBUG'
:
True
})
字典格式
app.config.from_object(
"python類或類的路徑"
)
app.config.from_object(
'pro_flask.settings.TestingConfig'
)
settings.py
class
Config(
object
):
DEBUG
=
False
TESTING
=
False
DATABASE_URI
=
'sqlite://:memory:'
class
ProductionConfig(Config):
DATABASE_URI
=
'mysql://user@localhost/foo'
class
DevelopmentConfig(Config):
DEBUG
=
True
class
TestingConfig(Config):
TESTING
=
True
PS: 從sys.path中已經存在路徑開始寫
PS: settings.py文件默認路徑要放在程序root_path目錄,若是instance_relative_config爲
True
,則就是instance_path目錄
|
@app.route('/user/<username>') #經常使用的 不加參數的時候默認是字符串形式的 @app.route('/post/<int:post_id>') #經常使用的 #指定int,說明是整型的 @app.route('/post/<float:post_id>') @app.route('/post/<path:path>') @app.route('/login', methods=['GET', 'POST'])
1
2
3
4
5
6
7
8
9
|
DEFAULT_CONVERTERS
=
{
'default'
: UnicodeConverter,
'string'
: UnicodeConverter,
'any'
: AnyConverter,
'path'
: PathConverter,
'int'
: IntegerConverter,
'float'
: FloatConverter,
'uuid'
: UUIDConverter,
}
|
endpoint("name") #別名,至關於django中的name
from flask import Flask, url_for @app.route('/index',endpoint="xxx") #endpoint是別名 def index(): v = url_for("xxx") print(v) return "index" @app.route('/zzz/<int:nid>',endpoint="aaa") #endpoint是別名 def zzz(nid): v = url_for("aaa",nid=nid) print(v) return "index2"
@app.route和app.add_url_rule參數: rule, URL規則 view_func, 視圖函數名稱 defaults=None, 默認值,當URL中無參數,函數須要參數時,使用defaults={'k':'v'}爲函數提供參數 endpoint=None, 名稱,用於反向生成URL,即: url_for('名稱') methods=None, 容許的請求方式,如:["GET","POST"] strict_slashes=None, 對URL最後的 / 符號是否嚴格要求, 如: @app.route('/index',strict_slashes=False), #當爲False時,url上加不加斜槓都行 訪問 http://www.xx.com/index/ 或 http://www.xx.com/index都可 @app.route('/index',strict_slashes=True) #當爲True時,url後面必須不加斜槓 僅訪問 http://www.xx.com/index redirect_to=None, 由原地址直接重定向到指定地址,原url有參數時,跳轉到的新url也得傳參,注意:新url中不用指定參數類型,直接用舊的參數的類型 如: @app.route('/index/<int:nid>', redirect_to='/home/<nid>') # 訪問index時,會直接自動跳轉到home,執行home的函數,
不執行index的
或 def func(adapter, nid): return "/home/888" @app.route('/index/<int:nid>', redirect_to=func) subdomain=None, 子域名訪問 from flask import Flask, views, url_for app = Flask(import_name=__name__) app.config['SERVER_NAME'] = 'haiyan.com:5000' @app.route("/", subdomain="admin") def static_index(): """Flask supports static subdomains This is available at static.your-domain.tld""" return "admin.xxx.com" #動態生成 @app.route("/dynamic", subdomain="<username>") def username_index(username): """Dynamic subdomains are also supported Try going to user1.your-domain.tld/dynamic""" return username + ".your-domain.tld" if __name__ == '__main__': app.run() 全部的域名都得與IP作一個域名解析: 若是你想經過域名去訪問,有兩種解決方式: 方式一: 1、租一個域名 haiyan.lalala 二、租一個公網IP 49.8.5.62 3、域名解析: haiyan.com 49.8.5.62 四、吧代碼放在49.8.5.62這個服務器上,程序運行起來 用戶能夠經過IP進行訪問 方式二:若是是本身測試用的就能夠用這種方式。先在本身本地的文件中找 C:\Windows\System32\drivers\etc 找到HOST,修改配置 而後吧域名修改爲本身的本地服務器127.0.0.1 加上配置:app.config["SERVER_NAME"] = "haiyan.com:5000"
# =============== 子域名訪問============ @app.route("/static_index", subdomain="admin") def static_index(): return "admin.bjg.com" # ===========動態生成子域名=========== @app.route("/index",subdomain='<xxxxx>') def index(xxxxx): return "%s.bjg.com" %(xxxxx,)
擴展Flask的路由系統,讓他支持正則,這個類必須這樣寫,必須去繼承BaseConverter
from flask import Flask,url_for from werkzeug.routing import BaseConverter app = Flask(__name__) # 定義轉換的類 class RegexConverter(BaseConverter): """ 自定義URL匹配正則表達式 """ def __init__(self, map, regex): super(RegexConverter, self).__init__(map) self.regex = regex def to_python(self, value): """ 路由匹配時,匹配成功後傳遞給視圖函數中參數的值 :param value: :return: """ return int(value) def to_url(self, value): """ 使用url_for反向生成URL時,傳遞的參數通過該方法處理,返回的值用於生成URL中的參數 :param value: :return: """ val = super(RegexConverter, self).to_url(value) return val # 添加到converts中 app.url_map.converters['regex'] = RegexConverter # 進行使用 @app.route('/index/<regex("\d+"):nid>',endpoint='xx') def index(nid): url_for('xx',nid=123) return "Index" if __name__ == '__main__': app.run()
http://docs.jinkan.org/docs/flask/views.html
https://aliang.org/Python/Flask-route.html
def auth(func): def inner(*args, **kwargs): result = func(*args, **kwargs) return result return inner class IndexView(views.MethodView): # methods = ['POST'] #只容許POST請求訪問 decorators = [auth,] # 若是想給全部的get,post請求加裝飾器,就能夠這樣來寫,也能夠單個指定 def get(self): #若是是get請求須要執行的代碼 v = url_for('index') print(v) return "GET" def post(self): #若是是post請求執行的代碼 return "POST" app.add_url_rule('/index', view_func=IndexView.as_view(name='index')) #name即FBV中的endpoint,指別名 if __name__ == '__main__': app.run()
視圖函數不能面向對象編程,利用類視圖來代替視圖函數來解決這個問題
from flask.views import View
class MyView(View): # MyView繼承於View def test(self): # 自定義的方法 return '測試類視圖' def dispatch_request(self): # 必須重寫這個方法 resp = self.test() return resp
MyView.as_view('test')
注意:.as_view方法的放回值是一個方法,並且該方法的名字就是傳進去的參數
app.add_url_rule('/test/', view_func=MyView.as_view('test')) # MyView.as_view('test') 返回的是一個方法
把as_view方法返回的結果賦值給view_func
as_view方法返回的是一個方法(注意:as_view方法傳入的參數就是as_view返回的那個方法的名字),該方法會調用dispatch_request方法
一旦路由進來,就會調用 dispatch_request 方法
類視圖的目的就是實現邏輯分離、方便管理
from flask import Flask from flask.views import View app = Flask(__name__) @app.route('/') def index(): return 'Hello World' class MyView(View): # MyView繼承於View def test(self): # 自定義的方法 return '測試類視圖' def dispatch_request(self): # 必須重寫這個方法 resp = self.test() return resp app.add_url_rule('/test/', view_func=MyView.as_view('test')) # MyView.as_view('test') 返回的是一個方法 print(app.url_map) if __name__ == '__main__': app.run(debug=True) # 把as_view方法返回的結果賦值給view_func # as_view方法返回的是一個方法(注意:as_view方法傳入的參數就是as_view返回的那個方法的名字),該方法會調用dispatch_request方法 # 一旦路由進來,就會調用 dispatch_request 方法 # 類視圖的目的就是實現邏輯分離、方便管理
利用視圖函數實現不一樣的請求執行不一樣的邏輯時比較複雜,須要在視圖函數函數中進行判斷;若是利用方法視圖實現就比較簡單
@app.route('/test/', methods=['GET', 'POST']) def test(): if request.method == 'GET': # 作GET的事情 pass elif request.method == 'POST': # 作POST的事情 pass return '測試'
from flask.views import MethodView
class TestMethodView(MethodView): def get(self): # 處理Get請求 return 'GET請求' def post(self): # 處理post請求 return 'POST請求'
注意:視圖類中的方法就是支持的請求類型
TestMethodView.as_view('testMethodView')
注意:as_view返回的是一個視圖函數,並且該視圖函數逇名稱就是傳進去的參數
app.add_url_rule('/test02/', view_func=TestMethodView.as_view('testMethodView'))
from flask import Flask from flask import request from flask.views import MethodView app = Flask(__name__) @app.route('/') def index(): return '測試主頁面' @app.route('/test/', methods=['GET', 'POST']) def test(): if request.method == 'GET': # 作GET的事情 pass elif request.method == 'POST': # 作POST的事情 pass return '測試' class TestMethodView(MethodView): def get(self): # 處理Get請求 return 'GET請求' def post(self): # 處理post請求 return 'POST請求' app.add_url_rule('/test02/', view_func=TestMethodView.as_view('testMethodView')) # method = TestMethodView.as_view('testMethodView'); # app.add_url_rule('/test02/<name>/', view_func=method, methods=['GET']) print(app.url_map) if __name__ == '__main__': app.run(debug=True)
method02 = TestMethodView.as_view('testMethodView02'); app.add_url_rule('/test02/<name>/', view_func=method02, methods=['GET'])
method03 = TestMethodView.as_view('testMethodView03') app.add_url_rule('/test03/', view_func=method03, methods=['POST'])
模擬POST請求
from flask import Flask from flask import request from flask.views import MethodView app = Flask(__name__) @app.route('/') def index(): return '測試主頁面' @app.route('/test/', methods=['GET', 'POST']) def test(): if request.method == 'GET': # 作GET的事情 pass elif request.method == 'POST': # 作POST的事情 pass return '測試' class TestMethodView(MethodView): def get(self, name): # 處理Get請求, 也能夠在這些方法中調用其餘的方法 return 'GET請求' + name def post(self): # 處理post請求, 也能夠在這些方法中調用其餘的方法 return 'POST請求' # app.add_url_rule('/test02/', view_func=TestMethodView.as_view('testMethodView')) method02 = TestMethodView.as_view('testMethodView02'); app.add_url_rule('/test02/<name>/', view_func=method02, methods=['GET']) method03 = TestMethodView.as_view('testMethodView03') app.add_url_rule('/test03/', view_func=method03, methods=['POST']) print(app.url_map) if __name__ == '__main__': app.run(debug=True)
from flask import Flask from flask import request from flask import render_template from flask import redirect from flask import make_response app = Flask(__name__) @app.route('/login.html', methods=['GET', "POST"]) def login(): # 請求相關信息 # request.method # request.args # request.form # request.values # request.cookies # request.headers # request.path # request.full_path # request.script_root # request.url # request.base_url # request.url_root # request.host_url # request.host # request.files # obj = request.files['the_file_name'] # obj.save('/var/www/uploads/' + secure_filename(f.filename)) # 響應相關信息 # return "字符串" # return render_template('html模板路徑',**{}) # return redirect('/index.html') # response = make_response(render_template('index.html')) # response是flask.wrappers.Response類型 # response.delete_cookie('key') # response.set_cookie('key', 'value') # response.headers['X-Something'] = 'A value' # return response return "內容" if __name__ == '__main__': app.run()
from flask import Flask,url_for,request,redirect,render_template,jsonify,make_response from urllib.parse import urlencode,quote,unquote app = Flask(__name__) @app.route('/index',endpoint='xx') def index(): from werkzeug.datastructures import ImmutableMultiDict ================= # get_data = request.args # get_dict = get_data.to_dict() # get_dict['xx'] = '18' # url = urlencode(get_dict) # print(url) ==================== # print(request.query_string) # print(request.args) ========================== # val = "%E6%8A%8A%E5%87%A0%E4%B8%AA" # print(unquote(val)) #把上面這樣的數據轉換成中文 # # return "Index" # return "Index" # return redirect() # return render_template() # return jsonify(name='alex',age='18') #至關於JsonResponse ======================= response = make_response('xxxxx') ##若是是返回更多的值,cookie,headers,或者其餘的就可用它 response.headers['xxx'] = '123123' return response if __name__ == '__main__': # app.__call__ app.run()
Flask使用的是Jinja2模板,因此其語法和Django無太大差異
Flask中模板裏面,執行函數時,須要帶()才執行
方法一:在後端使用Markup,等價於Django裏的mark_safe
v = Markup("<input type='text' />")
方法二:在前端使用safe
{{ v1|safe }}
目錄結構:
方式一:
方式二:
點擊選擇jinja2
Flask中自定義模板方法的方式和Bottle類似,建立一個函數並經過參數的形式傳入render_template,
run.py
from flask import Flask,url_for,render_template,Markup app = Flask(__name__) def test(a,b): #自定義的標籤,此方法在使用時,須要在render_temlate中傳入到指定以頁面使用 return a+b @app.template_global() # 不須要傳入,可直接在頁面使用 def sb(a1, a2): return a1 + a2 + 100 @app.template_filter() #不須要傳入,使用時要在一個值(此值做爲第一個參數傳入到過濾器中)的後面加入|,而後再加參數 def db(a1, a2, a3): return a1 + a2 + a3 @app.route('/index') def index(): v1 = "字符串" v2 = [11,22,33] v3 = {"k1":"v3","sdf":"sdgfgf"} v4 = "<input type='text' />" v5 = Markup("<input type='text' />") return render_template("index.html",v1=v1,v2=v2,v3=v3,v4=v4,v5=v5,test=test) if __name__ == '__main__': app.run(debug=True)
index.html
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta http-equiv="X-UA-Compatible" content="IE=edge"> <meta name="viewport" content="width=device-width"> <title>Title</title> </head> <body> {{ v1 }} <ul> {% for foo in v2 %} <li>{{ foo }}</li> {% endfor %} {{ v2.1 }} {% for k,v in v3.items() %} <li>{{ k }} {{ v }}</li> {% endfor %} {{ v3.k1 }} {{ v3.get("k1") }} {{ v4|safe }} {{ v5 }} <h1>{{ test(1,2) }}</h1> <p>{{ sb(1,2) }}</p> <p>{{ 1| db(2,3) }}</p> </ul> </body> </html>
PS:模板繼承的方法和django的同樣。
只有定義的東西在不少地方去使用的時候纔去用它,
html
{% macro xx(name, type='text', value='') %} <input type="{{ type }}" name="{{ name }}" value="{{ value }}">
<input type="{{ type }}" name="{{ name }}" value="{{ value }}">
<input type="{{ type }}" name="{{ name }}" value="{{ value }}">
{% endmacro %} {{ xx('n1') }}
至關於在頁面上定義了一個名爲xx的'函數',這個函數接收3個參數,咱們給type和value寫上了默認值,此時調用,咱們還須要傳入一個參數,咱們此時傳入了一個n1,則
頁面上會生成3個input框,name都爲n1
除請求對象以外,還有一個 session 對象。它容許你在不一樣請求間存儲特定用戶的信息。它是在 Cookies 的基礎上實現的,而且對 Cookies 進行密鑰簽名要使用會話,你須要設置一個密鑰。
設置:session['username'] = 'xxx'
from flask import Flask,url_for,session app = Flask(__name__) app.secret_key = "sdsfdgdgdgd" app.config['SESSION_COOKIE_NAME'] = 'session_lvning' #設置session的名字 @app.route('/index/') def index(nid): #session本質上操做的是字典, 全部對session操做的方法與字典方法相同 #session的原理:若是下一次訪問的時候帶着隨機字符串,會把session裏面對應的 # 值拿到內存,假設session保存在數據庫,每執行一次連接一次數據庫,每次都要時時更新的話,會很是損耗數據庫的效率 session["xxx"] = 123 session["xxx2"] = 123 session["xxx3"] = 123 session["xxx4"] = 123 del session["xxx2"] #在這刪除了,真正存儲的時候是沒有xxx2的 return "ddsf" if __name__ == '__main__': app.run()
app.config['SESSION_COOKIE_NAME'] = 'session_lvning'
- session超時時間如何設置? 'PERMANENT_SESSION_LIFETIME': timedelta(days=31) 如下是跟session相關的配置文件 """ 'SESSION_COOKIE_NAME': 'session', 'SESSION_COOKIE_DOMAIN': None, 'SESSION_COOKIE_PATH': None, 'SESSION_COOKIE_HTTPONLY': True, 'SESSION_COOKIE_SECURE': False, 'SESSION_REFRESH_EACH_REQUEST': True, #是否每次都跟新 'PERMANENT_SESSION_LIFETIME': timedelta(days=31)
from flask import Flask, session, redirect, url_for, escape, request app = Flask(__name__) @app.route('/') def index(): if 'username' in session: return 'Logged in as %s' % escape(session['username']) return 'You are not logged in' @app.route('/login', methods=['GET', 'POST']) def login(): if request.method == 'POST': session['username'] = request.form['username'] return redirect(url_for('index')) return ''' <form action="" method="post"> <p><input type=text name=username> <p><input type=submit value=Login> </form> ''' @app.route('/logout') def logout(): # remove the username from the session if it's there session.pop('username', None) return redirect(url_for('index')) # set the secret key. keep this really secret: app.secret_key = 'A0Zr98j/3yX R~XHH!jmN]LWX/,?RT'
pip3 install Flask-Session run.py from flask import Flask from flask import session from pro_flask.utils.session import MySessionInterface app = Flask(__name__) app.secret_key = 'A0Zr98j/3yX R~XHH!jmN]LWX/,?RT' app.session_interface = MySessionInterface() @app.route('/login.html', methods=['GET', "POST"]) def login(): print(session) session['user1'] = 'alex' session['user2'] = 'alex' del session['user2'] return "內容" if __name__ == '__main__': app.run() session.py #!/usr/bin/env python # -*- coding:utf-8 -*- import uuid import json from flask.sessions import SessionInterface from flask.sessions import SessionMixin from itsdangerous import Signer, BadSignature, want_bytes class MySession(dict, SessionMixin): def __init__(self, initial=None, sid=None): self.sid = sid self.initial = initial super(MySession, self).__init__(initial or ()) def __setitem__(self, key, value): super(MySession, self).__setitem__(key, value) def __getitem__(self, item): return super(MySession, self).__getitem__(item) def __delitem__(self, key): super(MySession, self).__delitem__(key) class MySessionInterface(SessionInterface): session_class = MySession container = {} def __init__(self): import redis self.redis = redis.Redis() def _generate_sid(self): return str(uuid.uuid4()) def _get_signer(self, app): if not app.secret_key: return None return Signer(app.secret_key, salt='flask-session', key_derivation='hmac') def open_session(self, app, request): """ 程序剛啓動時執行,須要返回一個session對象 """ sid = request.cookies.get(app.session_cookie_name) if not sid: sid = self._generate_sid() return self.session_class(sid=sid) signer = self._get_signer(app) try: sid_as_bytes = signer.unsign(sid) sid = sid_as_bytes.decode() except BadSignature: sid = self._generate_sid() return self.session_class(sid=sid) # session保存在redis中 # val = self.redis.get(sid) # session保存在內存中 val = self.container.get(sid) if val is not None: try: data = json.loads(val) return self.session_class(data, sid=sid) except: return self.session_class(sid=sid) return self.session_class(sid=sid) def save_session(self, app, session, response): """ 程序結束前執行,能夠保存session中全部的值 如: 保存到resit 寫入到用戶cookie """ domain = self.get_cookie_domain(app) path = self.get_cookie_path(app) httponly = self.get_cookie_httponly(app) secure = self.get_cookie_secure(app) expires = self.get_expiration_time(app, session) val = json.dumps(dict(session)) # session保存在redis中 # self.redis.setex(name=session.sid, value=val, time=app.permanent_session_lifetime) # session保存在內存中 self.container.setdefault(session.sid, val) session_id = self._get_signer(app).sign(want_bytes(session.sid)) response.set_cookie(app.session_cookie_name, session_id, expires=expires, httponly=httponly, domain=domain, path=path, secure=secure)
#!/usr/bin/env python # -*- coding:utf-8 -*- """ pip3 install redis pip3 install flask-session """ from flask import Flask, session, redirect from flask.ext.session import Session app = Flask(__name__) app.debug = True app.secret_key = 'asdfasdfasd' app.config['SESSION_TYPE'] = 'redis' from redis import Redis app.config['SESSION_REDIS'] = Redis(host='192.168.0.94',port='6379') Session(app) @app.route('/login') def login(): session['username'] = 'alex' return redirect('/index') @app.route('/index') def index(): name = session['username'] return name if __name__ == '__main__': app.run()
''' Django中,session保存在服務端的數據庫中,數據庫中保存請求用戶的全部數據,服務端數據中{'隨機字符串':加密後的客戶相關信息} 請求完成後,把隨機字符串做爲值,返回給客戶端,保存在客戶端的cookie中,鍵爲:sessionid,值爲:服務端返回的隨機字符串;即{'sessionid':'隨機字符串'} Flask中,服務端什麼都不存,用戶第一次請求時,在內存中生成一個空字典,將這個空字典加密後,返回給客戶端,保存在客戶端的cookie中,鍵爲’session',值爲:加密後的字典 下次訪問時,讀取客戶端cookie中key爲session對應的值 而後進行解密(若是不能按以前的的加密方式對應個解密方式解密,即認爲第一次請求,從新生成空字典),解密成功後,能夠對字典進行操做,保存新數據在字典中,請求完成後,會從新加密這個字典,返回個客戶端保存 '''
若是代碼很是多,要進行歸類。不一樣的功能放在不一樣的文件,吧相關的視圖函數也放進去。藍圖也就是對flask的目錄結構進行分配(應用於小,中型的程序)
藍圖用於爲應用提供目錄劃分:
manage.py
import fcrm if __name__ == '__main__': fcrm.app.run()
__init__.py(只要一導入fcrm就會執行__init__.py文件)
from flask import Flask #導入accout 和order from fcrm.views import accout from fcrm.views import order app = Flask(__name__) print(app.root_path) #根目錄 app.register_blueprint(accout.accout) #把藍圖註冊到app裏面,accout.accout是建立的藍圖對象 app.register_blueprint(order.order)
accout.py
from flask import Blueprint,render_template accout = Blueprint("accout",__name__) @accout.route('/accout') def xx(): return "accout" @accout.route("/login") def login(): return render_template("login.html")
order.py
from flask import Blueprint order = Blueprint("order",__name__) @order.route('/order') def register(): return "order
小型應用程序:代碼示例
大型應用程序:代碼示例
藍圖中的視圖函數的名字不能和藍圖對象的名字同樣!!!
session存在在服務端的一個字典裏面,session保存起來,取一次裏面仍是有的,直到你刪除以後纔沒有了
flash是基於session建立的,flash支持往裏邊放值,只要你取一下就沒有了,至關於pop了一下。不只能夠拿到值,並且能夠把其從session裏的去掉,
基於Session實現的用於保存數據的集合,其特色是:使用一次就刪除。
某個數據僅需用一次時,可使用閃現
from flask import Flask,session,Session,flash,get_flashed_messages,redirect,render_template,request app = Flask(__name__) app.secret_key ='sdfsdfsdf' @app.route('/users') def users(): # 方式一 # msg = request.args.get('msg','') # 方式二 # msg = session.get('msg') # if msg: # del session['msg'] # 方式三 v = get_flashed_messages() # 獲取flash中的值 print(v) msg = '' return render_template('users.html',msg=msg) @app.route('/useradd') def user_add(): # 在數據庫中添加一條數據 # 假設添加成功,在跳轉到列表頁面時,顯示添加成功 # 方式一 # return redirect('/users?msg=添加成功') # 方式二 # session['msg'] = '添加成功' # 方式三 flash('添加成功') return redirect('/users') if __name__ == '__main__': app.run(debug=True)
在函數執行以前或函數執行以後想作點事情,有2種方式
第一種:裝飾器
第二種:flask裏面的擴展,至關於django中的中間件
from flask import Flask,session,Session,flash,get_flashed_messages,redirect,render_template,request app = Flask(__name__) app.secret_key ='sdfsdfsdf' @app.before_request def process_request1(): print('process_request1') @app.after_request def process_response1(response): print('process_response1') return response @app.before_request def process_request2(): print('process_request2') @app.after_request def process_response2(response): #參數也得有 print('process_response2') return response #必須有返回值 @app.route('/index') def index(): print('index') return 'Index' @app.route('/order') def order(): print('order') return 'order' @app.route('/test') def test(): print('test') return 'test' if __name__ == '__main__': app.run()
運行結果:
還有一個@app.before_first_request:表示,當程序運行起來,第一個請求來的時候就只執行一次,下次再來就不會在執行了
#!/usr/bin/env python # -*- coding:utf-8 -*- from flask import Flask, Request, render_template app = Flask(__name__, template_folder='templates') app.debug = True @app.before_first_request # 只在第一次請求到來時執行一次,後面不會再執行 def before_first_request1(): print('before_first_request1') @app.before_first_request def before_first_request2(): print('before_first_request2') @app.before_request # 每次請求到來時,都會執行 def before_request1(): Request.nnn = 123 print('before_request1') @app.before_request def before_request2(): print('before_request2') @app.after_request # 每次響應時執行 def after_request1(response): print('before_request1', response) return response @app.after_request def after_request2(response): print('before_request2', response) return response @app.errorhandler(404) def page_not_found(error): return 'This page does not exist', 404 @app.template_global() # 自定義標籤,全部頁面都直接使用 def sb(a1, a2): return a1 + a2 @app.template_filter() # 自定義過濾器,全部頁面都直接使用 def db(a1, a2, a3): return a1 + a2 + a3 @app.route('/') # 訪問的url,不加其餘後綴時,也要有/ def hello_world(): return render_template('hello.html') if __name__ == '__main__': app.run()
自定義標籤和過濾器在頁面上的調用方式:{{sb(1,2)}} {{ 1|db(2,3)}}
from flask import abort
abort(404)
@app.errorhandler(404) # 404頁面鉤子 def page_404(er): # 參數是原始的404頁面提示信息 print(er) return '這是統一的錯誤頁面', 404, {} # 返回本身編寫的404頁面信息
from flask import Flask from flask import abort app = Flask(__name__) @app.route('/') def index(): return '測試主頁面' movies = [1,2,3,4,5] @app.route('/movie/<int:num>/') def movie(num): if num in movies: return '電影 {} 的詳細信息爲:...... '.format(num) abort(404) # 本身編寫的404頁面會顯示在這裏 @app.errorhandler(404) # 404頁面鉤子 def page_404(er): # 參數是原始的404頁面提示信息 print(er) return '這是統一的錯誤頁面', 404, {} # 返回本身編寫的404頁面信息 print(app.url_map) if __name__ == '__main__': app.run(debug=True)