使用AJAX異步通信技術實現(xiàn)搜索聯(lián)想和自動補全示例
一:AJAX實現(xiàn)搜索聯(lián)想和自動補全
(1)實現(xiàn)的原理
什么是搜索聯(lián)想?自動補全?
①百度是一個很典型的代表。在百度的搜索框中輸入相關信息的時候,會有搜索聯(lián)想以及自動補全。
②搜索聯(lián)想和自動補全:實際上是為了方便用戶的使用,讓用戶的體驗更好。
③搜索聯(lián)想:當用戶輸入一些單詞之后,自動聯(lián)想出用戶要搜索的信息,給一個提示。
④自動補全:當聯(lián)想出一些內(nèi)容之后,用戶點擊某個聯(lián)想的單詞,然后將這個單詞自動補全到搜索框當中。
⑤搜索聯(lián)想和自動補全功能,因為是頁面局部刷新效果,所以需要使用ajax請求來完成。
搜索聯(lián)想,自動補全功能的核心實現(xiàn)原理?
①當鍵盤事件發(fā)生之后,比如keyup:鍵彈起事件。
②會發(fā)送ajax請求,請求中提交用戶輸入的搜索內(nèi)容,例如:北京(發(fā)送ajax請求,攜帶“北京”兩個字)。
③后端接收到ajax請求,接收到“北京”兩個字,執(zhí)行select語句進行模糊查詢,返回查詢結果
④將查詢結果封裝成json格式的字符串,將json格式的字符串響應到前端。
⑤前端接收到json格式的字符串之后,解析這個json字符串,動態(tài)展示頁面。
(2)頁面實現(xiàn)
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>ajax實現(xiàn)搜索聯(lián)想和自動補全功能</title>
<style type="text/css">
/* 類選擇器,設置input標簽,text邊框*/
.userInput {
width: 300px; /*框?qū)?/
height: 25px; /*框高*/
font-size: 20px; /*里面字大小*/
padding-left: 5px; /*內(nèi)補丁,距離框的距離*/
}
/*類選擇器,設置div的樣式*/
.showDataDiv {
width: 309px; /*寬度*/
border: 1px solid lightgray; /*實線邊框*/
background-color: antiquewhite; /*設置背景顏色*/
display: none; /*設置開始的div是隱藏的,不顯示*/
}
/*設置p標簽*/
.showDataDiv p {
padding-left: 5px; /*內(nèi)補丁,距離框的距離*/
margin-top: 2px; /*外補丁,p距離頂部的寬度*/
margin-bottom: 2px; /*外補丁,p距離低部的寬度*/
}
/*p標簽增加動作,點到其中一個選項變色并變成小手*/
.showDataDiv p:hover {
cursor: pointer; /*鼠標變成小手*/
border: 1px blue solid; /*每選中一行,增加實線邊框*/
background-color: aliceblue; /*設置變換的背景色*/
}
</style>
</head>
<body>
<!--文本框-->
<input type="text" class="userInput" id="keywords">
<!--div盒子-->
<div id="datadiv" class="showDataDiv">
<!--<p>北京疫情最新情況</p>
<p>北京天氣</p>
<p>北京時間</p>
<p>北京人</p>-->
</div>
</body>
</html>頁面展示

(3)設置數(shù)據(jù)庫表
drop table if exists t_ajax;
create table t_ajax(
id int primary key auto_increment,
content varchar(255)
);
insert into t_ajax(content) values('javascript');
insert into t_ajax(content) values('javaweb');
insert into t_ajax(content) values('java');
insert into t_ajax(content) values('java123');
insert into t_ajax(content) values('mysql');
insert into t_ajax(content) values('myweb');
insert into t_ajax(content) values('myapp');
insert into t_ajax(content) values('jdk');
commit;
select * from t_ajax;效果展示

(4)前端代碼實現(xiàn)
①只要keyup鍵盤事件發(fā)生,就發(fā)送Ajax請求,把輸入的數(shù)據(jù)(this.value)發(fā)送出去
②首先就要連接數(shù)據(jù)庫進行模糊查詢,實現(xiàn)搜索聯(lián)想功能
③然后我們點擊div的<p>標簽的內(nèi)容時,會進行自動補全到文本框
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>ajax實現(xiàn)搜索聯(lián)想和自動補全功能</title>
<style type="text/css">
/* 類選擇器,設置input標簽,text邊框*/
.userInput {
width: 300px; /*框?qū)?/
height: 25px; /*框高*/
font-size: 20px; /*里面字大小*/
padding-left: 5px; /*內(nèi)補丁,距離框的距離*/
}
/*類選擇器,設置div的樣式*/
.showDataDiv {
width: 309px; /*寬度*/
border: 1px solid lightgray; /*實線邊框*/
background-color: antiquewhite; /*設置背景顏色*/
display: none; /*設置開始的div是隱藏的,不顯示*/
}
/*設置p標簽*/
.showDataDiv p {
padding-left: 5px; /*內(nèi)補丁,距離框的距離*/
margin-top: 2px; /*外補丁,p距離頂部的寬度*/
margin-bottom: 2px; /*外補丁,p距離低部的寬度*/
}
/*p標簽增加動作,點到其中一個選項變色并變成小手*/
.showDataDiv p:hover {
cursor: pointer; /*鼠標變成小手*/
border: 1px blue solid; /*每選中一行,增加實線邊框*/
background-color: aliceblue; /*設置變換的背景色*/
}
</style>
</head>
<body>
<script type="text/javascript">
window.onload = function() {
document.getElementById("keywords").onkeyup = function(){
if (this.value == "") {
// 如果為空串,就把div隱藏起來
// 不然查詢聯(lián)想之后,刪除查詢的內(nèi)容,下面div還是保持原狀
document.getElementById("datadiv").style.display = "none"
}else{
// 發(fā)送ajax請求
// 1. 創(chuàng)建AJAX核心對象
var xmlHttpRequest = new XMLHttpRequest();
// 2. 注冊回調(diào)函數(shù)
xmlHttpRequest.onreadystatechange = function() {
if (xmlHttpRequest.readyState == 4) {
if (xmlHttpRequest.status >= 200 && xmlHttpRequest.status < 300) {
// [{"content":"javascript"},{"content":"javaweb"},{"content":"java..."}]
var json = JSON.parse(xmlHttpRequest.responseText);
// 遍歷數(shù)組
var html = ""
for (var i = 0; i < json.length; i++) {
// 點擊p標簽執(zhí)行一個回調(diào)函數(shù),把內(nèi)容顯示到文本框中
html += "<p onclick='setInput(\""+json[i].content+"\")'>"+json[i].content+"</p>"
}
// 讓數(shù)據(jù)在div展示出來
document.getElementById("datadiv").innerHTML = html
// 讓div顯示出來
document.getElementById("datadiv").style.display = "block"
}
}
}
// 3. 開啟通道,并把數(shù)據(jù)傳過去,連接數(shù)據(jù)庫進行模糊查詢
xmlHttpRequest.open("GET", "/auto_complete/query?_="+new Date().getTime()+"&keywords=" + this.value, true)
// 4. 發(fā)送請求
xmlHttpRequest.send()
}
}
}
// 實現(xiàn)自動補全功能
function setInput(content){
// 先把數(shù)據(jù)顯示到文本框中
document.getElementById("keywords").value = content
// 顯示到文本框后,再次把div進行隱藏
document.getElementById("datadiv").style.display = "none"
}
</script>
<!--文本框-->
<input type="text" class="userInput" id="keywords">
<!--div盒子-->
<div id="datadiv" class="showDataDiv">
<!--<p>北京疫情最新情況</p>
<p>北京天氣</p>
<p>北京時間</p>
<p>北京人</p>-->
</div>
</body>
</html>(5)后端代碼實現(xiàn)
主要是根據(jù)發(fā)過來的數(shù)據(jù),連接數(shù)據(jù)庫進行模糊查詢;把查詢結果拼成JSON格式的字符串
package com.bjpowernode.zl;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.sql.*;
import java.util.ArrayList;
import java.util.List;
@WebServlet("/query")
public class QueryServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
// 獲取用戶輸入的關鍵字
String keywords = request.getParameter("keywords");
// jdbc代碼連接數(shù)據(jù)庫,根據(jù)關鍵字查詢數(shù)據(jù)庫,返回數(shù)據(jù),拼接json格式的字符串
Connection conn = null;
PreparedStatement ps = null;
ResultSet rs = null;
// 拼成JSON格式的字符串
StringBuilder sb = new StringBuilder();
sb.append("[");
try {
// 注冊驅(qū)動
Class.forName("com.mysql.jdbc.Driver");
// 獲取連接
String url = "jdbc:mysql://localhost:3306/bjpowernode?useUnicode=true&characterEncoding=UTF-8";
String user = "root";
String password = "***";
conn = DriverManager.getConnection(url, user, password);
String sql = "select content from t_ajax where content like ?"; // 模糊查詢的時候,條件不建議使用%開始,因為會讓字段上的索引失效,查詢效率降低。
ps = conn.prepareStatement(sql);
ps.setString(1, keywords + "%");
rs = ps.executeQuery();
// [{"content":"javascript"},{"content":"javaweb"},{"content":"java..."}]
while (rs.next()) {
String content = rs.getString("content");
sb.append("{\"content\":\""+content+"\"},");
}
}catch(Exception e){
e.printStackTrace();
} finally {
if (conn != null) {
try {
conn.close();
} catch (SQLException e) {
e.printStackTrace();
}
}
if (ps != null) {
try {
ps.close();
} catch (SQLException e) {
e.printStackTrace();
}
}
if (rs != null) {
try {
rs.close();
} catch (SQLException e) {
e.printStackTrace();
}
}
}
// 最后會多一個逗號,進行截串
response.setContentType("text/html;charset=UTF-8");
response.getWriter().print(sb.subSequence(0, sb.length() - 1) + "]");
}
}(6)動圖效果展示

二:HTTP狀態(tài)信息
1xx: 信息
消息:描述:100 Continue服務器僅接收到部分請求,但是一旦服務器并沒有拒絕該請求,客戶端應該繼續(xù)發(fā)送其余的請求。101 Switching Protocols服務器轉(zhuǎn)換協(xié)議:服務器將遵從客戶的請求轉(zhuǎn)換到另外一種協(xié)議。
2xx: 成功
消息:描述:200 OK請求成功(其后是對GET和POST請求的應答文檔。)201 Created請求被創(chuàng)建完成,同時新的資源被創(chuàng)建。202 Accepted供處理的請求已被接受,但是處理未完成。203 Non-authoritative Information文檔已經(jīng)正常地返回,但一些應答頭可能不正確,因為使用的是文檔的拷貝。204 No Content沒有新文檔。瀏覽器應該繼續(xù)顯示原來的文檔。如果用戶定期地刷新頁面,而Servlet可以確定用戶文檔足夠新,這個狀態(tài)代碼是很有用的。205 Reset Content沒有新文檔。但瀏覽器應該重置它所顯示的內(nèi)容。用來強制瀏覽器清除表單輸入內(nèi)容。206 Partial Content客戶發(fā)送了一個帶有Range頭的GET請求,服務器完成了它。
3xx: 重定向
消息:描述:300 Multiple Choices多重選擇。鏈接列表。用戶可以選擇某鏈接到達目的地。最多允許五個地址。301 Moved Permanently所請求的頁面已經(jīng)轉(zhuǎn)移至新的url。302 Found所請求的頁面已經(jīng)臨時轉(zhuǎn)移至新的url。303 See Other所請求的頁面可在別的url下被找到。304 Not Modified未按預期修改文檔。客戶端有緩沖的文檔并發(fā)出了一個條件性的請求(一般是提供If-Modified-Since頭表示客戶只想比指定日期更新的文檔)。服務器告訴客戶,原來緩沖的文檔還可以繼續(xù)使用。305 Use Proxy客戶請求的文檔應該通過Location頭所指明的代理服務器提取。306 Unused此代碼被用于前一版本。目前已不再使用,但是代碼依然被保留。307 Temporary Redirect被請求的頁面已經(jīng)臨時移至新的url。
4xx: 客戶端錯誤
消息:描述:400 Bad Request服務器未能理解請求。401 Unauthorized被請求的頁面需要用戶名和密碼。402 Payment Required此代碼尚無法使用。403 Forbidden對被請求頁面的訪問被禁止。404 Not Found服務器無法找到被請求的頁面。405 Method Not Allowed請求中指定的方法不被允許。406 Not Acceptable服務器生成的響應無法被客戶端所接受。407 Proxy Authentication Required用戶必須首先使用代理服務器進行驗證,這樣請求才會被處理。408 Request Timeout請求超出了服務器的等待時間。409 Conflict由于沖突,請求無法被完成。410 Gone被請求的頁面不可用。411 Length Required"Content-Length" 未被定義。如果無此內(nèi)容,服務器不會接受請求。412 Precondition Failed請求中的前提條件被服務器評估為失敗。413 Request Entity Too Large由于所請求的實體的太大,服務器不會接受請求。414 Request-url Too Long由于url太長,服務器不會接受請求。當post請求被轉(zhuǎn)換為帶有很長的查詢信息的get請求時,就會發(fā)生這種情況。415 Unsupported Media Type由于媒介類型不被支持,服務器不會接受請求。416服務器不能滿足客戶在請求中指定的Range頭。417 Expectation Failed
5xx: 服務器錯誤
消息:描述:500 Internal Server Error請求未完成。服務器遇到不可預知的情況。501 Not Implemented請求未完成。服務器不支持所請求的功能。502 Bad Gateway請求未完成。服務器從上游服務器收到一個無效的響應。503 Service Unavailable請求未完成。服務器臨時過載或當機。504 Gateway Timeout網(wǎng)關超時。505 HTTP Version Not Supported服務器不支持請求中指明的HTTP協(xié)議版本。
到此這篇關于使用AJAX異步通信技術實現(xiàn)搜索聯(lián)想和自動補全示例的文章就介紹到這了,更多相關AJAX實現(xiàn)搜索聯(lián)想和自動補全內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
jquery1.8版本使用ajax實現(xiàn)微信調(diào)用出現(xiàn)的問題分析及解決辦法
這篇文章主要介紹了jquery1.8版本使用ajax實現(xiàn)微信調(diào)用出現(xiàn)的問題分析及解決辦法的相關資料,需要的朋友可以參考下2015-11-11
Servlet 與 Ajax 交互一直報status=parsererror的解決辦法
這篇文章主要介紹了Servlet 與 Ajax 交互一直報status=parsererror的解決辦法,非常不錯,具有參考借鑒價值,需要的朋友可以參考下2017-03-03

