用 PHPRPC 实现 Ajax 级联下拉菜单 打印 E-mail
级联下拉菜单就是从一个下拉菜单中选中一项后,相应的另一个下拉菜单的内容会随之改变。

一般来说,最简单的,就是每次选中都提交一次表单,刷新整个页面。这也是用户体验度最差的。

另一种是把所有选项在第一次加载时就全部载入整个页面中的 javascript 数组中,然后级联通过 JavaScript 来控制,在整个数据量不大时,这是一个不错的实现无刷新并且快速的方法,但是当整个数据量非常大时,这种方法就会使第一次加载变得非常慢了。

还有就是采用 Ajax 方式,即开始只载入第一层菜单的内容,当用户选中第一层菜单的某项时,再通过 XmlHttpRequest 来获取相应选项所对应的第二层菜单的内容。这种方式效果最好,但是采用传统方式来编写这样的 Ajax 程序代码量会比较多。而且如果设计不好,服务器端返回菜单内容的程序的可复用性也会很差。

但是在本文中你会看到用 PHPRPC 来实现这种 Ajax 效果是多么的简单,并且还会具有非常高的可复用性。


本文以省市两级级联下拉菜单为例,为了举例方便,本文中采用的是 SQLite 数据库,因为这个文件型数据库比较容易部署,而且查询效率很高(当然创建该数据库的效率不高,但创建仅一次而已,该数据库在该程序中内容是不变的),不过服务器需要安装 SQLite 扩展。

这个数据库中的表只有 2 个,一个 province 表,一个 city 表。province 表中,只有 id 和 name 两个字段,分别是省份编号(主键)和省名。city 表中,有 id、name 和 pid 三个字段,id 是城市编号,name 是城市名,pid 是城市所在省的编号,与 province 表中的省份编号相对应。

创建该数据库的程序这里就不给出来了,它包含在后面提供的实例下载中。

下面来看看创建这个程序的服务器端有多么简单,为了提高可复用性,我们把服务器端分为了 2 个文件,一个是 function.php,另一个是 rpc.php。function.php 中定义了实际的远程调用函数,但是他们也可以作为服务器端的本地函数调用,你会发现他们跟服务器端的普通函数没有任何区别:

下载: function.php
<?php
$sqlite 
= new SQLiteDatabase('area.db'
);
 
function 
get_province
() {
    global 
$sqlite
;
    
$sql "select * from province order by id"
;
    return 
$sqlite->arrayQuery($sqlSQLITE_ASSOC
);
}
 
function 
get_city($pid
) {
    global 
$sqlite
;
    
$pid sqlite_escape_string($pid
);
    
$sql "select * from city where pid = $pid order by id"
;
    return 
$sqlite->arrayQuery($sqlSQLITE_ASSOC
);
}
?> 

而 rpc.php 更加简单,它是提供给客户端调用的接口,它只有 3 行语句:

下载: function.php
<?php
require_once('function.php'
);
require_once(
'phprpc_server.php'
);
new 
phprpc_server(array('get_province''get_city'
));
?> 

其中最后一句,就是指定哪些函数要暴露给客户端。只有指定的函数客户端才可以调用,这样可以保证服务器的安全性。

服务器端到此就创建完了。你会发现服务器端只负责把数据查询出来返回给客户端就完事了,其它的不做任何处理。

那么下面该看一看客户端了,客户端虽然很简单,但是我还是把它分成了两个文件,一个 js 文件,一个 html 文件,你会发现用 PHPRPC,客户端都不需要用 PHP。

下载: area.js
// 创建 phprpc 客户端对象 rpc
phprpc_client.create('rpc'
);
 
var 
city = []; 
// 用于缓存已加载的城市数据
 
/* 
 * 清除 select 中的选项,该方法可复用
 *
 * so: 要清除选项的 select 对象
 *
 */
function clear_select(so
) {
    for (var 
so.options.length 1> -1i
--) {
        
// 有些浏览器不支持 options 属性的 remove 方法,
        // 但支持 DOM 的 removeChild 方法(比如:Konqueror)
        
if (so.options.remove
) {
            
so.options.remove(i
);
        }
        else {
            
so.removeChild(so.options[i
]);
        }
    }
}
 
/*
 * 设置 select 中的选项,该方法可复用
 *
 * so: 要设置选项的 select 对象
 *  d: 选项数据数组
 * vf: 选项值所对应的数组中的字段名
 * tf: 选项文本所对应的数组中的字段名
 */
function set_select(sodvftf
) {
    for (var 
0d.lengthni
++) {
        var 
opt document.createElement('option'
);
        
opt.text d[i][tf
];
        
opt.value d[i][vf
];
        
// 有些浏览器不支持 options 属性的 add 方法,
        // 但支持 DOM 的 appendChild 方法(比如:Konqueror)
        
if (so.options.add
) {
            
so.options.add(opt
);
        }
        else {
            
so.appendChild(opt
);
        }
    }
}
 
// 设置省份的下拉菜单
function set_province_select(d
) {
    var 
so document.getElementById('province'
);
    
set_select(sod'id''name'
);
    
// 设置首选省份的城市下拉列表
    
change_province(1
);
}
 
// 设置城市的下拉菜单
function set_city_select(dvftf
) {
    var 
so document.getElementById('city'
);
    
// 清空原有选项
    
clear_select(so
);
    
// 设置新选项
    
set_select(sodvftf
);
}
 
// 当省份改变,相应的改变城市列表
function change_province(pid
) {
    
// 如果已缓存,则直接显示缓存中的列表
    
if (city[pid
]) {
        
set_city_select(city[pid], 'id''name'
);
    }
    else {
        
// 否则,先显示载入中
        
set_city_select([['''Loading...']], 01
);
        
// 然后调用远程过程载入城市信息
        // 调用远程过程时,最后一个参数指定的是回调函数
        
rpc.get_city(pid, function (result
) {
            
// 把载入的数据放入缓存
            
city[pid] = result
;
            
// 更新城市列表
            
set_city_select(result'id''name'
);
        });
    }
}
 
// 定义当 rpc 客户端初始化(use_service)完毕后执行的内容
rpc.onready 
= function () {
    
// 调用获取省份内容的远程过程,并设置回调函数为 set_province_select
    
rpc.get_province(set_province_select
);

下载: index.html
<html>
<
head
>
<
script type="text/javascript" src="phprpc_client.js"></script
>
<
script type="text/javascript" src="area.js"></script
>
</
head
>
<
body onload="rpc.use_service('rpc.php');"
>
<
select id="province" onchange="change_province(this.value);"></select
>
<
select id="city"></select
>
</
body
>
</
html

上面的 html 中包含的 phprpc_client.js 是压缩版本(因为不需要用到加密,这里是 lite 压缩版)的,这样可以免去包含多个 js 文件的麻烦。

大家会发现这个程序不但可复用性好(比如 clear_select 和 set_select 两个函数也可以在其它程序中使用),而且使得整个程序的思路清晰,比如那个缓存功能,在这里实现的就非常的简单,而且效果也非常的好。
发表评论
网友评论
Java Games
作者 访客 于 2008-11-19 13:41:19
上海优乐加城市工业园区坐落于上海工业园,园区占地12万平方米,有节能和健康舒适的生产环境,集标准厂房、独幢研发楼等,厂房出租商具有丰富的与客服沟通合作的经验,喜获“上海工业园优秀物业管理工业区”殊荣。建有标准工业厂房12栋,科技产业园的上海厂房租办公楼租赁令生产商务成本更为集约化、经济化,另有一栋全3层上海厂房标准厂房出租,面向园区景观绿化中心,环境优美!go8hg3 hyyqqgo 
 
日前,上海优乐加城市工业园区北区一期厂房已顺利完工,可交付使用! 
此次一期完工的工业厂房办公楼租赁共计有5栋,面积范围在2690平米——4662平米不等,物业形式为生产区域二层,办公区域局部三层,全为钢砼框架结构。厂房办公区域、生产车间可分割、整合,根据客户需要自由组合,灵活多样。外墙采取全澳洲进口环保涂料,高标准低能耗,在同等光照条件下,室内外温差可达2-3摄氏度 。 
 
目前已有两家客户正式签约,入驻我们优乐加城市工业园,且均为外资企业,在同行业中都处领先地位。其中,1#厂房已经顺利开始生产运作,经营状况良好。 
 
此外,园区的二期、三期厂房也正在加紧建设之中,8栋厂房将于今年年底,明年年中分批交付使用,敬请期待!
onestoptown
作者 访客 于 2008-11-20 12:18:14
shoes online buy shoes sandal shoes slippers shoes ecco shoes rockport shoes
steve madden shoes - puma shoes -  
nike shoes  
nike shoes replica  
replica nike shoes
air jordan shoes  
replica watches rolex  
replica watch - rolex watch - fake rolex watches
oyster rolex  
rolex watches
replica watches
replica rolex watches - rolex replica - watches online
oyster rolex watch - quartz watch fake - nike shoes - nike shoes replica - replica nike shoes
air jordan shoes - replica watches rolex - replica watch - rolex watch - fake rolex watches
oyster rolex - rolex watches - replica watches - replica rolex watches - rolex replica - watches online
oyster rolex watch - quartz watch fake -  
best replica watches 
rolex watches - replica watches - replica nike shoes - jordan shoes - watches online
oyster rolex watch - quartz shoes fake -  
nike shoeses
Replica Watches Jewelry Replica HandBags
作者 访客 于 2008-11-25 21:41:59
Replica Watches,Fake Watches,Replica Watch,Fake Watch,Wholesale Watches,Wholesale Replica Watches,Jewelry Watches,Replica Jewelry Watches. 
 
Replica Watches, Replica Watches, Replica Watches, Replica Watches, Replica Watches, Replica Watches, Replica Watches, Replica Watches, Replica Watches, Replica Watches, Replica Designer Handbags, Replica Designer Handbags, Replica Handbag, Replica Handbag, Wholesale handbags, Wholesale handbags, Wholesale Replica Watches, Wholesale Replica Watches, Replica Watches, Replica Watches, Baume
Replica Watches Jewelry Replica HandBags
作者 访客 于 2008-11-26 01:12:39
Replica Watches,Fake Watches,Replica Watch,Fake Watch,Wholesale Watches,Wholesale Replica Watches,Jewelry Watches,Replica Jewelry Watches. 
 
Jewelry Wholesale, Jewelry Wholesale, Replica Watches, Replica Watches, Replica Watches, Replica Watches, Wholesale Replica Watches, Wholesale Replica Watches, Replica Handbags, Replica Handbags, Replica Watches, Replica Watches, Replica Watches, Replica Watches, Replica Designer Handbags, Replica Designer Handbags, Replica Watches, Replica Watches, Replica Watches, Replica Watches, Rolex Replica Watch, Rolex Replica Watch, Bvlgari Watches, Bvlgari Watches, Fake Rado Watch, Fake Rado Watch, Breitling Watches, Breitling Watches, Replica Corum, Replica Corum, Patek Philippe Replica Watch, Patek Philippe Replica Watch, Replica Breitling, Replica Breitling, Fake Accutron Watch, Fake Accutron Watch, Fake Chopard, Fake Chopard, Fake Chanel Watches, Fake Chanel Watches, Zenith Replica, Zenith Replica, Fake Breitling Watches, Fake Breitling Watches, Tissot Watches, Tissot Watches, Patek Philippe Watch, Patek Philippe Watch, Emporio Armani Watches, Emporio Armani Watches, Replica Zenith Watch, Replica Zenith Watch, Rado Watches, Rado Watches, Breitling Watches Replica, Breitling Watches Replica, Replica Jaeger LeCoultre Watches, Replica Jaeger LeCoultre Watches, Replica TAG Heuer Watches, Replica TAG Heuer Watches,
fghdfgj
作者 访客 于 2008-12-07 15:07:26
澳洲留学中介北京留学中介法国留学中介韩国留学韩国留学中介合法留学中介荷兰留学中介荷兰预科留学服务留学公司留学咨询公司美国留学中介欧洲留学中介英国留学中介加拿大留学中介留学申请高中生留学|澳大利亚留学澳洲留学出国出国留学法国留学荷兰留学加拿大留学加拿大签证留学留学澳大利亚留学澳洲留学法国留学荷兰留学加拿大留学美国留学新加坡留学英国留学中介美国留学新加坡留学英国留学英国签证瑞典留学翻译公司翻译公司洁身器窃听器手机窃听器监听器手机监听器窃听器手机窃听器监听器手机监听器窃听器手机窃听器监听器手机监听器窃听器手机窃听器监听器手机监听器窃听器手机窃听器监听器手机监听器办证网上办证北京办证上海办证广州办证

发表评论
用户访客
标题
BBCode:Web AddressEmail AddressBold TextItalic TextUnderlined TextQuoteCodeOpen ListList ItemClose List
评论



下一篇 >