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

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

(一)场景

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

(二)准备工作

         最主要的就是数据了,去几个网站逛了一圈,感觉京东的不错,于是就想办法把里面的数据拉下来。京东页面 ,其实很简单,用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 网页增加动态功能。

    729 引用 • 1327 回帖
  • jQuery

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

    63 引用 • 134 回帖 • 722 关注

相关帖子

欢迎来到这里!

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

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

推荐标签 标签

  • CodeMirror
    1 引用 • 2 回帖 • 131 关注
  • GAE

    Google App Engine(GAE)是 Google 管理的数据中心中用于 WEB 应用程序的开发和托管的平台。2008 年 4 月 发布第一个测试版本。目前支持 Python、Java 和 Go 开发部署。全球已有数十万的开发者在其上开发了众多的应用。

    14 引用 • 42 回帖 • 764 关注
  • Redis

    Redis 是一个开源的使用 ANSI C 语言编写、支持网络、可基于内存亦可持久化的日志型、Key-Value 数据库,并提供多种语言的 API。从 2010 年 3 月 15 日起,Redis 的开发工作由 VMware 主持。从 2013 年 5 月开始,Redis 的开发由 Pivotal 赞助。

    286 引用 • 248 回帖 • 62 关注
  • FFmpeg

    FFmpeg 是一套可以用来记录、转换数字音频、视频,并能将其转化为流的开源计算机程序。

    23 引用 • 32 回帖
  • Spark

    Spark 是 UC Berkeley AMP lab 所开源的类 Hadoop MapReduce 的通用并行框架。Spark 拥有 Hadoop MapReduce 所具有的优点;但不同于 MapReduce 的是 Job 中间输出结果可以保存在内存中,从而不再需要读写 HDFS,因此 Spark 能更好地适用于数据挖掘与机器学习等需要迭代的 MapReduce 的算法。

    74 引用 • 46 回帖 • 553 关注
  • 链滴

    链滴是一个记录生活的地方。

    记录生活,连接点滴

    153 引用 • 3783 回帖 • 1 关注
  • 禅道

    禅道是一款国产的开源项目管理软件,她的核心管理思想基于敏捷方法 scrum,内置了产品管理和项目管理,同时又根据国内研发现状补充了测试管理、计划管理、发布管理、文档管理、事务管理等功能,在一个软件中就可以将软件研发中的需求、任务、bug、用例、计划、发布等要素有序的跟踪管理起来,完整地覆盖了项目管理的核心流程。

    6 引用 • 15 回帖 • 116 关注
  • GitHub

    GitHub 于 2008 年上线,目前,除了 Git 代码仓库托管及基本的 Web 管理界面以外,还提供了订阅、讨论组、文本渲染、在线文件编辑器、协作图谱(报表)、代码片段分享(Gist)等功能。正因为这些功能所提供的便利,又经过长期的积累,GitHub 的用户活跃度很高,在开源世界里享有深远的声望,并形成了社交化编程文化(Social Coding)。

    209 引用 • 2031 回帖
  • 音乐

    你听到信仰的声音了么?

    60 引用 • 511 回帖
  • Android

    Android 是一种以 Linux 为基础的开放源码操作系统,主要使用于便携设备。2005 年由 Google 收购注资,并拉拢多家制造商组成开放手机联盟开发改良,逐渐扩展到到平板电脑及其他领域上。

    334 引用 • 323 回帖 • 2 关注
  • VirtualBox

    VirtualBox 是一款开源虚拟机软件,最早由德国 Innotek 公司开发,由 Sun Microsystems 公司出品的软件,使用 Qt 编写,在 Sun 被 Oracle 收购后正式更名成 Oracle VM VirtualBox。

    10 引用 • 2 回帖 • 6 关注
  • 生活

    生活是指人类生存过程中的各项活动的总和,范畴较广,一般指为幸福的意义而存在。生活实际上是对人生的一种诠释。生活包括人类在社会中与自己息息相关的日常活动和心理影射。

    230 引用 • 1454 回帖
  • danl
    133 关注
  • 七牛云

    七牛云是国内领先的企业级公有云服务商,致力于打造以数据为核心的场景化 PaaS 服务。围绕富媒体场景,七牛先后推出了对象存储,融合 CDN 加速,数据通用处理,内容反垃圾服务,以及直播云服务等。

    27 引用 • 225 回帖 • 168 关注
  • JWT

    JWT(JSON Web Token)是一种用于双方之间传递信息的简洁的、安全的表述性声明规范。JWT 作为一个开放的标准(RFC 7519),定义了一种简洁的,自包含的方法用于通信双方之间以 JSON 的形式安全的传递信息。

    20 引用 • 15 回帖 • 3 关注
  • API

    应用程序编程接口(Application Programming Interface)是一些预先定义的函数,目的是提供应用程序与开发人员基于某软件或硬件得以访问一组例程的能力,而又无需访问源码,或理解内部工作机制的细节。

    77 引用 • 430 回帖
  • Solo

    Solo 是一款小而美的开源博客系统,专为程序员设计。Solo 有着非常活跃的社区,可将文章作为帖子推送到社区,来自社区的回帖将作为博客评论进行联动(具体细节请浏览 B3log 构思 - 分布式社区网络)。

    这是一种全新的网络社区体验,让热爱记录和分享的你不再感到孤单!

    1434 引用 • 10054 回帖 • 489 关注
  • C

    C 语言是一门通用计算机编程语言,应用广泛。C 语言的设计目标是提供一种能以简易的方式编译、处理低级存储器、产生少量的机器码以及不需要任何运行环境支持便能运行的编程语言。

    85 引用 • 165 回帖 • 4 关注
  • 持续集成

    持续集成(Continuous Integration)是一种软件开发实践,即团队开发成员经常集成他们的工作,通过每个成员每天至少集成一次,也就意味着每天可能会发生多次集成。每次集成都通过自动化的构建(包括编译,发布,自动化测试)来验证,从而尽早地发现集成错误。

    15 引用 • 7 回帖 • 1 关注
  • JSON

    JSON (JavaScript Object Notation)是一种轻量级的数据交换格式。易于人类阅读和编写。同时也易于机器解析和生成。

    52 引用 • 190 回帖 • 1 关注
  • 深度学习

    深度学习(Deep Learning)是机器学习的分支,是一种试图使用包含复杂结构或由多重非线性变换构成的多个处理层对数据进行高层抽象的算法。

    53 引用 • 40 回帖
  • ngrok

    ngrok 是一个反向代理,通过在公共的端点和本地运行的 Web 服务器之间建立一个安全的通道。

    7 引用 • 63 回帖 • 624 关注
  • GitLab

    GitLab 是利用 Ruby 一个开源的版本管理系统,实现一个自托管的 Git 项目仓库,可通过 Web 界面操作公开或私有项目。

    46 引用 • 72 回帖 • 2 关注
  • 运维

    互联网运维工作,以服务为中心,以稳定、安全、高效为三个基本点,确保公司的互联网业务能够 7×24 小时为用户提供高质量的服务。

    149 引用 • 257 回帖
  • Gzip

    gzip (GNU zip)是 GNU 自由软件的文件压缩程序。我们在 Linux 中经常会用到后缀为 .gz 的文件,它们就是 Gzip 格式的。现今已经成为互联网上使用非常普遍的一种数据压缩格式,或者说一种文件格式。

    9 引用 • 12 回帖 • 134 关注
  • 导航

    各种网址链接、内容导航。

    40 引用 • 173 回帖 • 1 关注
  • 程序员

    程序员是从事程序开发、程序维护的专业人员。

    568 引用 • 3532 回帖