
本文档旨在解决Flask应用中,如何将API调用获取的数据,通过session传递到另一个路由,并最终实现将数据以CSV格式下载的功能。我们将详细讲解如何使用session存储数据,并在下载路由中读取并处理数据,最终生成可下载的CSV文件。同时,也会讨论session大小限制以及替代方案。
在Flask应用中,需要在不同路由之间传递数据,并且最终提供CSV下载功能,常见的方案是利用Flask的session机制。然而,直接将大型DataFrame存储在session中可能会遇到问题,比如session大小限制。以下提供一种更健壮的实现方案。
核心问题:
解决方案:
1. 初始化Flask应用并设置Secret Key:
from flask import Flask, render_template, request, session, send_file import pandas as pd import io app = Flask(__name__) app.secret_key = "your_secret_key" # 必须设置,用于加密session数据
2. 数据处理路由(/result):
该路由接收表单数据,调用API获取数据,并将DataFrame序列化后存入session。
@app.route("/result", methods=['POST', 'GET'])
def result():
if request.method == 'POST':
params_a = request.form.get('params_a_input')
params_b = request.form.get('params_b_input')
params_c = int(request.form.get('params_c_input'))
data_result_1 = None
data_result_2 = None
if params_b == 'option_1':
# 假设miner_1.getData返回一个字典,包含'data_frame'键
from miner_1 import miner_1 # 假设miner_1是你的模块
data_result_1 = miner_1.getData(parameter_a=params_a, params_c=params_c)
df = data_result_1['data_frame']
elif params_b == 'option_2':
# 假设miner_2.getData返回一个字典,包含'data_frame'键
from miner_2 import miner_2 # 假设miner_2是你的模块
data_result_2 = miner_2.getData(parameter_a=params_a, parameter_c=params_c)
df = data_result_2['data_frame']
else:
return "Need to select parameter_b"
# 将DataFrame转换为JSON字符串并存入session
session['data'] = df.to_json()
return render_template('index.html', data_result_1 = data_result_1, data_result_2 = data_result_2)
else:
return render_template('index.html')3. 下载路由(/download):
该路由从session中读取JSON字符串,将其反序列化为DataFrame,并生成CSV文件供下载。
@app.route('/download')
def download():
data = session.get('data') # 使用session.get()避免KeyError
if data is None:
return "No data found in session", 400 # 返回错误信息
df = pd.read_json(data)
# 使用io.BytesIO在内存中创建CSV文件
csv_buffer = io.BytesIO()
df.to_csv(csv_buffer, index=False, encoding='utf-8')
csv_buffer.seek(0) # 将指针移动到buffer开头
return send_file(
csv_buffer,
mimetype='text/csv',
as_attachment=True,
download_name='data.csv'
)4. HTML模板(index.html):
在模板中添加下载链接。
{% if data_result_1 or data_result_2 %}
<a href="{{ url_for('download') }}" class="btn btn-outline-info">Download CSV</a>
{% endif %}from flask import Flask, render_template, request, session, send_file
import pandas as pd
import io
app = Flask(__name__)
app.secret_key = "your_secret_key"
@app.route("/")
def home():
return render_template('index.html')
@app.route("/result", methods=['POST', 'GET'])
def result():
if request.method == 'POST':
params_a = request.form.get('params_a_input')
params_b = request.form.get('params_b_input')
params_c = int(request.form.get('params_c_input'))
data_result_1 = None
data_result_2 = None
if params_b == 'option_1':
# 假设miner_1.getData返回一个字典,包含'data_frame'键
# from miner_1 import miner_1 # 假设miner_1是你的模块
# data_result_1 = miner_1.getData(parameter_a=params_a, params_c=params_c)
# df = data_result_1['data_frame']
data_result_1 = {'data_count': 18, 'data_frame': pd.DataFrame({'col1': [1, 2], 'col2': [3, 4]}), 'data_frame_html': '<table></table>'}
df = data_result_1['data_frame']
elif params_b == 'option_2':
# 假设miner_2.getData返回一个字典,包含'data_frame'键
# from miner_2 import miner_2 # 假设miner_2是你的模块
# data_result_2 = miner_2.getData(parameter_a=params_a, parameter_c=params_c)
# df = data_result_2['data_frame']
data_result_2 = {'data_count': 18, 'data_frame': pd.DataFrame({'col1': [5, 6], 'col2': [7, 8]}), 'data_frame_html': '<table></table>'}
df = data_result_2['data_frame']
else:
return "Need to select parameter_b"
# 将DataFrame转换为JSON字符串并存入session
session['data'] = df.to_json()
return render_template('index.html', data_result_1 = data_result_1, data_result_2 = data_result_2)
else:
return render_template('index.html')
@app.route('/download')
def download():
data = session.get('data') # 使用session.get()避免KeyError
if data is None:
return "No data found in session", 400 # 返回错误信息
df = pd.read_json(data)
# 使用io.BytesIO在内存中创建CSV文件
csv_buffer = io.BytesIO()
df.to_csv(csv_buffer, index=False, encoding='utf-8')
csv_buffer.seek(0) # 将指针移动到buffer开头
return send_file(
csv_buffer,
mimetype='text/csv',
as_attachment=True,
download_name='data.csv'
)
if __name__ == "__main__":
app.run(debug=True)index.html
<!DOCTYPE html>
<html>
<head>
<title>Flask CSV Download</title>
</head>
<body>
<h1>Data Miner</h1>
<form method="POST" action="/result">
<label for="params_a_input">Parameter A:</label><br>
<input type="text" id="params_a_input" name="params_a_input"><br><br>
<label for="params_b_input">Parameter B:</label><br>
<select id="params_b_input" name="params_b_input">
<option value="option_1">Option 1</option>
<option value="option_2">Option 2</option>
</select><br><br>
<label for="params_c_input">Parameter C:</label><br>
<input type="number" id="params_c_input" name="params_c_input"><br><br>
<input type="submit" value="Get Data">
</form>
{% if data_result_1 or data_result_2 %}
<a href="{{ url_for('download') }}" class="btn btn-outline-info">Download CSV</a>
{% endif %}
{% if data_result_1 %}
<h2>Data Result 1</h2>
<p>Data Count: {{ data_result_1['data_count'] }}</p>
{{ data_result_1['data_frame_html']|safe }}
{% endif %}
{% if data_result_2 %}
<h2>Data Result 2</h2>
<p>Data Count: {{ data_result_2['data_count'] }}</p>
{{ data_result_2['data_frame_html']|safe }}
{% endif %}
</body>
</html>注意事项:
如果数据量过大,无法存储在session中,可以考虑以下替代方案:
这些方案需要额外的配置和代码,但可以处理更大的数据量。
本文档详细介绍了如何使用Flask的session机制在不同路由之间传递DataFrame数据,并最终实现CSV下载功能。同时,也讨论了session大小限制以及替代方案。通过这些方法,可以有效地解决在Flask应用中处理和导出数据的常见问题。
以上就是Flask Session数据传递至另一路由并实现CSV下载教程的详细内容,更多请关注php中文网其它相关文章!
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号