0

0

保留表单提交或页面刷新后两级联动下拉框的选中状态(纯前端实现)

霞舞

霞舞

发布时间:2026-02-17 16:45:01

|

367人浏览过

|

来源于php中文网

原创

保留表单提交或页面刷新后两级联动下拉框的选中状态(纯前端实现)

本文介绍如何使用 localstorage 在纯前端环境下持久化保存两个依赖型下拉框(省/市)的用户选择,确保页面刷新或表单提交后仍自动恢复上次选中值,无需 php 服务端逻辑,完全基于 javascript 实现。

本文介绍如何使用 localstorage 在纯前端环境下持久化保存两个依赖型下拉框(省/市)的用户选择,确保页面刷新或表单提交后仍自动恢复上次选中值,无需 php 服务端逻辑,完全基于 javascript 实现。

在构建动态表单时,两级联动下拉框(如“省份 → 城市”)十分常见。当用户完成选择并提交表单,或因验证失败需回显数据时,若仅依赖服务端渲染(如 PHP 的 ),不仅代码冗余、维护困难,还违背了“客户端验证优先”的设计原则。本文提供一套纯前端、轻量、可复用的解决方案:利用 localStorage 持久化用户选择,并结合现代 DOM 操作与事件机制,实现选中状态的自动还原与联动更新。

Synthesys
Synthesys

Synthesys是一家领先的AI虚拟媒体平台,用户只需点击几下鼠标就可以制作专业的AI画外音和AI视频

下载

✅ 核心思路

  • 使用 localStorage 存储用户最后一次成功选择的 province 和 city 值;
  • 页面加载完成(DOMContentLoaded)后,优先从 localStorage 读取并设置下拉框值;
  • 触发 change 事件手动激活联动逻辑(确保城市列表正确生成并选中);
  • 在 change 事件监听器中实时更新 localStorage,保证状态同步。

? 完整实现代码

<!DOCTYPE html>
<html>
<head>
  <meta charset="UTF-8">
  <title>两级联动下拉框状态保持</title>
</head>
<body>

<form method="post" name="SignUpForm" id="SignUpForm">
  <table frame="box">
    <tr>
      <td dir="rtl" style="text-align: center"><b>省份</b></td>
      <td>
        <select name="province" id="province">
          <option value="" disabled selected>请选择省份</option>
        </select>
      </td>
    </tr>
    <tr>
      <td dir="rtl" style="text-align: center"><b>城市</b></td>
      <td>
        <select name="city" id="city">
          <option value="" disabled selected>请选择城市</option>
        </select>
      </td>
    </tr>
    <tr>
      <td colspan="2" style="text-align: center; padding-top: 15px;">
        <input type="submit" name="sign-up" id="sign-up" 
               value="提交" 
               style="font-weight: bold; font-size: large; background-color: mediumseagreen; color: darkred">
      </td>
    </tr>
  </table>
</form>

<script>
const provinceObject = {
  "province1": { "1city1p1": [1], "1city2p1": [2], "1city3p1": [3] },
  "province2": { "2city1p2": [4], "2city2p2": [5], "2city3p2": [6] },
  "province3": { "3city1p3": [7], "3city2p3": [8], "3city3p3": [9] }
};

document.addEventListener("DOMContentLoaded", function () {
  const provinceSel = document.getElementById("province");
  const citySel = document.getElementById("city");

  // 1️⃣ 初始化省份下拉框
  const provinces = Object.keys(provinceObject);
  provinceSel.innerHTML = `<option value="" disabled selected>请选择省份</option>` +
    provinces.map(p => `<option value="${p}">${p}</option>`).join("");

  // 2️⃣ 省份变更:动态生成并填充城市选项
  provinceSel.addEventListener("change", function () {
    const selectedProvince = this.value;
    citySel.innerHTML = `<option value="" disabled selected>请选择城市</option>`;

    if (selectedProvince && provinceObject[selectedProvince]) {
      const cities = Object.keys(provinceObject[selectedProvince]);
      const cityOptions = cities.map(c => `<option value="${c}">${c}</option>`).join("");
      citySel.innerHTML += cityOptions;
    }

    // ✅ 同步保存省份选择(可选:立即存,也可在 change 后统一存)
    localStorage.setItem("province", selectedProvince);
  });

  // 3️⃣ 城市变更:仅保存选择,不触发联动
  citySel.addEventListener("change", function () {
    localStorage.setItem("city", this.value);
  });

  // 4️⃣ 页面加载时:恢复历史选择(关键步骤!)
  const savedProvince = localStorage.getItem("province");
  const savedCity = localStorage.getItem("city");

  if (savedProvince && provinceSel.querySelector(`option[value="${savedProvince}"]`)) {
    provinceSel.value = savedProvince;
    // 手动触发 change 以加载对应城市列表
    provinceSel.dispatchEvent(new Event("change"));

    // 等待城市列表渲染完成后再设置 city 值(确保 DOM 已更新)
    setTimeout(() => {
      if (savedCity && citySel.querySelector(`option[value="${savedCity}"]`)) {
        citySel.value = savedCity;
      }
    }, 0);
  }
});
</script>

</body>
</html>

⚠️ 注意事项与最佳实践

  • setTimeout(..., 0) 的必要性:由于 provinceSel.dispatchEvent('change') 是异步触发城市渲染,直接设置 citySel.value 可能因 DOM 尚未更新而失效;使用微任务延迟可确保城市
  • 空值与容错处理:代码中加入了 disabled selected 占位选项,并校验 provinceObject[selectedProvince] 是否存在,避免因非法值导致脚本中断。
  • 清除状态:如需重置(例如用户点击“重填”按钮),调用 localStorage.removeItem("province"); localStorage.removeItem("city"); 即可。
  • 兼容性:localStorage 支持所有现代浏览器(IE8+),若需支持更老环境,可降级为 sessionStorage 或 cookie(需额外封装)。
  • 安全性提示:localStorage 数据可被用户轻易修改,切勿用于存储敏感信息或绕过服务端校验——它仅服务于用户体验优化。

✅ 总结

通过 localStorage + DOMContentLoaded + dispatchEvent 的组合,我们彻底摆脱了 PHP 模板嵌入式选中逻辑,实现了真正解耦、可维护、高性能的前端状态持久化。该方案不仅适用于两级联动,还可轻松扩展至三级(如“省→市→区”)甚至更复杂的动态表单场景。记住:状态交由客户端管理,验证仍须服务端兜底——这才是健壮 Web 表单的黄金法则。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
cookie
cookie

Cookie 是一种在用户计算机上存储小型文本文件的技术,用于在用户与网站进行交互时收集和存储有关用户的信息。当用户访问一个网站时,网站会将一个包含特定信息的 Cookie 文件发送到用户的浏览器,浏览器会将该 Cookie 存储在用户的计算机上。之后,当用户再次访问该网站时,浏览器会向服务器发送 Cookie,服务器可以根据 Cookie 中的信息来识别用户、跟踪用户行为等。

6466

2023.06.30

document.cookie获取不到怎么解决
document.cookie获取不到怎么解决

document.cookie获取不到的解决办法:1、浏览器的隐私设置;2、Same-origin policy;3、HTTPOnly Cookie;4、JavaScript代码错误;5、Cookie不存在或过期等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

362

2023.11.23

阻止所有cookie什么意思
阻止所有cookie什么意思

阻止所有cookie意味着在浏览器中禁止接受和存储网站发送的cookie。阻止所有cookie可能会影响许多网站的使用体验,因为许多网站使用cookie来提供个性化服务、存储用户信息或跟踪用户行为。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

431

2024.02.23

cookie与session的区别
cookie与session的区别

本专题整合了cookie与session的区别和使用方法等相关内容,阅读专题下面的文章了解更详细的内容。

97

2025.08.19

DOM是什么意思
DOM是什么意思

dom的英文全称是documentobjectmodel,表示文件对象模型,是w3c组织推荐的处理可扩展置标语言的标准编程接口;dom是html文档的内存中对象表示,它提供了使用javascript与网页交互的方式。想了解更多的相关内容,可以阅读本专题下面的文章。

3751

2024.08.14

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

316

2026.02.13

微博网页版主页入口与登录指南_官方网页端快速访问方法
微博网页版主页入口与登录指南_官方网页端快速访问方法

本专题系统整理微博网页版官方入口及网页端登录方式,涵盖首页直达地址、账号登录流程与常见访问问题说明,帮助用户快速找到微博官网主页,实现便捷、安全的网页端登录与内容浏览体验。

126

2026.02.13

Flutter跨平台开发与状态管理实战
Flutter跨平台开发与状态管理实战

本专题围绕Flutter框架展开,系统讲解跨平台UI构建原理与状态管理方案。内容涵盖Widget生命周期、路由管理、Provider与Bloc状态管理模式、网络请求封装及性能优化技巧。通过实战项目演示,帮助开发者构建流畅、可维护的跨平台移动应用。

45

2026.02.13

TypeScript工程化开发与Vite构建优化实践
TypeScript工程化开发与Vite构建优化实践

本专题面向前端开发者,深入讲解 TypeScript 类型系统与大型项目结构设计方法,并结合 Vite 构建工具优化前端工程化流程。内容包括模块化设计、类型声明管理、代码分割、热更新原理以及构建性能调优。通过完整项目示例,帮助开发者提升代码可维护性与开发效率。

19

2026.02.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Node.js 教程
Node.js 教程

共57课时 | 11.5万人学习

CSS3 教程
CSS3 教程

共18课时 | 5.7万人学习

Vue 教程
Vue 教程

共42课时 | 8.5万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号