如何做一个级联效果的,城市选择插件

本贴最后更新于 4119 天前,其中的信息可能已经天翻地覆

(一)场景

          老大安排我弄一个城市选择插件,用在公司系统。

(二)准备工作

         最主要的就是数据了,去几个网站逛了一圈,感觉京东的不错,于是就想办法把里面的数据拉下来。京东页面 ,其实很简单,用google浏览器自带的功能,捕捉下http包,查看下request的请求url,就可以看出,每一次请求,按参数返回json,然后组装显示在页面,几个js方法就可以了。然后再后台,服务器启动的时候,将这些数据从数据库查询出来放入内存中,(便于提高速度)。然后每次请求直接从内存中的数据结构拉下来就行。

1、写方法拉京东数据。

<script type="text/javascript">
	$(document).ready(function(){
		//根据市查询县
		function country(z){
			var st="";
			$.getJSON("http://buy.jd.com/purchase/flows/easybuy/FlowService.ashx?&action=GetCountys&Id="+z+"&callback=?", function(data){
				var Are= data.Obj.Areas;
				for (var x in Are){
					st +="insert into qly_bd_region (regionid,regionfid,regiontype,regionname)values('"+Are[x].Id+"' ,'"+z+"','县','"+Are[x].Name+ "');<br/>" ;	
				}
				$("#hh").append(st);
			});
		}
		//根据省id去查询对应的市
			function test(j){
				$.getJSON("http://buy.jd.com/purchase/flows/easybuy/FlowService.ashx?action=GetCitys&Id="+j+"&callback=?", function(data){
					var Areas = data.Obj.Areas;
					//console.log(Areas);
					var str="";
					for(var i in Areas ){
					str +=&quot;insert into qly_bd_region (regionid,regionfid,regiontype,regionname)values('&quot;+Areas[i].Id+&quot;' ,'&quot;+j+&quot;','市','&quot;+Areas[i].Name+ &quot;');&lt;br/&gt;&quot; ;
					setInterval(country(Areas[i].Id),1000);
					
				}
				$(&quot;#divs&quot;).append(str);
			});
		}
		
	
	//这里的i主要表示省id
		for( var i =1;i&lt;32;i++){
			setInterval(test(42),1000);
		}
});</pre> 

以上是查询京东数据,然后直接封装成sql。(这里要注意的是json和jsonp的区别,如果用普通的ajax方法会被服务器拒绝)

2、建数据库表

REGION

3、现在可以写后台处理方法了,封装了数据结构了

helper帮助类

package com.qly.b2b.config;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import org.apache.ibatis.session.SqlSession;
import org.apache.log4j.Logger;

import com.qly.b2b.dao.config.RegionMapper;
import com.qly.b2b.model.config.Region;
import com.qly.b2b.mybatisconn.DbHelper;

public class RegionHelper {
/** 日志组件 **/
private static Logger log = Logger.getLogger(SysConfigHelper.class);

/** 数据缓存是否加载完毕 **/
private static boolean loaded = false;

/** 所有省信息 **/
private static List&lt;Region&gt; provinceList = new ArrayList&lt;Region&gt;();

/** 按省划分的市信息 **/
private static Map&lt;Integer, List&lt;Region&gt;&gt; cityMap = new HashMap&lt;Integer, List&lt;Region&gt;&gt;();

/** 按市划分的区县信息 **/
private static Map&lt;Integer, List&lt;Region&gt;&gt; countyMap = new HashMap&lt;Integer, List&lt;Region&gt;&gt;();

/** 所有地域信息 **/
private static Map&lt;Integer, Region&gt; regionMap = new HashMap&lt;Integer, Region&gt;();

/** 禁止实例化 */
private RegionHelper() {

}

/**
 * 
 * load(进行数据初始化) (这里描述这个方法适用条件 – 可选)
 * 
 * @param force
 *            是否强制加载,用于重新初始化数据 void
 * @exception
 */
private static synchronized void load(boolean force) {
	// 如果非强制初始化(重新初始化),并且已经初始化完毕,返回
	if (loaded &amp;&amp; !force) {
		return;
	}

	log.info(&quot;========================== 加载全国地址静态配置信息到缓存开始 ==========================&quot;);
	loaded = false;
	provinceList.clear();
	cityMap.clear();
	countyMap.clear();
	regionMap.clear();
	
	// 从数据库中获取数据
	List&lt;Region&gt; list = getAllRegion();

	// 将数据加载到缓存
	for (Region region : list) {
        if (&quot;1&quot;.equals(region.getRegiontype())) {
            provinceList.add(region);
        } else if (&quot;2&quot;.equals(region.getRegiontype())) {
            List&lt;Region&gt; cityList = getListFromMap(region.getRegionfid(), cityMap);
            cityList.add(region);
        } else if (&quot;3&quot;.equals(region.getRegiontype())) {
            List&lt;Region&gt; countyList = getListFromMap(region.getRegionfid(), countyMap);
            countyList.add(region);
        }
        
        regionMap.put(region.getRegionid(), region);
    }

	loaded = true;
	log.info(&quot;========================== 加载全国地址静态配置信息到缓存结束 ==========================&quot;);
}

/**
 * getAllRegion(从数据库获取所有地域信息) 
 * (这里描述这个方法适用条件 – 可选) 
 * @return  
 * List&lt;Region&gt; 
 * @exception
 */
private static List&lt;Region&gt; getAllRegion() {
    SqlSession sqlSession = DbHelper.getSqlSession();
    RegionMapper mapper = sqlSession.getMapper(RegionMapper.class);
    List&lt;Region&gt; list = mapper.getAllRegion();
    sqlSession.close();
    return list;
}

/**
 * getListFromMap(根据地域父ID从 Hash map 中获取对应 List) 
 * (这里描述这个方法适用条件 – 可选) 
 * @param regionfid
 * @param map
 * @return  
 * List&lt;Region&gt; 
 * @exception
 */
private static List&lt;Region&gt; getListFromMap(Integer regionfid, Map&lt;Integer, List&lt;Region&gt;&gt; map) {
    if (map.containsKey(regionfid)) {
        return map.get(regionfid);
    } else {
        List&lt;Region&gt; list = new ArrayList&lt;Region&gt;();
        map.put(regionfid, list);
        return list;
    }
}

/**
 * 
 * reload(重新加载数据) (这里描述这个方法适用条件 – 可选) void
 * 
 * @exception
 */
public static void reload() {
	load(true);
}

/**
 * getProvinceList(这里用一句话描述这个方法的作用) 
 * (这里描述这个方法适用条件 – 可选) 
 * @return  
 * List&lt;Region&gt; 
 * @exception
 */
public static List&lt;Region&gt; getProvinceList() {
    if (!loaded) {
        load(false);
    }
    
    return provinceList;
}

/**
 * 根据省id,拿到对应的城市信息
 */
public static List&lt;Region&gt; getCityList(Integer provinceId) {
	if (!loaded) {
	    load(false);
	}

	return cityMap.get(provinceId);
}

/**
 * 根据市id拿到对应的县
 * 
 * @param cityId
 * @return
 */
public static List&lt;Region&gt; getCountyMap(Integer cityId) {
    if (!loaded) {
        load(false);
    }

	return countyMap.get(cityId);
}

/**
 * getRegion(根据地域ID获取地域信息) 
 * (这里描述这个方法适用条件 – 可选) 
 * @param regionId
 * @return  
 * Region 
 * @exception
 */
public static Region getRegion(Integer regionId) {
    if (!loaded) {
        load(false);
    }

    return regionMap.get(regionId);
}

}

controller类

@Controller
@RequestMapping("/region")
public class RegionController extends BaseController {
@RequestMapping(value = &quot;/getallprovince&quot;)
@ResponseBody
public ModelMap getallprovince(HttpServletRequest request, ModelMap modelMap)
		throws Exception {
	List&lt; Region&gt; provincelist = RegionHelper.getProvinceList();
	modelMap.put(&quot;provincelist&quot;, provincelist);
	return modelMap;
}

@RequestMapping(value = &quot;/getcity&quot;)
@ResponseBody
public ModelMap getcity(HttpServletRequest request, ModelMap modelMap)
		throws Exception {
	List&lt; Region&gt; citylist = RegionHelper.getCityList(Integer.parseInt(request.getParameter(&quot;id&quot;)));
	modelMap.put(&quot;citylist&quot;, citylist);
	return modelMap;
}

@RequestMapping(value = &quot;/getcounty&quot;)
@ResponseBody
public ModelMap getcounty(HttpServletRequest request, ModelMap modelMap)
		throws Exception {
	List&lt; Region&gt; countylist = RegionHelper.getCountyMap(Integer.parseInt(request.getParameter(&quot;id&quot;)));
	modelMap.put(&quot;countylist&quot;, countylist);
	return modelMap;
}

}

4、处理前台js展现

$(document).ready(function(){
	//获得省列表,初始化第一个 select
	$.getJSON(basepath+"/region/getallprovince", function(data){
	  setOptions("province", data.provincelist);
	});
});

//获取城市列表
function getCitys(){
setAddress();

var cityid = $(&quot;#province&quot;).find(&quot;option:selected&quot;).val();
if (cityid == &quot;&quot;)
{
	clearSelect(&quot;city&quot;);
	return;
}
$.getJSON(basepath+&quot;/region/getcity&quot;,{id:cityid}, function(data){
	 setOptions(&quot;city&quot;, data.citylist);
	});

  //清空地区列表
  clearSelect(&quot;county&quot;);

// });
}

//获取地区列表
function getAreas(){
setAddress();
var countyid = $("#city").find("option:selected").val();
if (countyid == "")
{
clearSelect("county");
return;
}
$.getJSON(basepath+"/region/getcounty",{id:countyid}, function(data){
setOptions("county", data.countylist);
});
}

//清空下拉列表
function clearSelect(id){
$("#" + id).empty() ;
$("#" + id).append("<option value=''>请选择</option>");
}

//将数据放到指定 select 中
function setOptions(id, datas) {
$("#" + id).empty() ;
$("#" + id).append("<option value=''>请选择</option>");
for (var i in datas )
{
$("#" + id).append("<option value='" + datas[i].regionid + "'>" + datas[i].regionname + "</option>");
}
}

function setAddress(){
var address = getOptionName("province") + getOptionName("city") + getOptionName("county");
$("#address1").html(address);
var address1 = $("#address1").text();
$("#provinceaddress").val(address1);
}

function getOptionName(id) {
if($("#" + id).find("option:selected").val() != ""){
return $("#" + id).find("option:selected").html();
}

return &quot;&quot;;

}

<select name="province" id="province" onchange="getCitys()">
<option value=''>请选择</option>
</select> <select name="city" id="city" onchange="getAreas()">
<option value=''>请选择</option>
</select> <select name="county" id="county"
onchange="setAddress()">
<option value=''>请选择</option>
</select>

<span id="address1"></span></span>

5、到此为止,一个js城市选择的插件就ok了

(三)技术点

      主要是封装数据结构,和获取数据。

(四)检查数据

SELECT a.regionid,a.regionname, b.regionid,b.regionname FROM qly_bd_region a LEFT JOIN qly_bd_region b ON b.regionfid = a.regionid WHERE b.regionid IS  NULL AND a.regiontype IN (1,2)

这条sql语句可以帮助你,数据是否正确。主要用了左连接,然后id和fid就可以找出数据是否缺失

  • JavaScript

    JavaScript 一种动态类型、弱类型、基于原型的直译式脚本语言,内置支持类型。它的解释器被称为 JavaScript 引擎,为浏览器的一部分,广泛用于客户端的脚本语言,最早是在 HTML 网页上使用,用来给 HTML 网页增加动态功能。

    728 引用 • 1275 回帖 • 1 关注
  • jQuery

    jQuery 是一套跨浏览器的 JavaScript 库,强化 HTML 与 JavaScript 之间的操作。由 John Resig 在 2006 年 1 月的 BarCamp NYC 上释出第一个版本。全球约有 28% 的网站使用 jQuery,是非常受欢迎的 JavaScript 库。

    63 引用 • 134 回帖 • 727 关注

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...

推荐标签 标签

  • DNSPod

    DNSPod 建立于 2006 年 3 月份,是一款免费智能 DNS 产品。 DNSPod 可以为同时有电信、网通、教育网服务器的网站提供智能的解析,让电信用户访问电信的服务器,网通的用户访问网通的服务器,教育网的用户访问教育网的服务器,达到互联互通的效果。

    6 引用 • 26 回帖 • 519 关注
  • flomo

    flomo 是新一代 「卡片笔记」 ,专注在碎片化时代,促进你的记录,帮你积累更多知识资产。

    5 引用 • 107 回帖 • 1 关注
  • 职场

    找到自己的位置,萌新烦恼少。

    127 引用 • 1706 回帖
  • Wide

    Wide 是一款基于 Web 的 Go 语言 IDE。通过浏览器就可以进行 Go 开发,并有代码自动完成、查看表达式、编译反馈、Lint、实时结果输出等功能。

    欢迎访问我们运维的实例: https://wide.b3log.org

    30 引用 • 218 回帖 • 635 关注
  • Electron

    Electron 基于 Chromium 和 Node.js,让你可以使用 HTML、CSS 和 JavaScript 构建应用。它是一个由 GitHub 及众多贡献者组成的活跃社区共同维护的开源项目,兼容 Mac、Windows 和 Linux,它构建的应用可在这三个操作系统上面运行。

    15 引用 • 136 回帖 • 1 关注
  • danl
    147 关注
  • ReactiveX

    ReactiveX 是一个专注于异步编程与控制可观察数据(或者事件)流的 API。它组合了观察者模式,迭代器模式和函数式编程的优秀思想。

    1 引用 • 2 回帖 • 161 关注
  • Ubuntu

    Ubuntu(友帮拓、优般图、乌班图)是一个以桌面应用为主的 Linux 操作系统,其名称来自非洲南部祖鲁语或豪萨语的“ubuntu”一词,意思是“人性”、“我的存在是因为大家的存在”,是非洲传统的一种价值观,类似华人社会的“仁爱”思想。Ubuntu 的目标在于为一般用户提供一个最新的、同时又相当稳定的主要由自由软件构建而成的操作系统。

    126 引用 • 169 回帖
  • 支付宝

    支付宝是全球领先的独立第三方支付平台,致力于为广大用户提供安全快速的电子支付/网上支付/安全支付/手机支付体验,及转账收款/水电煤缴费/信用卡还款/AA 收款等生活服务应用。

    29 引用 • 347 回帖 • 2 关注
  • PWL

    组织简介

    用爱发电 (Programming With Love) 是一个以开源精神为核心的民间开源爱好者技术组织,“用爱发电”象征开源与贡献精神,加入组织,代表你将遵守组织的“个人开源爱好者”的各项条款。申请加入:用爱发电组织邀请帖
    用爱发电组织官网:https://programmingwithlove.stackoverflow.wiki/

    用爱发电组织的核心驱动力:

    • 遵守开源守则,体现开源&贡献精神:以分享为目的,拒绝非法牟利。
    • 自我保护:使用适当的 License 保护自己的原创作品。
    • 尊重他人:不以各种理由、各种漏洞进行未经允许的抄袭、散播、洩露;以礼相待,尊重所有对社区做出贡献的开发者;通过他人的分享习得知识,要留下足迹,表示感谢。
    • 热爱编程、热爱学习:加入组织,热爱编程是首当其要的。我们欢迎热爱讨论、分享、提问的朋友,也同样欢迎默默成就的朋友。
    • 倾听:正确并恳切对待、处理问题与建议,及时修复开源项目的 Bug ,及时与反馈者沟通。不抬杠、不无视、不辱骂。
    • 平视:不诋毁、轻视、嘲讽其他开发者,主动提出建议、施以帮助,以和谐为本。只要他人肯努力,你也可能会被昔日小看的人所超越,所以请保持谦虚。
    • 乐观且活跃:你的努力决定了你的高度。不要放弃,多年后回头俯瞰,才会发现自己已经成就往日所仰望的水平。积极地将项目开源,帮助他人学习、改进,自己也会获得相应的提升、成就与成就感。
    1 引用 • 487 回帖 • 2 关注
  • Sillot

    Insights(注意当前设置 master 为默认分支)

    汐洛彖夲肜矩阵(Sillot T☳Converbenk Matrix),致力于服务智慧新彖乄,具有彖乄驱动、极致优雅、开发者友好的特点。其中汐洛绞架(Sillot-Gibbet)基于自思源笔记(siyuan-note),前身是思源笔记汐洛版(更早是思源笔记汐洛分支),是智慧新录乄终端(多端融合,移动端优先)。

    主仓库地址:Hi-Windom/Sillot

    文档地址:sillot.db.sc.cn

    注意事项:

    1. ⚠️ 汐洛仍在早期开发阶段,尚不稳定
    2. ⚠️ 汐洛并非面向普通用户设计,使用前请了解风险
    3. ⚠️ 汐洛绞架基于思源笔记,开发者尽最大努力与思源笔记保持兼容,但无法实现 100% 兼容
    29 引用 • 25 回帖 • 90 关注
  • 自由行
    4 关注
  • 服务器

    服务器,也称伺服器,是提供计算服务的设备。由于服务器需要响应服务请求,并进行处理,因此一般来说服务器应具备承担服务并且保障服务的能力。

    125 引用 • 588 回帖 • 1 关注
  • Scala

    Scala 是一门多范式的编程语言,集成面向对象编程和函数式编程的各种特性。

    13 引用 • 11 回帖 • 141 关注
  • SMTP

    SMTP(Simple Mail Transfer Protocol)即简单邮件传输协议,它是一组用于由源地址到目的地址传送邮件的规则,由它来控制信件的中转方式。SMTP 协议属于 TCP/IP 协议簇,它帮助每台计算机在发送或中转信件时找到下一个目的地。

    4 引用 • 18 回帖 • 622 关注
  • 面试

    面试造航母,上班拧螺丝。多面试,少加班。

    325 引用 • 1395 回帖 • 1 关注
  • Google

    Google(Google Inc.,NASDAQ:GOOG)是一家美国上市公司(公有股份公司),于 1998 年 9 月 7 日以私有股份公司的形式创立,设计并管理一个互联网搜索引擎。Google 公司的总部称作“Googleplex”,它位于加利福尼亚山景城。Google 目前被公认为是全球规模最大的搜索引擎,它提供了简单易用的免费服务。不作恶(Don't be evil)是谷歌公司的一项非正式的公司口号。

    49 引用 • 192 回帖 • 1 关注
  • Webswing

    Webswing 是一个能将任何 Swing 应用通过纯 HTML5 运行在浏览器中的 Web 服务器,详细介绍请看 将 Java Swing 应用变成 Web 应用

    1 引用 • 15 回帖 • 638 关注
  • ActiveMQ

    ActiveMQ 是 Apache 旗下的一款开源消息总线系统,它完整实现了 JMS 规范,是一个企业级的消息中间件。

    19 引用 • 13 回帖 • 669 关注
  • OpenResty

    OpenResty 是一个基于 NGINX 与 Lua 的高性能 Web 平台,其内部集成了大量精良的 Lua 库、第三方模块以及大多数的依赖项。用于方便地搭建能够处理超高并发、扩展性极高的动态 Web 应用、Web 服务和动态网关。

    17 引用 • 41 关注
  • GraphQL

    GraphQL 是一个用于 API 的查询语言,是一个使用基于类型系统来执行查询的服务端运行时(类型系统由你的数据定义)。GraphQL 并没有和任何特定数据库或者存储引擎绑定,而是依靠你现有的代码和数据支撑。

    4 引用 • 3 回帖 • 10 关注
  • InfluxDB

    InfluxDB 是一个开源的没有外部依赖的时间序列数据库。适用于记录度量,事件及实时分析。

    2 引用 • 79 关注
  • NGINX

    NGINX 是一个高性能的 HTTP 和反向代理服务器,也是一个 IMAP/POP3/SMTP 代理服务器。 NGINX 是由 Igor Sysoev 为俄罗斯访问量第二的 Rambler.ru 站点开发的,第一个公开版本 0.1.0 发布于 2004 年 10 月 4 日。

    313 引用 • 547 回帖
  • BAE

    百度应用引擎(Baidu App Engine)提供了 PHP、Java、Python 的执行环境,以及云存储、消息服务、云数据库等全面的云服务。它可以让开发者实现自动地部署和管理应用,并且提供动态扩容和负载均衡的运行环境,让开发者不用考虑高成本的运维工作,只需专注于业务逻辑,大大降低了开发者学习和迁移的成本。

    19 引用 • 75 回帖 • 651 关注
  • TGIF

    Thank God It's Friday! 感谢老天,总算到星期五啦!

    288 引用 • 4485 回帖 • 664 关注
  • Log4j

    Log4j 是 Apache 开源的一款使用广泛的 Java 日志组件。

    20 引用 • 18 回帖 • 30 关注
  • 负能量

    上帝为你关上了一扇门,然后就去睡觉了....努力不一定能成功,但不努力一定很轻松 (° ー °〃)

    88 引用 • 1235 回帖 • 407 关注