文章目录

  • 一、结果预览
  • 代码:
  • 欢迎页面
  • 二、代码结构
  • springboot简单入门


一、结果预览

代码:

欢迎页面

jquery前端实现记录阅读量 jquery前端开发实战教程_html

二、代码结构

对前端开发的一些理解

安卓开发 vue html

前端开发个人觉得都是这样的框子 : 页面元素 + 样式 + 向后端发送请求

jquery前端实现记录阅读量 jquery前端开发实战教程_java_02

html开发页面元素 样式

jQuery发送请求,与前端页面元素关联

springboot简单入门

如何简单入门springboot做一个网页呢?
准备工作
1、新建一个SpringBoot项目,构建web模块
2、导入pojo和dao层的类
pojo层

package com.star.pojo;

public class Department {

    private Integer id;
    private String departmentName;

    public Department() {
    }
    
    public Department(int i, String string) {
        this.id = i;
        this.departmentName = string;
    }

    public Integer getId() {
        return id;
    }

    public void setId(Integer id) {
        this.id = id;
    }

    public String getDepartmentName() {
        return departmentName;
    }

    public void setDepartmentName(String departmentName) {
        this.departmentName = departmentName;
    }

    @Override
    public String toString() {
        return "Department [id=" + id + ", departmentName=" + departmentName + "]";
    }
    
}
package com.star.pojo;

import java.util.Date;

public class Employee {

    private Integer id;
    private String lastName;

    private String email;
    //1 male, 0 female
    private Integer gender;
    private Department department;
    private Date birth;

    public Integer getId() {
        return id;
    }

    public void setId(Integer id) {
        this.id = id;
    }

    public String getLastName() {
        return lastName;
    }

    public void setLastName(String lastName) {
        this.lastName = lastName;
    }

    public String getEmail() {
        return email;
    }

    public void setEmail(String email) {
        this.email = email;
    }

    public Integer getGender() {
        return gender;
    }

    public void setGender(Integer gender) {
        this.gender = gender;
    }

    public Department getDepartment() {
        return department;
    }

    public void setDepartment(Department department) {
        this.department = department;
    }

    public Date getBirth() {
        return birth;
    }

    public void setBirth(Date birth) {
        this.birth = birth;
    }
    public Employee(Integer id, String lastName, String email, Integer gender,
                    Department department) {
        super();
        this.id = id;
        this.lastName = lastName;
        this.email = email;
        this.gender = gender;
        this.department = department;
        this.birth = new Date();
    }

    public Employee() {
    }

    @Override
    public String toString() {
        return "Employee{" +
                "id=" + id +
                ", lastName='" + lastName + '\'' +
                ", email='" + email + '\'' +
                ", gender=" + gender +
                ", department=" + department +
                ", birth=" + birth +
                '}';
    }
}

dao层

package com.star.dao;

import java.util.Collection;
import java.util.HashMap;
import java.util.Map;

import com.star.pojo.Department;
import org.springframework.stereotype.Repository;

//模拟数据库
@Repository
public class DepartmentDao {

    private static Map<Integer, Department> departments = null;
    
    static{
        departments = new HashMap<Integer, Department>();
        
        departments.put(101, new Department(101, "D-AA"));
        departments.put(102, new Department(102, "D-BB"));
        departments.put(103, new Department(103, "D-CC"));
        departments.put(104, new Department(104, "D-DD"));
        departments.put(105, new Department(105, "D-EE"));
    }

    //获得全部部门信息
    public Collection<Department> getDepartments(){
        return departments.values();
    }

    //通过id获取部门信息
    public Department getDepartment(Integer id){
        return departments.get(id);
    }
    
}
package com.star.dao;

import java.util.Collection;
import java.util.HashMap;
import java.util.Map;

import com.star.pojo.Department;
import com.star.pojo.Employee;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Repository;

//模拟数据库
@Repository
public class EmployeeDao {

    private static Map<Integer, Employee> employees = null;
    
    @Autowired
    private DepartmentDao departmentDao;
    
    static{
        employees = new HashMap<Integer, Employee>();

        employees.put(1001, new Employee(1001, "E-AA", "aa@163.com", 1, new Department(101, "D-AA")));
        employees.put(1002, new Employee(1002, "E-BB", "bb@163.com", 1, new Department(102, "D-BB")));
        employees.put(1003, new Employee(1003, "E-CC", "cc@163.com", 0, new Department(103, "D-CC")));
        employees.put(1004, new Employee(1004, "E-DD", "dd@163.com", 0, new Department(104, "D-DD")));
        employees.put(1005, new Employee(1005, "E-EE", "ee@163.com", 1, new Department(105, "D-EE")));
    }
    
    private static Integer initId = 1006;

    //新增员工实现id自增
    public void save(Employee employee){
        if(employee.getId() == null){
            employee.setId(initId++);
        }
        
        employee.setDepartment(departmentDao.getDepartment(employee.getDepartment().getId()));
        employees.put(employee.getId(), employee);
    }

    //获得全部的员工
    public Collection<Employee> getAll(){
        return employees.values();
    }

    //通过id获取员工
    public Employee get(Integer id){
        return employees.get(id);
    }

    //通过id删除员工
    public void delete(Integer id){
        employees.remove(id);
    }
}

3、导入静态资源(页面放在Temleate,资源放在static目录下)
链接:https://pan.baidu.com/s/1FH8gy0WlMQXED6inbquPzQ

提取码:77ck

4、所有的页面增加头文件 xmlns:th=“http://www.thymeleaf.org”
5、所有的资源链接修改格式为thymeleaf语法:th:xxx=“@{}”
6、可以增加一个配置类,配置我们的SpringMVC

//代表这是一个配置类
@Configuration
public class MyMvcConfig implements WebMvcConfigurer {

    //通过配置类来设置试图跳转
    @Override
    public void addViewControllers(ViewControllerRegistry registry) {
        registry.addViewController("/").setViewName("index");//定制首页为index.html页面
        registry.addViewController("/index.html").setViewName("index");
        registry.addViewController("/main.html").setViewName("dashboard");
    }

MVC配置原理:

7、我们可以设置项目的路径名,application.properties

# 配置自己的服务器路径 localhost:8080/star
service.servlet.context-path=/star

回到顶部
实现功能
1、国际化
可以看到再首页的下面有中英文转换

jquery前端实现记录阅读量 jquery前端开发实战教程_jQuery_03

这就是国际化,我们来实现这个功能!

首先修改编码

jquery前端实现记录阅读量 jquery前端开发实战教程_jquery前端实现记录阅读量_04

在Rources下创建目录 i18n(国际化 internationalization 单词的缩写)

创建login.properties(默认语言)文件,再创建login_zh_CN.properties(中文)文件。可以看到会将两个文件合并在一个目录下!

jquery前端实现记录阅读量 jquery前端开发实战教程_html_05

添加login_en_US.properties(英文)文件

jquery前端实现记录阅读量 jquery前端开发实战教程_spring boot_06

编写login.properties,使用可视化编写!

jquery前端实现记录阅读量 jquery前端开发实战教程_spring boot_07

jquery前端实现记录阅读量 jquery前端开发实战教程_java_08

然后修改index.html代码,使用thymeleaf绑定

<form class="form-signin" th:action="@{user/login}">
	<img class="mb-4" th:src="@{/asserts/img/bootstrap-solid.svg}" alt="" width="72" height="72">

	<h1 class="h3 mb-3 font-weight-normal" th:text="#{login.tip}">Please Sign In</h1>

	<label class="sr-only">Username</label>
	<input type="text" class="form-control" name="username" th:placeholder="#{login.username}" required="" autofocus="">
	<label class="sr-only">Password</label>
	<input type="password" class="form-control" name="password" th:placeholder="#{login.password}" required="">

	<div class="checkbox mb-3">
		<label>
  			<input type="checkbox" value="remember-me" th:text="#{login.properties}">
		</label>
	</div>
	<button class="btn btn-lg btn-primary btn-block" type="submit">[[#{login.btn}]]</button>
	<p class="mt-5 mb-3 text-muted">© 2017-2018</p>
	<a class="btn btn-sm"}">中文</a>
	<a class="btn btn-sm"}">English</a>
</form>

启动项目测试:

jquery前端实现记录阅读量 jquery前端开发实战教程_java_09

接下来我们使用下面的链接来进行中英文切换!

首先我们进入国际化的源码;

进入WebMvcAutoConfiguration.java类,找到LocaleResolver方法;

jquery前端实现记录阅读量 jquery前端开发实战教程_java_10

分析,代码,可以看出,默认使用AcceptHeaderLocaleResolver类,查看源码:实现LocaleResolver接口!

jquery前端实现记录阅读量 jquery前端开发实战教程_html_11

重写方法!

所以我们可以写一个自己的国际化解析器,实现LocaleResolver接口!

参照AcceptHeaderLocaleResolver类重写方法的格式!

我们先在前端传入语言的参数,thymeleaf传参用()

<a class="btn btn-sm" th:href="@{/index.html(l='zh_CN')}">中文</a>
<a class="btn btn-sm" th:href="@{/index.html(l='en_US')}">English</a>

这个链接的地址还是当前的页面,不过要传国际化参数。

编写自己的国际化解析器!

package com.star.config;

import org.springframework.web.servlet.LocaleResolver;
import org.thymeleaf.util.StringUtils;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.util.Locale;

public class MyLocalResolver implements LocaleResolver {
    @Override
    public Locale resolveLocale(HttpServletRequest request) {
        //获取请求语言中的语言参数
        String language = request.getParameter("l");
        Locale locale = Locale.getDefault();//如果没有使用默认值       
        //如果请求的链接携带了国际化的参数
        if(!StringUtils.isEmpty(language)){
            //zh_CN分割
            String[] s = language.split("_");
            //国家_地区
            locale = new Locale(s[0], s[1]);
        }

        return locale;
    }

    @Override
    public void setLocale(HttpServletRequest request, HttpServletResponse response, Locale locale) {
            //无返回值,不用写代码
    }
}

然后在配置类中注册bean,自定义的国际化组件就生效了

@Bean
public LocaleResolver localeResolver(){
    return new MyLocalResolver();
}

重启项目测试:

jquery前端实现记录阅读量 jquery前端开发实战教程_html_12

jquery前端实现记录阅读量 jquery前端开发实战教程_jquery前端实现记录阅读量_13

功能实现成功!

2、增加登录和拦截的功能!
1、前端提交登录请求 (看表单)

<form class="form-signin" th:action="@{user/login}">

2、后端处理登录请求 (登录成功,跳转到主页,显示用户的名字,如果没有成功告诉用户:用户名或密码错误)

@Controller
public class userController {

    //登录实现
    @RequestMapping("/user/login")
    public String login(@RequestParam("username") String username, @RequestParam("password") String password, Model model, HttpSession session){
        if(!StringUtils.isEmpty(username)&& "123456".equals(password)){
            model.addAttribute("msg","");
            session.setAttribute("loginUser",username);
            //登陆成功
            //登录重定向,避免一直提交
            return "redirect:/main.html";
        }else {
            model.addAttribute("msg","登录失败,请重新登录");
            return "index";
        }
    }
}

3、发现问题:没有登录也能进入主页

4、增加一个拦截器,判断用户是否登录

package com.star.interceptor;

import org.springframework.web.servlet.HandlerInterceptor;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

public class LoginInterceptor implements HandlerInterceptor {

    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        if(request.getSession().getAttribute("loginUser")==null){
            request.setAttribute("msg","没有权限,请先登录!");
            request.getRequestDispatcher("/index.html").forward(request,response);//转发到登录页面
            return false;
        }else{
            return true;
        }
    }
}

配置类中注册拦截器

//注册拦截器
@Override
public void addInterceptors(InterceptorRegistry registry) {
    //选择要过滤和排除的请求
    registry.addInterceptor(new LoginInterceptor())
            .addPathPatterns("/**")
            .excludePathPatterns("/","/index.html","/user/login")//排除登录的请求
            .excludePathPatterns("/asserts/**");//排除静态资源的请求
}

将 登录失败,请重新登录 的参数传入前端页面

<!--判断是否有错误标签-->
<p style="color: red" th:if="${not #strings.isEmpty(msg)}" th:text="${msg}"></p>

5、测试登录和拦截是否成功;

我们如果未登录,main.html请求就不能通过,并且转发到登录页面!

jquery前端实现记录阅读量 jquery前端开发实战教程_jquery前端实现记录阅读量_14

3、展示员工列表
我们将所有员工的信息查询并展示出来

编写controller类

package com.star.controller;

import com.star.dao.EmployeeDao;
import com.star.pojo.Employee;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.RequestMapping;

import java.util.Collection;

@Controller
public class EmployeeController {

    @Autowired  //自动转配,或者在配置类中注册bean
    EmployeeDao employeeDao;

    @RequestMapping("/emps")
    public String list(Model model){
        Collection<Employee> employees = employeeDao.getAll();//通过该方法查询所有的员工信息
        model.addAttribute("emps",employees);
        return "emp/list";//跳转至在templates下的emp目录下list.html页面
    }
}

为了减少代码量,我们提取公共页面,在templates目录下创建commons目录,新建commons.html页面;

由于list页面和dashboard页面的顶部栏和侧边栏是相同的。所以我们抽取出来,放在commens.html页面中

在头部导航栏的标签中添加 th:fragment=“topBar”(使用thymeleaf语法)

在侧边导航栏的标签中添加 th:fragment=“sideBar”

然后在两个页面的原位置添加如下代码:

<!--顶部导航栏  commons目录下的commons页面的topBar-->
<div th:replace="~{commons/commons::topBar}"></div>
<!--侧边栏-->
<div th:replace="~{commons/commons::sideBar}"></div>

jquery前端实现记录阅读量 jquery前端开发实战教程_java_15

我们需要选定某个页面,侧边拦中的该页面要变亮;

接下来实现这个功能

看公共页面中 首页 的代码中可以看到有active

jquery前端实现记录阅读量 jquery前端开发实战教程_jquery前端实现记录阅读量_16

我们可以使用thymeleaf语法来判断;

请求首页时传参数active=‘main.html’

在dashboard.html页面中修改代码

<!--侧边栏-->
<div th:replace="~{commons/commons::sideBar(active='main.html')}"></div>

请求员工管理即list页面时传参数active=‘list.html’

在list.html页面中修改代码

<!--侧边栏-->
<div th:replace="~{commons/commons::sideBar(active='list.html')}"></div>

修改公共页的代码

<!--修改侧边栏的首页代码-->
th:class="${active=='main.html'?'nav-link active':'nav-link'}"
<!--修改侧边栏员工管理代码-->
th:class="${active=='list.html'?'nav-link active':'nav-link'}"

测试:

jquery前端实现记录阅读量 jquery前端开发实战教程_jquery前端实现记录阅读量_17

接下来我们展示所有的员工信息

修改list.html页面中thead标签和tbody标签中的代码(使用thymeleaf语法)

<thead>
	<tr>
		<th>id</th>
		<th>lastName</th>
		<th>email</th>
		<th>gender</th>
		<th>department</th>
		<th>birth</th>
		<th>操作</th>
	</tr>
</thead>
<tbody>
	<tr th:each="emp:${emps}">
		<td th:text="${emp.getId()}"></td>
		<td th:text="${emp.getLastName()}"></td>
		<td th:text="${emp.getEmail()}"></td>
		<td th:text="${emp.getGender()==1?'男':'女'}"></td>
		<td th:text="${emp.getDepartment().getDepartmentName()}"></td>
		<td th:text="${#dates.format(emp.getBirth(),'yyyy-HH-dd HH:mm:ss')}"></td>
		<td>
			<a href="" class="btn btn-sm btn-success">编辑</a>
			<a href="" class="btn btn-sm btn-danger">删除</a>
		</td>
	</tr>
</tbody>

测试:

jquery前端实现记录阅读量 jquery前端开发实战教程_jquery前端实现记录阅读量_18

OK!我们实现了展示所有员工信息的功能

4、实现增加员工的功能
在展示列表的上面添加一个增加员工的链接

<h2><a class="btn btn-lg btn-success" th:href="@{/toAdd}">添加员工</a></h2>

这里是get请求

我们编写controller类

@Autowired
    DepartmentDao departmentDao;

    @GetMapping("/toAdd")
    public String add(Model model){
        Collection<Department> departments = departmentDao.getDepartments();
        model.addAttribute("departments",departments);//页面传入departments
        return "emp/add";
    }

新建一个增加员工的页面add.html

代码框架和list一样,修改main标签的内容

<main role="main" class="col-md-9 ml-sm-auto col-lg-10 pt-3 px-4">
    <!--增加一个表单  提交到/emp请求-->
    <form th:action="@{/emp}" method="post">
        <div class="form-group">
            <label>lastName</label>
            <input type="text" class="form-control" name="lastName"  placeholder="lastName">
        </div>
        <div class="form-group">
            <label>email</label>
            <input type="email" class="form-control" name="email" placeholder="email">
        </div>
        <div class="radio">
            <label>gender</label>
            <br>
            <label>
                <input type="radio" name="gender" value="1">男
                <input type="radio" name="gender" value="0">女
            </label>
        </div>
        <div class="form-group">
            <label>department</label>
            <select class="form-control" name="department.id">
                <option th:each="dept:${departments}" th:text="${dept.getDepartmentName()}" th:value=
            </select>
        </div>
        <div class="form-group">
            <label>birth</label>
            <input type="text" name="birth" class="form-control" placeholder="birth">
        </div>
        <br>
        <div class="form-group">
            <button type="submit" class="btn btn-sm btn-success">添加</button>
        </div>
    </form>
</main>

表单使用post请求到emp,编写controller页面

@Autowired
EmployeeDao  employeeDao;

@PostMapping("/emp")
public String addEmp(Employee employee){
    employeeDao.save(employee);//使用save方法增加员工
    return "redirect:/emps";//重定向到emps也就是list.hrml页面
}

测试功能:

jquery前端实现记录阅读量 jquery前端开发实战教程_html_19

注意:springBoot里面的时间默认格式为:yyyy/MM/dd 所以表单的时间的格式为2020/01/01;

jquery前端实现记录阅读量 jquery前端开发实战教程_java_20

jquery前端实现记录阅读量 jquery前端开发实战教程_html_21

OK!增加员工功能实现成功!

5、修改员工功能
编写修改的链接以及请求

<a th:href="@{/toUpdate/}+${emp.getId()}" class="btn btn-sm btn-success">编辑</a>

编写对应的controlelr类

@GetMapping("/toUpdate/{id}")
public String toUpdate(@PathVariable("id") int id, Model model){
    Collection<Department> departments = departmentDao.getDepartments();
    model.addAttribute("departments",departments);
    Employee employee = employeeDao.get(id);
    model.addAttribute("emp",employee);
    return "emp/update";
}

编写修改页面update.html,也是一个表单,同样提交到emp请求,这里注意要加一个隐藏域显示id,这样id就不会自增从而修改成功!

<main role="main" class="col-md-9 ml-sm-auto col-lg-10 pt-3 px-4">
    <form th:action="@{/emp}" method="post">
        <!--隐藏域-->
        <input type="hidden" name="id" th:value="${emp.getId()}">

        <div class="form-group">
            <label>lastName</label>
            <input type="text" class="form-control" name="lastName" th:value="${emp.getLastName()}"  placeholder="lastName" >
        </div>
        <div class="form-group">
            <label>email</label>
            <input type="email" class="form-control" name="email" th:value="${emp.getEmail()}" placeholder="Email">
        </div>
        <div class="radio">
            <label>gender</label>
            <br>
            <label>
                <input th:checked="${emp.getGender()==1}" type="radio" name="gender" value="1">男
                <input th:checked="${emp.getGender()==0}" type="radio" name="gender" value="0">女
            </label>
        </div>
        <div class="form-group">
            <label>department</label>
            <select class="form-control" name="department.id">
                <option th:selected="${dept.getId()==emp.getDepartment().getId()}" th:each="dept:${departments}" th:text="${dept.getDepartmentName()}" th:value="${dept.getId()}"></option>
            </select>
        </div>
        <div class="form-group">
            <label>birth</label>
            <input type="text" name="birth" class="form-control" th:value="${#dates.format(emp.getBirth(),'yyyy/MM/dd HH:mm:ss')}">
        </div>
        <br>
        <div class="form-group">
            <button type="submit" class="btn btn-sm btn-success">修改</button>
        </div>
    </form>
</main>

测试:

jquery前端实现记录阅读量 jquery前端开发实战教程_html_22

jquery前端实现记录阅读量 jquery前端开发实战教程_jquery前端实现记录阅读量_23

OK!修改功能成功!

6、删除员工页面
编写删除的链接以及请求

<a th:href="@{/delete/}+${emp.getId()}" class="btn btn-sm btn-danger">删除</a>

编写对应的controlelr类

@GetMapping("/delete/{id}")
public String delete(@PathVariable("id") int id){
    employeeDao.delete(id);//根据id删除对应的员工
    return "redirect:/emps";
}

测试功能:点击删除即可!