如何在PHP中实现用户登录功能?使用Session和数据库验证用户身份

答案:PHP用户登录核心安全考量包括密码哈希、SQL注入防护、Session安全、输入验证与错误处理。具体需使用password_hash和password_verify处理密码,通过PDO预处理语句防止SQL注入,登录后调用session_regenerate_id防止Session固定攻击,设置HttpOnly和Secure Cookie标志,对用户输入进行验证与输出转义,并在生产环境记录错误日志而非显示详细错误信息。

如何在php中实现用户登录功能?使用session和数据库验证用户身份

在PHP中实现用户登录功能,核心在于收集用户凭证(通常是用户名和密码),通过数据库验证这些凭证的有效性,并在验证成功后利用Session机制来维护用户的登录状态。这不仅涉及到数据交互,更深层次地触及了Web应用安全的基础。

解决方案

实现用户登录功能,我通常会这样一步步构建:

首先,我们需要一个前端界面让用户输入他们的凭证。一个简单的HTML表单,包含用户名和密码输入框,以及一个提交按钮。提交方式通常是POST,以避免敏感信息在URL中暴露。

        



接着,后端

process_login.php

文件会接收这些数据。这里是关键:我们不能直接拿用户输入的密码去和数据库里存储的密码字符串比对。正确的做法是,数据库里存的是密码的哈希值,我们拿到用户输入的密码后,对其进行哈希处理,再与数据库中的哈希值进行比对。

立即学习“PHP免费学习笔记(深入)”;

// process_login.phpsession_start(); // 启动Sessionif ($_SERVER["REQUEST_METHOD"] == "POST") {    $username = $_POST['username'] ?? '';    $password = $_POST['password'] ?? '';    // 1. 数据库连接 (使用PDO,更安全和现代)    $dsn = 'mysql:host=localhost;dbname=your_database_name;charset=utf8mb4';    $db_user = 'your_db_user';    $db_pass = 'your_db_password';    try {        $pdo = new PDO($dsn, $db_user, $db_pass, [            PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION,            PDO::ATTR_DEFAULT_FETCH_MODE => PDO::FETCH_ASSOC,        ]);    } catch (PDOException $e) {        // 生产环境中不直接显示错误,而是记录日志        die("数据库连接失败: " . $e->getMessage());    }    // 2. 查询用户 (使用预处理语句防止SQL注入)    $stmt = $pdo->prepare("SELECT id, username, password_hash FROM users WHERE username = :username");    $stmt->execute([':username' => $username]);    $user = $stmt->fetch();    if ($user) {        // 3. 验证密码        // password_verify() 是PHP内置的推荐函数,用于验证哈希密码        if (password_verify($password, $user['password_hash'])) {            // 4. 登录成功:设置Session            session_regenerate_id(true); // 每次登录成功后重新生成Session ID,防止Session固定攻击            $_SESSION['user_id'] = $user['id'];            $_SESSION['username'] = $user['username'];            $_SESSION['logged_in'] = true;            // 重定向到用户仪表盘或受保护页面            header("Location: dashboard.php");            exit();        } else {            // 密码不匹配            $_SESSION['login_error'] = "用户名或密码不正确。";            header("Location: login.php"); // 返回登录页并显示错误            exit();        }    } else {        // 用户名不存在        $_SESSION['login_error'] = "用户名或密码不正确。";        header("Location: login.php");        exit();    }} else {    // 非POST请求,重定向回登录页    header("Location: login.php");    exit();}

在用户登录后,我们需要在受保护的页面(如

dashboard.php

)检查用户的登录状态:

// dashboard.phpsession_start();if (!isset($_SESSION['logged_in']) || $_SESSION['logged_in'] !== true) {    // 如果用户未登录,重定向到登录页    header("Location: login.php");    exit();}// 用户已登录,显示受保护的内容echo "欢迎回来," . htmlspecialchars($_SESSION['username']) . "!";// 这里可以放用户的个人信息、功能模块等?>
退出登录

最后,退出登录功能也很简单:

// logout.phpsession_start();// 清除所有Session变量$_SESSION = array();// 如果Session使用了Cookie,需要删除Cookieif (ini_get("session.use_cookies")) {    $params = session_get_cookie_params();    setcookie(session_name(), '', time() - 42000,        $params["path"], $params["domain"],        $params["secure"], $params["httponly"]    );}// 销毁Sessionsession_destroy();// 重定向回登录页或首页header("Location: login.php");exit();

PHP用户登录功能的核心安全考量有哪些?

在我看来,实现一个用户登录功能,看似简单,实则处处是学问,尤其是安全问题,常常让人防不胜防。仅仅能登录还不够,我们更要思考,这个登录过程真的安全吗?以下是我认为几个不可或缺的安全考量:

1. 密码哈希与验证: 这是最基础也是最重要的。很多新手可能会直接把密码存进数据库,或者用MD5、SHA1这样已经不安全的哈希算法简单加密。这在今天看来简直是灾难。正确的做法是使用PHP内置的

password_hash()

password_verify()

函数。

password_hash()

会使用一个强哈希算法(如Bcrypt或Argon2,根据PHP版本和配置),并自动生成一个随机的盐值(salt)并将其与哈希结果一同存储。这意味着即使两个用户设置了相同的密码,它们的哈希值也会不同,大大增加了破解难度。

password_verify()

则负责安全地比对用户输入的密码和存储的哈希值。

2. SQL注入防护: 任何与数据库交互的地方,都必须警惕SQL注入。如果直接将用户输入拼接到SQL查询语句中,恶意用户可以通过输入特定的SQL代码来绕过登录、窃取数据甚至破坏数据库。解决方案是使用预处理语句(Prepared Statements),无论是PDO还是MySQLi都支持。通过占位符绑定参数,数据库会区分代码和数据,从而有效地阻止SQL注入攻击。

3. Session安全: Session管理是维护用户登录状态的关键,但也容易成为攻击目标。

Session固定攻击 (Session Fixation): 攻击者在用户登录前获取一个有效的Session ID,然后诱导用户使用这个ID登录。一旦用户登录成功,攻击者就拥有了该用户的Session权限。为了防止这种攻击,每次用户成功登录后,务必调用

session_regenerate_id(true)

来生成新的Session ID。Session劫持 (Session Hijacking): 攻击者窃取了用户的Session ID(例如通过XSS攻击、嗅探网络流量等),然后冒充用户。除了使用HTTPS加密所有通信来防止嗅探,还应设置Session Cookie的

HttpOnly

Secure

标志。

HttpOnly

阻止JavaScript访问Cookie,减少XSS攻击的危害;

Secure

则确保Cookie只在HTTPS连接下发送。同时,设置合理的Session过期时间也很重要。CSRF (Cross-Site Request Forgery) 攻击: 虽然登录表单本身受CSRF影响较小(因为攻击者无法知道用户的密码),但登录后的敏感操作(如修改密码、转账)则需要CSRF令牌来防护。每次提交表单时生成一个随机令牌,存储在Session中,并在表单中包含该令牌。后端验证提交的令牌与Session中的令牌是否一致。

4. 输入验证与净化: 永远不要相信用户输入!即使是用户名,也需要进行基本的长度、字符类型验证。如果用户输入的内容最终会被显示在页面上,那么必须使用

htmlspecialchars()

或类似的函数进行输出转义,以防止XSS攻击。

5. 错误处理与信息泄露: 生产环境中,不应该将详细的错误信息(如数据库连接错误、SQL查询错误)直接显示给用户。这可能会泄露服务器配置、数据库结构等敏感信息,给攻击者提供便利。应该记录错误日志,并向用户显示友好的通用错误提示。

如何在PHP中使用PDO安全地进行数据库操作?

在我看来,PDO(PHP Data Objects)是PHP中进行数据库操作的首选方式,它不仅提供了一个统一的接口来访问多种数据库,更重要的是,它通过预处理语句机制,为我们提供了强大的SQL注入防护能力。我几乎在所有新项目中都推荐使用PDO。

1. 建立安全的PDO连接:首先,我们需要建立一个到数据库的连接。这里的关键是正确配置DSN(Data Source Name),并处理可能出现的连接错误。

$dsn = 'mysql:host=localhost;dbname=your_database_name;charset=utf8mb4'; // 数据库类型、主机、数据库名、字符集$db_user = 'your_db_user'; // 数据库用户名$db_pass = 'your_db_password'; // 数据库密码try {    $pdo = new PDO($dsn, $db_user, $db_pass, [        // 关键配置项:        PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION, // 遇到错误时抛出PDOException,便于捕获和处理        PDO::ATTR_DEFAULT_FETCH_MODE => PDO::FETCH_ASSOC, // 默认以关联数组形式返回结果集        PDO::ATTR_EMULATE_PREPARES => false, // 禁用模拟预处理,强制MySQL服务器执行真正的预处理,增强安全性    ]);} catch (PDOException $e) {    // 在生产环境中,这里应该记录错误日志,而不是直接输出给用户    error_log("数据库连接失败: " . $e->getMessage());    die("抱歉,服务暂时不可用,请稍后再试。");}
PDO::ATTR_EMULATE_PREPARES => false

这一行非常重要,它能确保PHP不会在客户端模拟预处理语句,而是将SQL和参数分开发送给数据库服务器,让数据库服务器进行真正的预处理,从而彻底避免SQL注入。

2. 使用预处理语句:这是PDO的核心安全特性。当你需要执行一个带有用户输入参数的SQL查询时,不要直接拼接字符串,而是使用占位符。

示例:用户登录查询

// 用户输入的用户名和密码$username = $_POST['username'];$password = $_POST['password'];// 准备SQL语句,使用命名占位符或问号占位符$stmt = $pdo->prepare("SELECT id, username, password_hash FROM users WHERE username = :username");// 绑定参数并执行// 使用命名占位符时,参数名要与占位符一致$stmt->execute([':username' => $username]);// 获取结果$user = $stmt->fetch();if ($user) {    // 验证密码    if (password_verify($password, $user['password_hash'])) {        echo "登录成功!";        // ... 设置Session ...    } else {        echo "密码不正确。";    }} else {    echo "用户不存在。";}

使用问号占位符的例子:

$stmt = $pdo->prepare("INSERT INTO products (name, price) VALUES (?, ?)");$stmt->execute(['Apple', 1.99]); // 按照占位符的顺序传入参数

为什么预处理语句安全?预处理语句的工作原理是,你先将SQL查询的结构(包含占位符)发送给数据库服务器进行“预编译”。然后,你再将实际的数据(参数)发送给数据库。数据库服务器会严格区分SQL代码和数据,它知道占位符位置传入的只是数据,不会将其解析为SQL命令的一部分。这样,即使数据中包含了恶意的SQL代码片段,它们也只会作为普通字符串被处理,无法改变查询的意图。

3. 错误处理:PDO的

ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION

配置让PDO在遇到错误时抛出异常。这意味着你可以使用

try-catch

块来优雅地捕获和处理数据库操作中可能出现的错误,而不是让脚本直接崩溃或显示不友好的PHP错误信息。

通过以上这些实践,我们可以显著提高PHP应用中数据库操作的安全性。

PHP Session的工作原理及其在用户身份验证中的作用?

PHP Session在我看来,是Web开发中一个非常巧妙且实用的机制,它解决了HTTP协议无状态的根本问题,使得我们能够在用户多次请求之间“记住”用户的身份和数据。在用户身份验证中,Session扮演着核心的角色,它就像一个临时的身份卡,让服务器知道当前是谁在访问。

Session的工作原理:

简单来说,Session的工作流程是这样的:

启动Session (

session_start()

): 当一个用户首次访问网站或需要使用Session时,服务器端的PHP脚本会调用

session_start()

如果这是用户首次访问,PHP会生成一个唯一的Session ID(例如:

phpsessid=abc123xyz

)。如果用户之前已经访问过并有一个Session ID,PHP会尝试从HTTP请求中获取这个ID(通常是通过Cookie)。

Session ID的传递:

通过Cookie: 这是最常见和推荐的方式。PHP会将这个Session ID作为Cookie发送给用户的浏览器。浏览器在后续对同一网站的请求中,会自动将这个Session ID的Cookie包含在HTTP请求头中发送回服务器。通过URL参数(不推荐): 在某些不支持Cookie的环境下,Session ID也可以通过URL参数(如

yoursite.com?PHPSESSID=abc123xyz

)传递。但这不安全,容易被窃取,且影响SEO。

服务器端数据存储:

当服务器收到一个带有Session ID的请求时,它会根据这个ID去查找服务器上对应的Session数据文件(默认情况下,Session数据存储在服务器的临时目录下的文件中,文件名为

sess_SessionID

)。这些数据以PHP序列化的形式存储,可以通过

$_SESSION

超全局数组进行读写。

$_SESSION

超全局数组: 这是PHP提供的一个特殊数组,它映射了当前用户的Session数据。你可以像操作普通数组一样,往

$_SESSION

里存入任何数据类型(字符串、数组、对象等),这些数据在用户会话期间都将保持有效。

session_start();$_SESSION['username'] = 'Alice';$_SESSION['cart'] = ['item1' => 2, 'item2' => 1];echo $_SESSION['username']; // Alice

Session的生命周期: Session在以下情况下会结束:

用户主动退出登录(调用

session_destroy()

)。浏览器关闭(如果Session Cookie没有设置过期时间)。Session在服务器端过期(由

session.gc_maxlifetime

配置决定,默认通常是24分钟)。

Session在用户身份验证中的作用:

Session是实现用户登录状态持久化的基石。它将无状态的HTTP请求转化为有状态的交互,使得服务器能够“记住”哪个用户正在操作。

登录状态的标记: 当用户成功通过用户名和密码验证后,我们会在

$_SESSION

中设置一个或多个标志来表示用户已登录,并存储用户的关键信息(如用户ID、用户名)。

// 登录成功后session_start();session_regenerate_id(true); // 安全最佳实践$_SESSION['user_id'] = $user_id_from_db;$_SESSION['username'] = $username_from_db;$_SESSION['logged_in'] = true;

访问控制: 在所有需要登录才能访问的页面(受保护页面)中,我们首先会检查

$_SESSION

中是否存在这些登录标志。如果不存在,就说明用户未登录或Session已过期,此时可以将用户重定向到登录页面。

// 受保护页面顶部session_start();if (!isset($_SESSION['logged_in']) || $_SESSION['logged_in'] !== true) {    header("Location: login.php");    exit();}// 用户已登录,可以访问页面内容echo "欢迎," . $_SESSION['username'];

用户数据持久化: 除了登录状态,我们还可以将用户的偏好设置、购物车内容、权限信息等数据存储在

$_SESSION

中。这样,在用户浏览网站的不同页面时,这些数据都能被方便地访问,而无需每次都从数据库中查询。

退出登录: 当用户点击“退出登录”时,通过销毁Session(

session_destroy()

),可以清除所有存储的用户登录信息,从而结束用户的会话。

总结来说,Session机制就像是为每个访问者在服务器上创建了一个专属的“储物柜”,通过一个唯一的“钥匙”(Session ID)来管理这个储物柜。在用户身份验证中,这个储物柜里存放着用户的“通行证”,让服务器能够识别并授权用户访问受保护的资源。

以上就是如何在PHP中实现用户登录功能?使用Session和数据库验证用户身份的详细内容,更多请关注php中文网其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1272600.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
Silverstripe 子页面重排后自动发布实现指南
上一篇 2025年12月10日 14:56:17
Silverstripe子页面重排后自动发布实现指南
下一篇 2025年12月10日 14:56:41

相关推荐

  • composer require-dev和require有什么不同_Composer Require与Require-Dev区别解析

    require用于声明项目运行必需的依赖,如框架、数据库组件和第三方SDK,这些包会随项目部署到生产环境;2. require-dev用于声明仅在开发和测试阶段需要的工具,如PHPUnit、PHPStan、Faker等,不会默认部署到生产环境;3. 安装时composer install根据环境决定…

    2026年5月10日
    1000
  • 修复Django电商项目中AJAX过滤产品列表图片不显示问题

    在Django电商项目中,当使用AJAX动态加载过滤后的产品列表时,常遇到图片无法正常显示的问题。这通常是由于前端模板中图片加载方式(如data-setbg属性结合JavaScript库)与AJAX动态内容更新机制不兼容所致。解决方案是直接在AJAX返回的HTML中使用标准的标签来渲染图片,确保浏览…

    2026年5月10日
    000
  • 开源免费PHP工具 PHP开发效率提升利器

    推荐开源免费PHP开发工具以提升效率:VS Code、Sublime Text轻量高效,PhpStorm专业强大;调试用Xdebug、Kint、Ray;依赖管理选Composer;代码质量工具包括PHPStan、Psalm、PHP_CodeSniffer;数据库管理可用%ignore_a_1%MyA…

    2026年5月10日
    000
  • Matplotlib 地图中多类型图例的创建与优化

    Matplotlib 地图中多类型图例的创建与优化Matplotlib 地图中多类型图例的创建与优化Matplotlib 地图中多类型图例的创建与优化Matplotlib 地图中多类型图例的创建与优化

    本教程旨在解决matplotlib地图可视化中,如何在一个图例中同时展示颜色块(如区域分类)和自定义标记(如特定兴趣点)的问题。文章详细介绍了当传统`patch`对象无法正确显示标记时,如何利用`matplotlib.lines.line2d`创建标记图例句柄,并将其与颜色块图例句柄合并,从而生成一…

    2026年5月10日 用户投稿
    100
  • Golang JSON序列化:控制敏感字段暴露的最佳实践

    本教程探讨golang中如何高效控制结构体字段在json序列化时的可见性。当需要将包含敏感信息的结构体数组转换为json响应时,通过利用`encoding/json`包提供的结构体标签,特别是`json:”-“`,可以轻松实现对特定字段的忽略,从而避免敏感数据泄露,确保api…

    2026年5月10日
    000
  • 怎么在PHP代码中实现图片上传功能_PHP图片上传功能实现与安全处理教程

    首先创建含enctype的HTML表单,再用PHP接收文件,检查目录、移动临时文件,验证类型与大小,生成唯一文件名,并调整php.ini限制以确保上传成功。 如果您尝试在PHP项目中添加图片上传功能,但服务器无法正确接收或保存文件,则可能是由于表单配置、文件处理逻辑或安全限制的问题。以下是实现该功能…

    2026年5月10日
    100
  • 获取日期中的周数:CodeIgniter 教程

    本教程旨在帮助开发者在 CodeIgniter 框架中,从日期字符串中准确提取周数。我们将使用 PHP 内置的 DateTime 类,并提供详细的代码示例和注意事项,确保您能够轻松地在项目中实现此功能。 使用 DateTime 类获取周数 PHP 的 DateTime 类提供了一种便捷的方式来处理日…

    2026年5月10日
    100
  • HTML如何隐藏滚动条或去除滚动条

    滚动条可以存在也可以不存在,本文主要介绍了html 隐藏滚动条和去除滚动条的方法的相关资料,大家一起来学习一下html隐藏滚动条或去除滚动条的方法吧。 1. html 标签加属性 XML/HTML Code复制内容到剪贴板 2.body中加入以下代码 立即学习“前端免费学习笔记(深入)”; html…

    用户投稿 2026年5月10日
    100
  • Golang gRPC流式请求异常处理

    在Golang的gRPC流式通信中,必须通过context.Context处理异常。应监听上下文取消或超时,及时释放资源,设置合理超时,避免连接长时间挂起,并在goroutine中通过context控制生命周期。 在使用 Golang 和 gRPC 实现流式通信时,异常处理是确保服务健壮性的关键部分…

    2026年5月10日
    000
  • Go语言mgo查询构建:深入理解bson.M与日期范围查询的正确实践

    本文旨在解决go语言mgo库中构建复杂查询时,特别是涉及嵌套`bson.m`和日期范围筛选的常见错误。我们将深入剖析`bson.m`的类型特性,解释为何直接索引`interface{}`会导致“invalid operation”错误,并提供一种推荐的、结构清晰的代码重构方案,以确保查询条件能够正确…

    2026年5月10日
    100
  • vscode上怎么运行html_vscode上运行html步骤【指南】

    首先保存文件为.html格式,再通过浏览器或Live Server插件打开预览;推荐安装Live Server实现本地服务器运行与实时刷新,提升开发体验。 在 VS Code 上运行 HTML 文件并不需要复杂的配置,只需几个简单步骤即可预览页面效果。VS Code 本身是一个代码编辑器,不直接运行…

    2026年5月10日
    100
  • 修复点击时按钮抖动:CSS垂直对齐实践

    本文探讨了在Web开发中,交互式按钮(如播放/暂停按钮)在点击时发生意外垂直位移的问题。通过分析CSS样式变化对元素布局的影响,我们发现这是由于按钮不同状态下的边框样式和内边距改变,以及默认的垂直对齐行为共同作用所致。核心解决方案是利用CSS的vertical-align属性,将其设置为middle…

    2026年5月10日
    100
  • Golang goroutine与channel调试技巧

    使用go run -race检测数据竞争,结合runtime.NumGoroutine监控协程数量,通过pprof分析阻塞调用栈,利用select超时避免永久阻塞,有效排查goroutine泄漏、死锁和数据竞争问题。 Go语言的goroutine和channel是并发编程的核心,但它们也带来了调试上…

    2026年5月10日
    000
  • 页面中文本域的值怎么设置

    标签定义多行的文本输入控件。 文本区中可容纳无限数量的文本,其中的文本的默认字体是等宽字体(通常是 Courier)。 可以通过 cols 和 rows 属性来规定 textarea 的尺寸,不过更好的办法是使用 CSS 的 height 和 width 属性。 注释:在文本输入区内的文本行间,用 …

    2026年5月10日
    000
  • 使用 Jupyter Notebook 进行探索性数据分析

    Jupyter Notebook通过单元格实现代码与Markdown结合,支持数据导入(pandas)、清洗(fillna)、探索(matplotlib/seaborn可视化)、统计分析(describe/corr)和特征工程,便于记录与分享分析过程。 Jupyter Notebook 是进行探索性…

    2026年5月10日
    000
  • 《魔兽世界》将于6月11日开启国服回归技术测试

    《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试

    《%ign%ignore_a_1%re_a_1%》官方宣布,将于6月11日开启国服回归技术测试,时间为7天,并称可以在6月内正式开服,玩家们可以访问官网下载战网客户端并预下载“巫妖王之怒”客户端,技术测试详情见下图。 WordAi WordAI是一个AI驱动的内容重写平台 53 查看详情 以上就是《…

    2026年5月10日 用户投稿
    200
  • php常量怎么用_PHP常量(define/const)定义与使用方法

    PHP中可通过define函数和const关键字定义常量,用于存储不可变值。define适用于全局作用域,支持动态名称和条件定义,如define(‘SITE_NAME’, ‘MyWebsite’);const在编译时生效,语法简洁但限制多,只能在类或全…

    2026年5月10日
    000
  • 如何在HTML中插入表单元素_HTML表单控件与输入类型使用指南

    HTML表单通过标签构建,包含action和method属性定义数据提交目标与方式,常用input类型如text、password、email等适配不同输入需求,配合label、required、placeholder提升可用性,结合textarea、select、button等控件实现完整交互,是…

    2026年5月10日
    100
  • 前端缓存策略与JavaScript存储管理

    根据数据特性选择合适的存储方式并制定清晰的读写与清理逻辑,能显著提升前端性能;合理运用Cookie、localStorage、sessionStorage、IndexedDB及Cache API,结合缓存策略与定期清理机制,可在保证用户体验的同时避免安全与性能隐患。 前端缓存和JavaScript存…

    2026年5月10日
    200
  • HTML5网页如何实现手势操作 HTML5网页移动端交互的处理技巧

    首先利用原生touch事件实现滑动判断,再通过preventDefault解决滚动冲突,接着引入Hammer.js处理复杂手势,最后通过优化点击区域、避免事件冲突和增加视觉反馈提升体验。 在移动端浏览器中,HTML5网页可以通过触摸事件实现手势操作,提升用户体验。虽然原生JavaScript提供了基…

    2026年5月10日
    000

发表回复

登录后才能评论
关注微信