Spring BootのWEB画面上では、Ajax通信も行うことができる。今回は、jQueryを利用しない形で、Ajax通信を含むサンプルプログラムを作成してみたので、共有する。
前提条件
下記記事のSpring BootでのOracle接続処理の実装が完了していること。
また、以下の記事の「hibernate_sequence」シーケンスの作成が完了していること。
サンプルプログラムの作成
まず、HTMLは以下の内容になっていて、検索ボタンを押下した場合に、後述のdemo.jsの関数「getUserData」を呼び出すようになっている。
<!DOCTYPE html>
<html lang="ja" xmlns:th="http://www.thymeleaf.org">
<head>
<meta charset="UTF-8">
<script type="text/javascript" th:src="@{/demo.js}"></script>
<title>index page</title>
</head>
<body>
検索IDを指定し、「検索」ボタンを押下してください。<br/>
検索ID: <input type="text" id="searchId" name="searchId" size="12" maxlength="4" />
<input type="button" value="検索" onclick="getUserData();" />
<br/><br/><br/>
★以下に検索されたデータが表示されます
<table border="0">
<tr>
<td align="left" valign="top">名前:</td>
<td>
<span id="name"></span>
</td>
</tr>
<tr>
<td align="left" valign="top">生年月日:</td>
<td>
<span id="birthDay"></span>
</td>
</tr>
<tr>
<td align="left" valign="top">性別:</td>
<td>
<span id="sex"></span>
</td>
</tr>
<tr>
<td align="left" valign="top">メモ:</td>
<td>
<span id="memo"></span>
</td>
</tr>
</table>
</body>
</html>また、demo.jsの内容は以下の通りで、検索IDの値を引数に、Ajax通信でsearchメソッドを呼び出し、その処理が正常終了した場合は、取得したUserDataオブジェクトの値を、画面上に表示するようになっている。
'use strict';
function getUserData(){
// index.htmlの検索IDの値を取得
const id = document.getElementById('searchId').value;
// AjaxにてDemoControllerクラスのsearchメソッドを呼び出す
let request = new XMLHttpRequest();
request.open("get", "/search?id=" + id, true);
request.send(null);
request.onload = function (event) {
// Ajaxが正常終了した場合
if (request.readyState === 4 && request.status === 200) {
// 該当するデータが無かった場合
if(!request.responseText){
alert("該当するデータはありませんでした");
return;
}
// 該当するデータがあった場合は、取得したUserDataオブジェクトの内容を画面に
// 表示する。その際、名前・性別・メモはデコードする
const userData = JSON.parse(request.responseText);
document.getElementById('name').textContent = decodeURI(userData.name);
document.getElementById('birthDay').textContent
= userData.birthY + '年' + userData.birthM + '月' + userData.birthD + '日';
document.getElementById('sex').textContent = decodeURI(userData.sex);
document.getElementById('memo').textContent = decodeURI(userData.memo);
// Ajaxが異常終了した場合
}else{
alert("エラーが発生し、データが取得できませんでした");
}
};
// Ajaxが異常終了した場合
request.onerror = function (event) {
alert("エラーが発生し、データが取得できませんでした");
}
}
さらに、コントローラクラスの内容は以下の通りで、Ajax通信で呼ばれるsearchメソッドは、@ResponseBodyアノテーションを付与し、取得したUserDataオブジェクトをJSON文字列に変換して返却している。
package com.example.demo;
import com.fasterxml.jackson.core.JsonProcessingException;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.ResponseBody;
import com.fasterxml.jackson.databind.ObjectMapper;
import org.thymeleaf.util.StringUtils;
import java.io.UnsupportedEncodingException;
import java.net.URLEncoder;
@Controller
public class DemoController {
/**
* ユーザーデータテーブル(user_data)へアクセスするリポジトリ
*/
@Autowired
private UserDataRepository repository;
/**
* 初期表示画面に遷移する
* @return 初期表示画面へのパス
*/
@GetMapping("/")
public String index(){
return "index";
}
/**
* 引数の検索IDに対応するDemoFormデータを取得する
* @param id 検索ID
* @return DemoFormデータ(JSON形式)
*/
//JSON文字列を返却するために、@ResponseBodyアノテーションを付与
@GetMapping("/search")
@ResponseBody
public String search(@RequestParam("id") String id){
// ユーザーデータを取得し、取得できなければそのまま返す
UserData userData = repository.findUserDataById(Long.valueOf(id));
// ユーザーデータが取得できなかった場合は、null値を返す
if(userData == null){
return null;
}
// 性別を表示用(男,女)に変換
userData.setSex("1".equals(userData.getSex()) ? "男" : "女");
// 名前・メモ・性別をエンコード
userData.setName(encode(userData.getName()));
userData.setMemo(encode(userData.getMemo()));
userData.setSex(encode(userData.getSex()));
// 取得したユーザーデータをJSON文字列に変換し返却
return getJson(userData);
}
/**
* 引数の文字列をエンコードする
* @param data 任意の文字列
* @return エンコード後の文字列
*/
private String encode(String data){
// 引数がnullまたは空文字の場合は、その値を返す
if(StringUtils.isEmpty(data)){
return data;
}
String retVal = null;
try{
retVal = URLEncoder.encode(data, "UTF-8");
}catch (UnsupportedEncodingException e) {
System.err.println(e);
}
return retVal;
}
/**
* 引数のUserDataオブジェクトをJSON文字列に変換する
* @param userData UserDataオブジェクト
* @return 変換後JSON文字列
*/
private String getJson(UserData userData){
String retVal = null;
ObjectMapper objectMapper = new ObjectMapper();
try{
retVal = objectMapper.writeValueAsString(userData);
} catch (JsonProcessingException e) {
System.err.println(e);
}
return retVal;
}
}その他のソースコード内容は、以下のサイトを参照のこと。
https://github.com/purin-it/java/tree/master/spring-boot-web-ajax/demo
サンプルプログラムの実行結果
user_dataテーブルのデータの内容は以下の通りで、ID=1のデータは存在し、ID=2のデータは存在しないものとする。

また、Spring Bootアプリケーションを起動し、「http:// (ホスト名):(ポート番号)」とアクセスすると、以下の画面が表示される。

次に、以下のように、検索IDに1を指定し「検索」ボタンを押下すると、検索されたデータが表示される。


さらに、以下のように、検索IDに2を指定し「検索」ボタンを押下すると、該当するデータがない旨のダイアログが表示される。


要点まとめ
- Spring Bootのコントローラメソッド上でJSON文字列を返却するには、メソッドに@ResponseBodyアノテーションを付与し、戻り値はJSON文字列とする。
- Javaソース内でオブジェクトをJSON文字列に変換するには、ObjectMapperクラスのwriteValueAsStringメソッドを利用する。
- JavaScriptソース内でJSON文字列をオブジェクトに変換するには、JSON.parseメソッドを利用する。






