2.3. はじめてのSpring MVCアプリケーション¶
Spring MVCの、詳細な使い方の解説に入る前に、実際にSpring MVCに触れることで、Spring MVCを用いたWebアプリケーションの開発に対するイメージをつかむ。
2.3.1. 検証環境¶
本節の説明では、次の環境で動作検証している。(他の環境で実施する際は、本書をベースに適宜読み替えて設定していくこと。)
種別 |
プロダクト |
|---|---|
OS |
Windows 11 |
JVM |
Java 17 |
IDE |
Spring Tool Suite 5.0.0.RELEASE (以降「STS」と呼ぶ。設定方法はSTSの設定手順を参照されたい。) |
Build Tool |
Apache Maven 3.9.9 (以降「Maven」と呼ぶ) |
Application Server |
Apache Tomcat 10.1.50 |
Web Browser |
Google Chrome 145 |
Note
インターネット接続するためにプロキシサーバーを介する必要がある場合は、STSのProxy設定とMavenのProxy設定が必要である。
2.3.2. 新規プロジェクト作成¶
新規プロジェクト作成手順については、ブランクプロジェクトの作成のシングルプロジェクトの「O/R Mapperに依存しないプロジェクトを生成するためのArchetype」を参照されたい。
2.3.3. サーバーを起動する¶
サーバーを起動する手順については、ブランクプロジェクトの動作確認を参照されたい。
2.3.4. エコーアプリケーションの作成¶
続いて、簡単なアプリケーションを作成する。作成するのは、次の図のようなテキストフィールドに、名前を入力すると メッセージを表示する、いわゆるエコーアプリケーションである。
2.3.4.1. フォームオブジェクトの作成¶
com.example.todo.app.echoパッケージにEchoFormクラスを作成する。プロパティを1つだけ持つ、単純なJavaBeanである。package com.example.todo.app.echo;
import java.io.Serializable;
public class EchoForm implements Serializable {
private static final long serialVersionUID = 1L;
private String name;
public void setName(String name) {
this.name = name;
}
public String getName() {
return name;
}
}
2.3.4.2. Controllerの作成¶
次に、Controller(com.example.todo.app.echo.EchoController)クラスを作成する。
package com.example.todo.app.echo;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.ModelAttribute;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
@Controller
@RequestMapping("echo") // (1)
public class EchoController {
@ModelAttribute // (2)
public EchoForm setUpEchoForm() {
EchoForm form = new EchoForm();
return form;
}
@GetMapping // (3)
public String index(Model model) {
return "echo/index"; // (4)
}
@PostMapping(value = "hello") // (5)
public String hello(EchoForm form, Model model) {// (6)
model.addAttribute("name", form.getName()); // (7)
return "echo/hello";
}
}
項番 |
説明 |
|---|---|
(1)
|
Controllerクラスに
@RequestMappingを付与した場合、該当のクラスのルートパスは”<contextPath> + / + value属性の値“となる。本実装例では
<contextPath>/echoがControllerクラスのルートパスとなる。Controllerクラスに
@RequestMappingを付与していない場合、該当のクラスのルートパスは”<contextPath>“となる。 |
(2)
|
@ModelAttributeというアノテーションを、メソッドに付加する。このアノテーションがついたメソッドの返り値は、自動でModelに追加される。Modelの属性名を、
@ModelAttributeで指定することもできるが、デフォルトでは、クラス名の先頭を小文字にした値が、属性名になる。この場合は、echoFormである。フォームの属性名は、次に説明するform:form タグのmodelAttribute属性の値に一致している必要がある。 |
(3)
|
メソッドに
@GetMappingを付与した場合、メソッドへのマッピングは”クラスのルートパス + / + value属性の値“となる。本実装例では
value属性に何も指定していないため、”クラスのルートパス(<contextPath>/echo)“にGETメソッドでアクセスするとindexメソッドが呼ばれるようになる。 |
(4)
|
View名で”
echo/index“を返すので、ViewResolverにより、”WEB-INF/views/echo/index.jsp“がレンダリングされる。 |
(5)
|
メソッドに
@PostMappingを付与した場合、メソッドへのマッピングは”クラスのルートパス + / + value属性の値“となる。本実装例では
value属性の値にhelloを指定しているため、”クラスのルートパス + value属性の値(<contextPath>/echo/hello)“にPOSTメソッドでアクセスするとhelloメソッドが呼ばれるようになる。 |
(6)
|
引数に、EchoFormには(1)によりModelに追加されたEchoFormオブジェクトが渡される。
|
(7)
|
フォームで入力された
nameを、Viewにそのまま渡す。 |
Note
@GetMappingアノテーションもしくは@PostMappingアノテーションをメソッドに指定する場合は、クライアントから送信されたデータの扱い方によって変えるのが一般的である。
データをサーバに保存する場合(更新系の処理の場合)は、
@PostMappingアノテーション(POSTメソッド)。データをサーバに保存しない場合(参照系の処理の場合)は、
@GetMappingアノテーション(GETメソッド)。
エコーアプリケーションでは、
indexメソッドはデータをサーバに保存しない処理なのでGETメソッド@GetMappingアノテーションhelloメソッドはデータをModelオブジェクトに保存する処理なので@PostMappingアノテーション
を指定している。
2.3.4.3. Viewの作成¶
最後に、入力画面と、出力画面のViewを作成する。それぞれのファイルパスは、View名に合わせて、次のようになる。
入力画面 (src/main/webapp/WEB-INF/views/echo/index.jsp) を作成する。
<!DOCTYPE html>
<html>
<head>
<title>Echo Application</title>
</head>
<body>
<%-- (1) --%>
<form:form modelAttribute="echoForm" action="${pageContext.request.contextPath}/echo/hello">
<form:label path="name">Input Your Name:</form:label>
<form:input path="name" />
<input type="submit" />
</form:form>
</body>
</html>
項番 |
説明 |
|---|---|
(1)
|
タグライブラリを利用し、HTMLフォームを構築している。
modelAttribute属性に、Controllerで用意したフォームオブジェクトの名前を指定する。タグライブラリはSpring Framework Documentation -The Form Tag-を参照されたい。
|
Note
<form:form>タグのmethod属性を省略した場合は、POSTメソッドが使用される。
出力されるHTMLは、
<!DOCTYPE html>
<html>
<head>
<title>Echo Application</title>
</head>
<body>
<form id="echoForm" action="/todo/echo/hello" method="post">
<label for="name">Input Your Name:</label>
<input id="name" name="name" type="text" value="" />
<input type="submit" />
<input type="hidden" name="_csrf" value="LTH-77MOCsDmzZ6emzXQ1yehUN_x649j3SHgZ3p4bO8J8HLvGlTO2oNsb_bL-vv7_Rjk70bDfb3F3rdOuBKDBkJOCNs4k0uJ" />
</form>
</body>
</html>
となる。
出力画面 (src/main/webapp/WEB-INF/views/echo/hello.jsp) を作成する。
<!DOCTYPE html>
<html>
<head>
<title>Echo Application</title>
</head>
<body>
<p>Hello <c:out value="${name}" /> <%-- (2) --%></p>
</body>
</html>
項番 |
説明 |
|---|---|
(2)
|
Controllerから渡された
nameを出力する。c:outタグにより、XSS対策を行っている。 |
入力画面 (src/main/webapp/WEB-INF/views/echo/index.html) を作成する。
<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org"> <!--/* (1) */-->
<head>
<title>Echo Application</title>
</head>
<body>
<!--/* (2) */-->
<form th:object="${echoForm}" th:action="@{/echo/hello}" method="post">
<label for="name">Input Your Name:</label>
<input th:field="*{name}" /> <!--/* (3) */-->
<input type="submit" />
</form>
</body>
</html>
項番 |
説明 |
|---|---|
(1)
|
スタンダードダイアレクトが提供する属性を使用したとき、EclipseなどのIDEでの警告を抑止するため、ネームスペースを付与する。
|
(2)
|
Thymeleafの属性を利用し、HTMLフォームを構築している。
th:object属性に、Controllerで用意したフォームオブジェクトの名前を指定する。また、ThymeleafのリンクURL式
@{}に “/“から始まるパスを記述することでコンテキスト相対パスが生成され、th:action属性に指定できる。これらの属性の詳細についてはTutorial: Thymeleaf + Spring -Creating a Form-を参照されたい。
|
(3)
|
Thymeleaf + Springで提供される
th:field属性を用いて、特定のプロパティをHTML formにバインドすることができる。th:field属性はid属性、name属性、value属性をHTMLに出力し、id属性、name属性にはプロパティ名が出力される。th:field属性の詳細については、アプリケーション層の実装を参照されたい。 |
Note
<form>タグのmethod属性を省略した場合は、GETメソッドが使用される。
出力されるHTMLは、
<!DOCTYPE html>
<html>
<head>
<title>Echo Application</title>
</head>
<body>
<form action="/todo/echo/hello" method="post">
<input type="hidden" name="_csrf" value="LTH-77MOCsDmzZ6emzXQ1yehUN_x649j3SHgZ3p4bO8J8HLvGlTO2oNsb_bL-vv7_Rjk70bDfb3F3rdOuBKDBkJOCNs4k0uJ" />
<label for="name">Input Your Name:</label>
<input id="name" name="name" value="" />
<input type="submit" />
</form>
</body>
</html>
となる。
出力画面 (src/main/webapp/WEB-INF/views/echo/hello.html) を作成する。
<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head>
<title>Echo Application</title>
</head>
<body>
<p th:text="|Hello ${name}|"></p> <!--/* (4) */-->
</body>
</html>
項番 |
説明 |
|---|---|
(4)
|
Controllerから渡された
nameを出力する。th:text属性により、XSS対策を行っている。 |
2.3.4.4. 入力チェックの実装¶
EchoFormのnameフィールドに、入力チェックルールを指定するアノテーションを付与する。
package com.example.todo.app.echo;
import java.io.Serializable;
import jakarta.validation.constraints.NotNull;
import jakarta.validation.constraints.Size;
public class EchoForm implements Serializable {
private static final long serialVersionUID = 1L;
@NotNull // (1)
@Size(min = 1, max = 5) // (2)
private String name;
public void setName(String name) {
this.name = name;
}
public String getName() {
return name;
}
}
項番 |
説明 |
|---|---|
(1)
|
@NotNullアノテーションをつけることで、HTTPリクエスト中にnameパラメータがあることを確認する。 |
(2)
|
@Size(min = 1, max = 5)をつけることで、nameのサイズが、1以上5以下であることを確認する。 |
入力チェックが実行されるように修正し、入力チェックでエラーが発生した場合の処理を実装する。
package com.example.todo.app.echo;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.validation.BindingResult;
import org.springframework.validation.annotation.Validated;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.ModelAttribute;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
@Controller
@RequestMapping("echo")
public class EchoController {
@ModelAttribute
public EchoForm setUpEchoForm() {
EchoForm form = new EchoForm();
return form;
}
@GetMapping
public String index(Model model) {
return "echo/index";
}
@PostMapping(value = "hello")
public String hello(@Validated EchoForm form, BindingResult result, Model model) { // (1)
if (result.hasErrors()) { // (2)
return "echo/index";
}
model.addAttribute("name", form.getName());
return "echo/hello";
}
}
項番 |
説明 |
|---|---|
(1)
|
コントローラー側には、Validation対象の引数に
@Validatedアノテーションを付加し、BindingResultオブジェクトを引数に追加する。Bean Validationによる入力チェックは、自動で行われる。結果は、
BindingResultオブジェクトに渡される。 |
(2)
|
hasErrorsメソッドを実行して、エラーがあるかどうかを確認する。入力エラーがある場合は、入力画面を表示するためのView名を返却する。 |
入力画面に、入力エラーのメッセージを表示するための実装を追加する。
src/main/webapp/WEB-INF/views/echo/index.jsp
<!DOCTYPE html>
<html>
<head>
<title>Echo Application</title>
</head>
<body>
<form:form modelAttribute="echoForm" action="${pageContext.request.contextPath}/echo/hello">
<form:label path="name">Input Your Name:</form:label>
<form:input path="name" />
<form:errors path="name" cssStyle="color:red" /><%-- (1) --%>
<input type="submit" />
</form:form>
</body>
</html>
項番 |
説明 |
|---|---|
(1)
|
入力画面には、エラーがあった場合に、エラーメッセージを表示するため、
form:errorsタグを追加する。 |
出力されるHTMLは、
<!DOCTYPE html>
<html>
<head>
<title>Echo Application</title>
</head>
<body>
<form id="echoForm" action="/todo/echo/hello" method="post">
<label for="name">Input Your Name:</label>
<input id="name" name="name" type="text" value="" />
<span id="name.errors" style="color: red">size must be between 1 and 5</span>
<input type="submit" />
<div>
<input type="hidden" name="_csrf" value="LTH-77MOCsDmzZ6emzXQ1yehUN_x649j3SHgZ3p4bO8J8HLvGlTO2oNsb_bL-vv7_Rjk70bDfb3F3rdOuBKDBkJOCNs4k0uJ" />
</div>
</form>
</body>
</html>
となる。
src/main/webapp/WEB-INF/views/echo/index.html
<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head>
<title>Echo Application</title>
</head>
<body>
<form th:object="${echoForm}" th:action="@{/echo/hello}" method="post">
<label for="name">Input Your Name:</label>
<input th:field="*{name}" />
<span th:errors="*{name}" style="color: red"></span> <!--/* (1) */-->
<input type="submit" />
</form>
</body>
</html>
項番 |
説明 |
|---|---|
(1)
|
入力画面には、エラーがあった場合に、エラーメッセージを表示するため、
th:errors 属性を追加する。 |
出力されるHTMLは、
<!DOCTYPE html>
<html>
<head>
<title>Echo Application</title>
</head>
<body>
<form action="/todo/echo/hello" method="post">
<input type="hidden" name="_csrf" value="LTH-77MOCsDmzZ6emzXQ1yehUN_x649j3SHgZ3p4bO8J8HLvGlTO2oNsb_bL-vv7_Rjk70bDfb3F3rdOuBKDBkJOCNs4k0uJ">
<label for="name">Input Your Name:</label>
<input id="name" name="name" value="">
<span style="color:red">size must be between 1 and 5</span>
<input type="submit">
</form>
</body>
</html>
となる。
名前を空にして送信した場合
5文字より大きいサイズで送信した場合
2.3.4.5. まとめ¶
この章では、
mvn archetype:generateを利用したブランクプロジェクトの作成方法Spring MVCの基本的な設定方法
最も簡易な、画面遷移方法
画面間での値の引き渡し方法
シンプルな入力チェック方法
を学んだ。
上記の内容が理解できていない場合は、もう一度、本節を読み、環境構築から始めて、進めていくことで理解が深まる。