ページの作成
親となるページを選択してください。
親ページに紐づくページを子ページといいます。
例: 親=スポーツ, 子1=サッカー, 子2=野球
子ページを親ページとして更に子ページを作成することも可能です。
例: 親=サッカー, 子=サッカーのルール
親ページはいつでも変更することが可能なのでとりあえず作ってみましょう!
| この記事の要点 |
|
Express + EJS とは
Express は Node.js 上で動く最も普及した Web アプリケーションフレームワークです。HTTP ルーティング・ミドルウェア・テンプレート連携などの最小機能だけを提供する「薄い」フレームワークで、フロント主流時代でも API サーバや SSR の土台として現役です。
EJS(Embedded JavaScript Templates)は、HTML に JavaScript を埋め込めるシンプルなテンプレートエンジン。<%= name %> のように Erb/JSP 系の構文で、サーバ側でレンダリングします。学習コストが低く、Express の標準的なお供として広く使われています。
プロジェクト作成
# 1. ディレクトリ作成
mkdir myapp && cd myapp
# 2. package.json 作成
npm init -y
# 3. 依存追加
npm install express ejs
# 4. 開発用にホットリロード(任意)
npm install --save-dev nodemon
ファイル構成
myapp/
├── package.json
├── app.js ← エントリポイント
├── public/ ← 静的ファイル (CSS, JS, 画像)
│ └── css/style.css
├── views/ ← EJS テンプレート
│ ├── index.ejs
│ ├── about.ejs
│ └── partials/
│ ├── header.ejs
│ └── footer.ejs
└── routes/ ← ルート定義 (規模が大きくなったら分割)
└── users.js
最小の app.js
// app.js
const express = require('express');
const path = require('path');
const app = express();
// テンプレートエンジン: EJS
app.set('view engine', 'ejs');
app.set('views', path.join(__dirname, 'views'));
// 静的ファイル
app.use(express.static(path.join(__dirname, 'public')));
// フォーム解析ミドルウェア
app.use(express.urlencoded({ extended: true }));
app.use(express.json());
// ルーティング
app.get('/', (req, res) => {
res.render('index', { title: 'Home', name: 'Taro' });
});
app.get('/about', (req, res) => {
res.render('about', { title: 'About' });
});
// 起動
const PORT = process.env.PORT || 3000;
app.listen(PORT, () => {
console.log(`http://localhost:${PORT} で待機中`);
});
EJS テンプレート例
<%# views/index.ejs %>
<%- include('partials/header', { title: title }) %>
<main>
<h1>こんにちは、<%= name %> さん</h1>
<% if (name === 'Taro') { %>
<p>管理者です。</p>
<% } else { %>
<p>一般ユーザです。</p>
<% } %>
<ul>
<% ['apple', 'banana', 'cherry'].forEach(fruit => { %>
<li><%= fruit %></li>
<% }) %>
</ul>
<p>生 HTML をそのまま出す(XSS 注意): <%- '<strong>太字</strong>' %></p>
</main>
<%- include('partials/footer') %>
| タグ | 用途 |
|---|---|
<%= value %> | 変数をエスケープして出力(XSS 防止) |
<%- value %> | 生のまま HTML 出力(信頼できる値のみ) |
<% if () { %> ... <% } %> | JS 制御構文(出力しない) |
<%# コメント %> | テンプレート内コメント |
<%- include('partial') %> | 他テンプレート読込 |
パーシャル(共通ヘッダ・フッタ)
<%# views/partials/header.ejs %>
<!DOCTYPE html>
<html lang="ja">
<head>
<meta charset="UTF-8">
<title><%= title %></title>
<link rel="stylesheet" href="/css/style.css">
</head>
<body>
<nav>
<a href="/">Home</a> | <a href="/about">About</a>
</nav>
<%# views/partials/footer.ejs %>
<footer>© 2026 MyApp</footer>
</body>
</html>
フォーム POST の処理
// app.js に追加
app.get('/contact', (req, res) => {
res.render('contact', { title: 'Contact', message: null });
});
app.post('/contact', (req, res) => {
const { name, email, body } = req.body;
// 簡易バリデーション
if (!name || !email) {
return res.render('contact', {
title: 'Contact',
message: { type: 'error', text: '名前とメールは必須です' },
});
}
// DB 保存などの処理
console.log('Received', { name, email, body });
// PRG パターン: リダイレクト
res.redirect('/contact?ok=1');
});<%# views/contact.ejs %>
<%- include('partials/header', { title: title }) %>
<h1>お問い合わせ</h1>
<% if (message) { %>
<p class="<%= message.type %>"><%= message.text %></p>
<% } %>
<% if (typeof query !== 'undefined' && query.ok) { %>
<p class="success">送信しました。</p>
<% } %>
<form method="POST" action="/contact">
<label>名前 <input name="name" required></label>
<label>メール <input name="email" type="email" required></label>
<label>本文 <textarea name="body"></textarea></label>
<button type="submit">送信</button>
</form>
<%- include('partials/footer') %>
ルーティングを別ファイルに分割
// routes/users.js
const express = require('express');
const router = express.Router();
router.get('/', (req, res) => {
res.render('users/list', { users: ['Alice', 'Bob'] });
});
router.get('/:id', (req, res) => {
res.render('users/show', { id: req.params.id });
});
module.exports = router;
// app.js
const usersRouter = require('./routes/users');
app.use('/users', usersRouter);
// → /users → list, /users/123 → show
開発: ホットリロード
// package.json
{
"scripts": {
"start": "node app.js",
"dev": "nodemon app.js"
}
}npm run dev
# → ファイル変更で自動再起動
本番デプロイ
PM2 でデーモン化
npm install -g pm2
# 起動
pm2 start app.js --name myapp -i max # CPU コア数だけクラスタ起動
# 再起動・停止
pm2 restart myapp
pm2 stop myapp
pm2 logs myapp
# 起動時自動起動
pm2 startup
pm2 save
Nginx リバースプロキシ
# /etc/nginx/sites-available/myapp
server {
listen 80;
server_name example.com;
location / {
proxy_pass http://127.0.0.1:3000;
proxy_http_version 1.1;
proxy_set_header Upgrade $http_upgrade;
proxy_set_header Connection 'upgrade';
proxy_set_header Host $host;
proxy_cache_bypass $http_upgrade;
}
# 静的ファイルは Nginx 直配信(高速化)
location /public/ {
alias /var/www/myapp/public/;
expires 30d;
}
}
Dockerfile
FROM node:20-alpine
WORKDIR /app
COPY package*.json ./
RUN npm ci --omit=dev
COPY . .
ENV NODE_ENV=production
EXPOSE 3000
CMD ["node", "app.js"]
セキュリティのチェックポイント
helmetミドルウェアでセキュリティ HTTP ヘッダ自動付与 (npm install helmet→app.use(helmet()))express-rate-limitでブルートフォース防止- セッションは
express-session+ Redis Store(本番) - CSRF は
csurf(非推奨化済→csrf-csrf等)で対策 - EJS は
<%= %>(エスケープ)を基本とし、<%- %>は信頼データのみ
FAQ
Q: EJS と Pug、どちらを選ぶべき?
A: HTML をそのまま書きたいなら EJS、インデントベースの簡潔さなら Pug。チーム経験で選んで OK。新規は EJS が無難。
Q: SSR と SPA、どちらに向く?
A: Express + EJS は SSR 寄り。SPA なら Next.js / Nuxt / SvelteKit を直接使う方が現代的です。Express は API サーバとして残せます。
Q: 大規模化したらどうする?
A: routes/ 分割、controllers/ 切り出し、ORM (Prisma / Sequelize) 採用、TypeScript 化、を順番に。さらに大規模なら NestJS への移行も検討。
ページの作成
親となるページを選択してください。
親ページに紐づくページを子ページといいます。
例: 親=スポーツ, 子1=サッカー, 子2=野球
子ページを親ページとして更に子ページを作成することも可能です。
例: 親=サッカー, 子=サッカーのルール
親ページはいつでも変更することが可能なのでとりあえず作ってみましょう!
子ページはありません
- インストール方法(Windows編)
- インストール方法(CentOS編)
- クイックスタート
- 簡単なサーバー構築と起動方法
- ExpressとEJSを使用した簡単なアプリの作成
人気ページ
- 1 Eclipseで「サーバーに追加または除去できるリソースがありません。」の原因と対処法
- 2 tomcat の起動 / 停止ログと catalina.log・catalina.out の違い
- 3 JavaScript base URL 取得方法|window.location.origin と SSR/Node.js 対応
- 4 YouTube Data API v3 エラー一覧|403/400/404 の主要原因と切り分け
- 5 Spring Frameworkのアノテーション一覧
- 6 Laravel エラー一覧|500/Blade/DB 接続/ルーティングの代表エラー
- 7 3Dグラフィックスとは|モデリング/レンダリング/主要ソフトウェア (Blender / Maya)
- 8 【Spring】@Valueアノテーションとは
- 9 CATALINA_HOME の確認方法 (Linux / Mac)
- 10 【Spring】@Autowiredアノテーションとは
最近更新/作成されたページ
- IPv6とは|128bitアドレス・コロン16進表記/::省略・リンクローカル・SLAAC・デュアルスタック NEW 2026-06-22 12:34:44
- VPNとは|暗号トンネル・サイト間/リモートアクセス・IPsec/SSL-VPN/WireGuardを解説 NEW 2026-06-22 12:19:10
- MAC アドレスフィルタリングの仕組みと限界 | ネットワーク入門 NEW 2026-06-22 12:19:10
- WebRTC とは ブラウザ間 P2P の音声・映像・データ通信 | ネットワーク入門 NEW 2026-06-22 12:17:25
- gRPC とは HTTP/2 + Protocol Buffers の高速 RPC | ネットワーク入門 NEW 2026-06-22 12:17:25
- HTTP/3 (QUIC) とは UDP ベースの低遅延 Web 通信 | ネットワーク入門 NEW 2026-06-22 12:17:25
- HTTP/2 とは 多重化・HPACK・バイナリフレーム | ネットワーク入門 NEW 2026-06-22 12:17:25
- Web通信プロトコル入門 HTTP/2・HTTP/3・WebSocket・gRPC・WebRTC | ネットワーク入門 NEW 2026-06-22 12:17:25
- WebSocket とは 全二重リアルタイム通信 ws/wss | ネットワーク入門 NEW 2026-06-22 12:17:25
- ファイアウォールとは|パケットフィルタ・ステートフル・DMZ・次世代FW(L4/L7)を解説 NEW 2026-06-22 12:17:24
- iptables/nftablesとは|テーブル・チェーン・ルール例・永続化をLinux視点で解説 NEW 2026-06-22 12:17:24
- HAProxy とは frontend/backend と設定例 | ネットワーク入門 NEW 2026-06-22 12:17:24
- 証明書と認証局(CA)とは|X.509・信頼チェーン・DV/OV/EV・失効(CRL/OCSP)を解説 NEW 2026-06-22 12:17:24
- CDN とは エッジキャッシュ・TTL・Cloudflare/CloudFront | ネットワーク入門 NEW 2026-06-22 12:17:24
- TLS/SSLの仕組み|ハンドシェイク・暗号スイート・前方秘匿性・証明書検証をわかりやすく解説 NEW 2026-06-22 12:17:24
コメントを削除してもよろしいでしょうか?