一篇文章入门Python Flask框架前后端数据库开发实践(pycharm在anaconda环境下)
Python Flask 是一个轻量级的 Web 应用框架,也被称为微框架。它以简洁、灵活和易于上手的特点而受到开发者的喜爱。
核心特点
轻量级:Flask 核心代码简洁,仅包含 Web 开发的基本功能,不强制使用特定的数据库、模板引擎等,开发者可按需选择适合的扩展。
灵活性高:可以自由选择使用的数据库(如 MySQL、SQLite)、模板引擎(如 Jinja2)和其他工具,方便根据项目需求进行定制。
易于上手:对于初学者而言,Flask 的文档清晰易懂,代码结构简单,能够快速搭建起一个 Web 应用。
基本组件
路由系统:用于将 URL 映射到对应的 Python 函数,当用户访问特定 URL 时,执行相应的函数并返回结果。
请求和响应对象:在处理 Web 请求时,Flask 提供了request对象用于获取客户端发送的请求信息,如表单数据、URL 参数等;同时,使用response对象返回响应内容给客户端。
模板引擎:默认使用 Jinja2 作为模板引擎,它允许将 Python 代码和 HTML 模板结合,方便生成动态的 HTML 页面。
会话管理:支持会话(session)管理,通过会话可以在多个请求之间保存用户的状态信息,如用户登录状态。
1、创建新的anaconda环境

conda create --name todoenv python=3.10


创建了todoenv这个conda环境,我们查看一下有已有的环境
conda env list

二、创建项目文件
项目结构如下:
todoapp/
├── app/
│ ├── __init__.py
│ ├── models.py
│ ├── routes.py
│ ├── templates/
│ │ ├── base.html
│ │ ├── index.html
│ │ └── edit.html
│ └── static/
│ └── style.css
├── config.py
├── requirements.txt
└── run.py
安装依赖:requirements.txt,通过 pip install -r requirements.txt 一键安装依赖
Flask==3.0.2
Flask-SQLAlchemy==3.1.1
Flask-Migrate==4.0.5
python-dotenv==1.0.1
配置文件:config.py
import os
from dotenv import load_dotenvbasedir = os.path.abspath(os.path.dirname(__file__))
load_dotenv(os.path.join(basedir, '.env'))class Config:SECRET_KEY = os.environ.get('SECRET_KEY') or 'your-secret-key-here'SQLALCHEMY_DATABASE_URI = os.environ.get('DATABASE_URL') or \'sqlite:///' + os.path.join(basedir, 'app.db')SQLALCHEMY_TRACK_MODIFICATIONS = False
应用工厂:app/__init.py__
from flask import Flask
from flask_sqlalchemy import SQLAlchemy
from flask_migrate import Migrate
from config import Configdb = SQLAlchemy()
migrate = Migrate()def create_app(config_class=Config):app = Flask(__name__)app.config.from_object(config_class)db.init_app(app)migrate.init_app(app, db)from app.routes import bpapp.register_blueprint(bp)return app
数据模型(Model):app/models.py
from datetime import datetime
from app import dbclass Todo(db.Model):id = db.Column(db.Integer, primary_key=True)title = db.Column(db.String(100), nullable=False)description = db.Column(db.String(200))completed = db.Column(db.Boolean, default=False)created_at = db.Column(db.DateTime, default=datetime.utcnow)def __repr__(self):return f'<Todo {self.title}>'
路由和视图(Controller):app/routes.py
from flask import Blueprint, render_template, request, redirect, url_for, jsonify
from app.models import Todo
from app import dbbp = Blueprint('main', __name__)@bp.route('/', methods=['GET', 'POST'])
def index():if request.method == 'POST':title = request.form['title']description = request.form['description']new_todo = Todo(title=title, description=description)db.session.add(new_todo)db.session.commit()return redirect(url_for('main.index'))todos = Todo.query.order_by(Todo.created_at.desc()).all()return render_template('index.html', todos=todos)@bp.route('/edit/<int:id>', methods=['GET', 'POST'])
def edit(id):todo = Todo.query.get_or_404(id)if request.method == 'POST':todo.title = request.form['title']todo.description = request.form['description']todo.completed = 'completed' in request.formdb.session.commit()return redirect(url_for('main.index'))return render_template('edit.html', todo=todo)@bp.route('/delete/<int:id>')
def delete(id):todo = Todo.query.get_or_404(id)db.session.delete(todo)db.session.commit()return redirect(url_for('main.index'))# API 端点示例
@bp.route('/api/todos', methods=['GET'])
def get_todos():todos = Todo.query.all()return jsonify([{'id': todo.id,'title': todo.title,'description': todo.description,'completed': todo.completed} for todo in todos])
编辑模板:app/templates/edit.htm
{% extends "base.html" %}{% block content %}<h1 class="mb-4">Edit Todo</h1><form method="POST"><div class="mb-3"><label class="form-label">Title</label><input type="text" class="form-control" name="title" value="{{ todo.title }}" required></div><div class="mb-3"><label class="form-label">Description</label><textarea class="form-control" name="description" rows="3">{{ todo.description }}</textarea></div><div class="mb-3 form-check"><input type="checkbox" class="form-check-input" name="completed"{% if todo.completed %}checked{% endif %}><label class="form-check-label">Completed</label></div><button type="submit" class="btn btn-primary">Update</button><a href="{{ url_for('main.index') }}" class="btn btn-secondary">Cancel</a></form>
{% endblock %}
样式文件:app/static/style.css
/* 基础样式 */
body {background-color: #f8f9fa;padding: 20px;
}.container {max-width: 800px;background: white;padding: 30px;border-radius: 10px;box-shadow: 0 0 10px rgba(0,0,0,0.1);
}/* Todo项样式 */
.list-group-item {margin-bottom: 10px;transition: all 0.3s ease;
}.list-group-item-success {background-color: #d4edda;border-color: #c3e6cb;
}/* 表单样式 */
form {background: #f8f9fa;padding: 20px;border-radius: 8px;margin-bottom: 30px;
}.form-control:focus {box-shadow: none;border-color: #80bdff;
}/* 按钮间距 */
.btn {margin-right: 8px;
}/* 完成状态文字 */
.list-group-item-success h5 {text-decoration: line-through;color: #155724;
}/* 时间戳样式 */
.text-muted {font-size: 0.85em;display: block;margin-top: 5px;
}
模板文件:base.html (app/templates/base.html)
<!DOCTYPE html>
<html>
<head><title>Todo App</title><link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.2/dist/css/bootstrap.min.css"><link rel="stylesheet" href="{{ url_for('static', filename='style.css') }}">
</head>
<body><div class="container mt-4">{% block content %}{% endblock %}</div>
</body>
</html>
index.html (app/templates/index.html)
{% extends "base.html" %}{% block content %}<h1 class="mb-4">Todo List</h1><form method="POST" class="mb-4"><div class="mb-3"><input type="text" class="form-control" name="title" placeholder="Title" required></div><div class="mb-3"><textarea class="form-control" name="description" placeholder="Description"></textarea></div><button type="submit" class="btn btn-primary">Add Todo</button></form><div class="list-group">{% for todo in todos %}<div class="list-group-item {% if todo.completed %}list-group-item-success{% endif %}"><div class="d-flex justify-content-between align-items-center"><div><h5>{{ todo.title }}</h5><p class="mb-0">{{ todo.description }}</p><small class="text-muted">{{ todo.created_at.strftime('%Y-%m-%d %H:%M') }}</small></div><div><a href="{{ url_for('main.edit', id=todo.id) }}" class="btn btn-sm btn-warning">Edit</a><a href="{{ url_for('main.delete', id=todo.id) }}" class="btn btn-sm btn-danger">Delete</a></div></div></div>{% endfor %}</div>
{% endblock %}
运行脚本:run.py
from app import create_appapp = create_app()if __name__ == '__main__':app.run(debug=True)
以上为全部代码,可以将其复制到对应文件创建。
三、install flask和运行项目
在todo文件目录下打开cmd窗口,激活todoenv环境
conda activate todoenv

刚才requirement.txt已经有安装依赖了,执行命令,安装项目依赖
pip install -r requirements.txt


设置环境变量,我是Windows
set FLASK_APP=run.py
set FLASK_DEBUG=1

初始化数据库
flask db init # 首次运行需要初始化迁移仓库
flask db migrate # 生成迁移脚本
flask db upgrade # 应用迁移到数据库

我这里已经创建过了,所以一次存在不为空了
运行应用
flask run
在本地浏览器打开http://127.0.0.1:5000网址

可以看到,我之前已经运行过这个,数据库在本地也有值了,所以数据库正常运行了,可以新建todo,以及修改删除


Ctrl + C结束程序退出
四、项目结构讲解
项目根目录 todoapp/
todoapp/
├── app/ # 核心应用代码
├── config.py # 全局配置文件
├── requirements.txt # 项目依赖清单
└── run.py # 启动脚本
核心应用模块 app/
| 文件/目录 | 作用 |
|---|---|
__init__.py | 应用工厂函数 • 初始化 Flask 应用实例 • 整合配置/路由/模型等模块 |
models.py | 数据模型 • 定义数据库表结构 (使用 SQLAlchemy 或 Peewee 等 ORM) |
routes.py | 路由控制器 • 处理 HTTP 请求与响应 • 业务逻辑的主要实现位置 |
templates/ | Jinja2 模板 • 存放 HTML 模板文件 • 通过 base.html 实现模板继承 |
static/ | 静态资源 • 存放 CSS/JavaScript/图片等文件 • 通过 /static/路径 访问 |
该项目遵循MVC模式:模型(Model)、视图(View)和控制器(Controller)
模型(Model):负责处理数据和业务逻辑,像数据的存储、检索、更新等操作都由它完成。
app/models.py:此文件通常定义了应用的数据模型。在一个待办事项应用(todoapp)里,会定义待办事项的类,包含待办事项的属性(如标题、描述、完成状态等)以及对这些数据进行操作的方法(如添加、删除、更新待办事项等)。
视图(View):负责将模型中的数据呈现给用户,通常是用户界面,比如网页、图形界面等。
app/templates/目录:该目录存放 HTML 模板文件,这些文件用于展示数据给用户。base.html:通常是基础模板,包含所有页面共有的部分,像头部、导航栏、底部等,其他模板可以继承它。index.html:一般用于展示待办事项列表,从模型获取待办事项数据并渲染成 HTML 页面。edit.html:用于编辑单个待办事项的页面,同样从模型获取数据并提供编辑界面。
控制器(Controller):接收用户的输入,调用模型进行相应的业务处理,再选择合适的视图展示处理结果。
app/routes.py:该文件定义了应用的路由和视图函数,负责接收用户的请求,调用模型进行数据处理,然后选择合适的视图进行渲染
整体交互流程
- 用户通过浏览器访问应用的 URL,请求会发送到
app/routes.py中的相应路由。 - 路由对应的视图函数被调用,视图函数根据需求调用
app/models.py中的模型进行数据处理。 - 模型处理完数据后,视图函数选择合适的 HTML 模板(
app/templates/目录下的文件),并将处理后的数据传递给模板。 - 模板引擎将数据填充到 HTML 模板中,生成最终的 HTML 页面返回给用户浏览器进行显示。
浏览器 → run.py → app/__init__.py (创建app) → routes.py (处理逻辑) → models.py (操作数据库) → templates/*.html (渲染) → 返回响应
浏览器->>run.py: 发起HTTP请求run.py->>__init__.py: 创建app实例__init__.py->>routes.py: 注册路由routes.py->>models.py: 调用数据库操作models.py->>数据库: 执行SQL数据库-->>models.py: 返回数据models.py-->>routes.py: 返回模型对象routes.py-->>浏览器: 渲染模板响应
相关文章:
一篇文章入门Python Flask框架前后端数据库开发实践(pycharm在anaconda环境下)
Python Flask 是一个轻量级的 Web 应用框架,也被称为微框架。它以简洁、灵活和易于上手的特点而受到开发者的喜爱。 核心特点 轻量级:Flask 核心代码简洁,仅包含 Web 开发的基本功能,不强制使用特定的数据库、模板引擎等…...
SpringBoot分布式定时任务实战:告别重复执行的烦恼
场景再现:你刚部署完基于SpringBoot的集群服务,凌晨3点突然收到监控告警——优惠券发放量超出预算两倍!检查日志发现,两个节点同时执行了定时任务。这种分布式环境下的定时任务难题,该如何彻底解决? 本文将…...
AI+金融 应用 使用DeepSeek、Qwen等大模型输入自然语言,得到通达信等行情软件公式代码,导入后使用
AI金融 应用 使用DeepSeek、Qwen等大模型输入自然语言,得到通达信等行情软件公式代码,导入后使用。不会编程,也能行情软件中实现个性化条件选股,个性化技术指标。 AIbxm低估值趋势选股策略,参考提示词: 编…...
C++20 中的同步输出流:`std::basic_osyncstream` 深入解析与应用实践
文章目录 一、std::basic_osyncstream 的背景与动机二、std::basic_osyncstream 的基本原理三、std::basic_osyncstream 的使用方法(一)基本用法(二)多线程环境下的使用(三)与文件流的结合 四、std::basic_…...
Android 关于compose的一些坑和理解
** 1.如何在 WindowManager.addView 中使用 Jetpack Compose** 一、引出问题 Android 开发中,很常见的一个场景,通过 WindowManager.addView() 添加一个 View 到屏幕上。Android 最新的视图框架 Jetpack Compose,如何应用进来。这个被添加的…...
LeetCode 30 —— 30.串联所有单词的子串
题目: 给定一个字符串 s 和一些长度相同的单词 words。找出 s 中恰好可以由 words 中所有单词串联形成的子串的起始位置。 注意子串要与 words 中的单词完全匹配,中间不能有其他字符,但不需要考虑 words 中单词串联的顺序。 示例 1ÿ…...
go语言中的strings库
strings库 func EqualFold func EqualFold(s, t string) bool判断两个utf-8编码字符串(将unicode大写、小写、标题三种格式字符视为相同)是否相同。 func main() {fmt.Println(strings.EqualFold("hello", "hello")) //truefmt.…...
【嵌入式硬件】三款DCDC调试笔记
关于开关电源芯片,重点关注输入电源范围、输出电流、最低压降。 1.MP9943: 以MP9943为例,输入电压范围4-36V,输出最大电流3A,最低压降为0.3V 调整FB使正常输出为5.06V 给定6V空载、5V空载、5V带2A负载的情况: 6V带2A…...
Cannot find module @rollup/rollup-win32-x64-msvc
方法1 在package.json中添加postinstall: "scripts": {"postinstall": "node -e \"const { platform } process; if (platform win32) { require(child_process).execSync(npm install rollup/rollup-win32-x64-msvc, { stdio: inherit });…...
Linux中修改文件的权限用什么命令?
一、核心语法 chmod [选项] [权限模式] 文件名二、权限模式详解 1. 数字模式(推荐使用) 通过rwx权限对应的数值组合: r(读)→ 4w(写)→ 2x(执行)→ 1无权限 → 0 组合规…...
【FPGA开发】FPGA点亮LED灯(增加按键暂停恢复/复位操作)
目录 一、VScode下载安装 1.1 官网下载 1.2 安装插件 二、LED流水灯点亮 2.1 任务说明 2.2 新建项目 2.3 创建Verilog文件添加至顶层实体 2.4 引脚分配 2.5 选择烧录器 2.6 添加烧录文件&下载 2.7 烧录结果 三、增加按键操作 3.1 按键暂停和恢复功能&…...
companion object和object 从kotlin转java分析
说明 companion object 中 companion类中的方法是普通的方法 在外部类中生成静态变量,静态companion 对象 object MyClass2 中 类中方法是普通方法 在MyClass2中生成静态变量,静态MyClass2对象, 一 companion object 使用 kotlin转java pa…...
Spring MVC 执行流程:一个请求在 Spring MVC 中是如何执行的?
当用户发送一个 HTTP 向 Spring MVC 应用,该请求在 Spring MVC 的执行流程如下: 当用户向 Spring MVC 发起一个 HTTP 请求,该请求会被 Dispatcher Servlet(前端控制器)拦截;DispatcherServlet 调用 Handler…...
三主热备架构
1.要求 角色主机名软件IP地址用户client192.168.72.90keepalivedvip192.168.72.100masterserverAkeepalived, nginx192.168.72.30backupserverBkeepalived, nginx192.168.72.31backupserverCkeepalived, nginx192.168.72.32webtomcat1tomcat192.168.72.41webtomcat2tomcat192.1…...
HTML 表单处理进阶:验证与提交机制的学习心得与进度(二)
步步为营:表单提交机制全面解析 提交方式详解 GET 与 POST 对比 在 HTML 表单提交中,GET 和 POST 是最为常用的两种提交方式,它们在诸多方面存在显著差异。 安全性:GET 方式将表单数据附加在 URL 的查询字符串中,数…...
JavaScript | 爬虫逆向 | 语法基础| 01
一、摘要 实践是最好的导师 二、环境配置 在开始之前,需要确保你的计算机上已经安装了 Node.js。Node.js 是一个开源的、跨平台的 JavaScript 运行时环境,它允许你在服务器端运行 JavaScript 代码。 1. 下载 安装地址:https://nodejs.org…...
python解决多个矢量点图层合并为一个点图层
1、解决矢量点图层的合并 2、解决多个点图层分别合并为不同图层(一个文件夹下所有点图层合并为一个图层,以下代码为两个文件夹,分别合并为两个总的图层) import geopandas as gpd import os import pandas as pddef merge_shapef…...
VL开源模型实现文本生成图片
一、 基础知识 根据描述生成图片的视觉-语言模型(Vision-Language Models, VL 模型)是近年来多模态生成领域的热点研究方向。这些模型能够根据自然语言描述生成高质量的图像,广泛应用于艺术创作、设计辅助、虚拟场景构建等领域。 1 根据描述…...
字节跳动实习生主导开发强化学习算法,助力大语言模型性能突破
目录 禹棋赢的背景与成就 主要成就 DAPO算法的技术细节 算法优势 禹棋赢的研究历程 关键时间节点 字节跳动的“Top Seed人才计划” 计划特点 小编总结 在大模型时代,经验不再是唯一的衡量标准,好奇心、执行力和对新技术的敏锐洞察力成为推动技术…...
九、JavaScript作用域、预解析
一、JavaScript作用域 1.JavaScript作用域 ①代码名字(变量)在某个范围内起作用和效果 目的是为了提高程序的可靠性更重要的是减少命名冲突 ②js的作用域(es6)之前:全局作用域 局部作用域 ③全局作用域:整…...
前后端+数据库的项目实战:hbu迎新网-较复杂(下)javaweb
目录 十一、实现对内容的富文本编辑(换行、图片颜色等等样式) (1)下载富文本编辑器,引入资源 (2)将原项目的内容部分替换为富文本编辑器 1、替换添加页面 2、替换修改页面(和添…...
Java-模块二-2
整数类型 byte:在 Java 中占用8位(1字节),因此它的取值范围是从 -128 到 127。这是最小的整数类型,适合用于节省空间的情况。 short:这种类型的大小是16位(2字节),允许的…...
Redis、Memcached应用场景对比
环境 Redis官方网站: Redis - The Real-time Data Platform Redis社区版本下载地址:Install Redis | Docs Memcached官方网站:memcached - a distributed memory object caching system Memcached下载地址:memcached - a dis…...
【单片机通信技术应用——学习笔记三】液晶屏显示技术,取模软件的应用
一、液晶显示技术简介 1.RGB信号线 RGB是一种色彩模式,是工业界的一种颜色标准,是通过红(R)、绿(G)、蓝(B)三个颜色通道的变化,以及它们相互之间的叠加来得到各式各样的…...
MySQL颠覆版系列————MySQL新特性(开启数据库的新纪元)上篇
文章目录 前言一、窗口函数(Window Functions)1.1 窗口函数概念1.2 常见的窗口函数 二、公用表表达式(Common Table Expressions, CTEs)2.1 公用表表达式的概念2.2 常见的公用表表达式 三、JSON增强3.1 JSON增强的概念3.2 常见的J…...
MySQL 调优:查询慢除了索引还能因为什么?
文章目录 情况一:连接数过小情况二:Buffer Pool 太小 MySQL 查询慢除了索引还能因为什么?MySQL 查询慢,我们一般也会想到是因为索引,但除了索引还有哪些原因会导致数据库查询变慢呢? 以下以 MySQL 中一条 S…...
Java实习生面试题(2025.3.23 be)
一、v-if与v-show的区别 v-show 和 v-if 都是 Vue 中的条件渲染指令,它们的主要区别在于渲染策略:v-if 会根据条件决定是否编译元素,而 v-show 则始终编译元素,只是通过改变 CSS 的 display 属性来控制显示与隐藏。 二、mybatis-…...
如何在百度搜索上删除与自己名字相关的资料
个人信息的网络足迹如同一张无形的网,将我们与世界的每一个角落紧密相连。然而,当某些与自己名字相关的资料不再希望被公众轻易检索到时,如何在百度搜索中有效“隐身”,成为了一个亟待解决的问题。面对复杂多变的网络环境…...
【C语言】C语言使用随机数srand,rand
C语言使用随机数srand,rand 可直接编译使用: #include <stdio.h> #include <time.h> #include <stdlib.h> #include <unistd.h>/* c语言提供的跟随机数有关的函数:int rand(void);返回值:产生的随机数void srand(unsiqned int seed);参数…...
为容器指定固定IP地址
文章目录 为容器指定固定IP地址可以通过以下步骤实现,适用于Docker环境:**方法一:使用Docker自定义桥接网络****方法二:Docker Compose配置****关键注意事项** 为容器指定固定IP地址可以通过以下步骤实现,适用于Docker…...
