先後端分離的一個簡單用戶登陸Demo
。css
Vue
BootstrapVue
Kotlin
Spring Boot
MyBatis Plus
使用vue-cli
建立,沒安裝的能夠先安裝:html
sudo cnpm install -g vue @vue/cli
查看版本:前端
vue -V
出現版本就安裝成功了。vue
建立初始工程:java
vue create bvdemo
因爲目前Vue3
尚未發佈正式版本,推薦使用Vue2
:node
等待一段時間構建好了以後會提示進行文件夾並直接運行:mysql
cd bvdemo yarn serve
直接經過本地的8080
端口便可訪問:jquery
進入項目文件夾:ios
cd bvdemo
安裝依賴:git
cnpm install bootstrap-vue axios jquery vue-router
應該會出現popper.js
過時的警告,這是bootstrap-vue
的緣由,能夠忽略:
依賴說明以下:
bootstrap-vue
:一個結合了Vue
與Bootstrap
的前端UI
框架axios
是一個簡潔易用高效的http
庫,本項目使用其發送登陸請求jquery
:一個強大的JS
庫vue-router
:Vue
的官方路由管理器在正式編寫代碼以前開啓對bootstrap-vue
的補全支持,打開設置:
將項目路徑下的node_modules
添加到庫中,把前面的勾給勾上,接着更新緩存並重啓(`File->Invalidate Cache/Restart
`)。
App.vue
去掉默認的HelloWorld
組件,並修改App.vue
以下:
<template> <div id="app"> <router-view></router-view> </div> </template> <script> export default { name: 'App', } </script> <style> #app { font-family: Avenir, Helvetica, Arial, sans-serif; -webkit-font-smoothing: antialiased; -moz-osx-font-smoothing: grayscale; text-align: center; color: #2c3e50; margin-top: 60px; } </style>
<router-view>
是一個functional
組件,渲染路徑匹配到的視圖組件,這裏使用<router-view>
根據訪問路徑(路由)的不一樣顯示(渲染)相應的組件。
vue
組件刪除默認的HelloWorld.vue
,新建Index.vue
以及Login.vue
:
在main.js
同級目錄下新建router.js
,內容以下:
import Vue from "vue" import VueRouter from "vue-router" import Login from "@/components/Login" import Index from "@/components/Index" Vue.use(VueRouter) const routes = [ { path: '/', component: Login, props: true }, { path:'/index/:val', name:'index', component: Index, props: true } ] const router = new VueRouter({ mode:'history', routes:routes }) export default router
routes
表示路由,其中包含了兩個路由,一個是Login
組件的路由/
,一個是Index
組件的路由/index/:val
,後者中的:val
是佔位符,用於傳遞參數。router
表示路由器,mode
能夠選擇hash
或history
:
hash
會使用URL
的hash
來模擬一個完整的URL
,當URL
改變時頁面不會從新加載history
就是普通的正常URL
router
中的routes
參數聲明瞭對應的路由,最後要記得把router
添加到main.js
中。
vue.config.js
在package.json
同級目錄下建立vue.config.js
,內容以下:
module.exports = { chainWebpack: config => { config.module .rule('vue') .use('vue-loader') .loader('vue-loader') .tap(options => { options.transformAssetUrls = { img: 'src', image: 'xlink:href', 'b-img': 'src', 'b-img-lazy': ['src', 'blank-src'], 'b-card': 'img-src', 'b-card-img': 'src', 'b-card-img-lazy': ['src', 'blank-src'], 'b-carousel-slide': 'img-src', 'b-embed': 'src' } return options }) } }
使用該配置文件主要是由於<b-img>
的src
屬性不能正常讀取圖片,添加了該配置文件後便可按路徑正常讀取。
main.js
添加依賴以及路由:
import Vue from 'vue' import App from './App.vue' import {BootstrapVue, BootstrapVueIcons} from 'bootstrap-vue' import router from "@/router"; import 'bootstrap/dist/css/bootstrap.css' import 'bootstrap-vue/dist/bootstrap-vue.css' Vue.use(BootstrapVue) Vue.use(BootstrapVueIcons) Vue.config.productionTip = false new Vue({ render: h => h(App), router }).$mount('#app')
引入BootstrapVue
,並把路由註冊到Vue
實例中(就是倒數第2行,做爲建立Vue
實例的參數,注意這個很重要,否則路由功能不能正常使用)。
也就是Login.vue
,內容以下:
<template> <div> <b-img src="../assets/logo.png"></b-img> <br> <b-container> <b-row> <b-col offset="3" cols="6"> <b-input-group size="lg"> <b-input-group-text>用戶名</b-input-group-text> <b-form-input type="text" v-model="username"></b-form-input> </b-input-group> </b-col> </b-row> <br> <b-row> <b-col offset="3" cols="6"> <b-input-group size="lg"> <b-input-group-text>密碼</b-input-group-text> <b-form-input type="password" v-model="password"></b-form-input> </b-input-group> </b-col> </b-row> <br> <b-row> <b-col offset="3" cols="6"> <b-button variant="success" @click="login"> 一鍵註冊/登陸 </b-button> </b-col> </b-row> </b-container> </div> </template> <script> import axios from 'axios' import router from "@/router" export default { name: "Login.vue", data:function (){ return{ username:'', password:'' } }, methods:{ login:function(){ axios.post("http://localhost:8080/login",{ username:this.username, password:this.password }).then(function (res){ router.push({ name:"index", params:{ val:res.data.code === 1 } }) }) } } } </script> <style scoped> </style>
採用了網格系統佈局<b-row>
+<b-col>
,其餘組件就不說了,大部分組件官網都有說明(能夠戳這裏),發送請求採用了axios
,參數包裝在請求體中,注意須要與後端(@RequestBody
,寫在請求頭請使用@RequestParm
)對應。
另外還須要注意的是跨域問題,這裏的跨域問題交給後端處理:
@CrossOrigin("http://localhost:8081")
(本地測試中後端運行在8080
端口,而前端運行在8081
端口)
發送請求後使用路由進行跳轉,攜帶的是res.data.code
參數 ,其中res.data
是響應中的數據,後面的code
是後端自定義的數據,返回1
表示註冊成功,返回2
表示登陸成功。
首頁簡單地顯示了登陸或註冊成功:
<template> <div> <b-img src="../assets/logo.png"></b-img> <b-container> <b-row align-h="center"> <b-col> <b-jumbotron header="註冊成功" lead="歡迎" v-if="val"></b-jumbotron> <b-jumbotron header="登陸成功" lead="歡迎" v-else></b-jumbotron> </b-col> </b-row> </b-container> </div> </template> <script> export default { name: "Index.vue", props:['val'] } </script> <style scoped> </style>
props
表示val
是來自其餘組件的參數,並將其做爲在v-if
中進行條件渲染的參數。
這樣前端就作好了。下面開始介紹後端。
採用Kotlin
+Gradle
+MyBatisPlus
構建,新建工程以下:
引入MyBatis Plus
依賴便可:
implementation("com.baomidou:mybatis-plus-boot-starter:3.4.0")
create database if not exists test; use test; drop table if exists user; create table user( id int auto_increment primary key , username varchar(30) default '', password varchar(30) default '' )
數據庫用戶名+密碼+url
:
spring: datasource: url: jdbc:mysql://localhost:3306/test username: root password: 123456
新建以下六個包,分別表示配置類、控制層、持久層、實體類、響應類、業務層。
package com.example.demo.entity class User(var username:String,var password:String)
package com.example.demo.dao import com.baomidou.mybatisplus.core.mapper.BaseMapper import com.example.demo.entity.User import org.apache.ibatis.annotations.Mapper import org.apache.ibatis.annotations.Select @Mapper interface DemoMapper :BaseMapper<User>{ @Select("select * from user where username=#{username} and password = #{password}") fun selectByUsernameAndPassword(username:String,password:String):List<User> }
@Mapper
表示給Mapper
接口生成一個實現類,而且不須要編寫xml
配置文件。@Select
表示進行查詢的sql
語句。
package com.example.demo.response class DemoResponse { var data = Any() var code = 0 var message = "" }
package com.example.demo.response class DemoResponseBuilder { private var response = DemoResponse() fun data(t:Any): DemoResponseBuilder { response.data = t return this } fun code(t:Int): DemoResponseBuilder { response.code = t return this } fun message(t:String): DemoResponseBuilder { response.message = t return this } fun build() = response }
這裏響應體分爲:
與前端約定便可。生成響應體經過一個Builder
類生成。
package com.example.demo.service import com.demo.response.DemoResponse import com.demo.response.DemoResponseBuilder import com.example.demo.dao.DemoMapper import com.example.demo.entity.User import org.springframework.beans.factory.annotation.Autowired import org.springframework.stereotype.Service import org.springframework.transaction.annotation.Transactional @Service @Transactional class DemoService { @Autowired lateinit var mapper: DemoMapper fun login(username:String, password:String): DemoResponse { val result = mapper.selectByUsernameAndPassword(username,password).size if(result == 0) mapper.insert(User(username,password)) return DemoResponseBuilder().code(if(result == 0) 1 else 2).message("").data(true).build() } }
@Service
標記爲業務層,@Transactional
表示添加了事務管理,持久層操做失敗會進行回滾。@Autowired
表示自動注入,在Java
中能夠使用直接使用@Autowired
,而在Kotlin
中須要使用lateinit var
。
package com.example.demo.controller import com.demo.response.DemoResponse import com.example.demo.entity.User import com.example.demo.service.DemoService import org.springframework.beans.factory.annotation.Autowired import org.springframework.web.bind.annotation.* @RestController @RequestMapping("/") @CrossOrigin("http://localhost:8081") class DemoController { @Autowired lateinit var service: DemoService @PostMapping("login") fun login(@RequestBody user: User):DemoResponse { return service.login(user.username, user.password) } }
主要就是添加了一個跨域處理@CrossOrigin
,開發時請對應上前端的端口。
package com.example.demo.config import org.mybatis.spring.annotation.MapperScan import org.springframework.context.annotation.Configuration @Configuration @MapperScan("com.example.demo.dao") class MyBatisConfig
@MapperScan
表示掃描對應包下的@Mapper
。
package com.example.demo import com.example.demo.service.DemoService import org.junit.jupiter.api.Test import org.springframework.beans.factory.annotation.Autowired import org.springframework.boot.test.context.SpringBootTest @SpringBootTest class DemoApplicationTests { @Autowired lateinit var service: DemoService @Test fun contextLoads() { println(service.login("123", "456")) } }
測試經過後後端就算完成了。
先運行後端,Kotlin
不像Java
,生成工程時能自動配置了啓動配置,須要手動運行啓動類中的main
:
再運行前端:
npm run serve
不想用命令行的話能夠使用圖形界面配置一下:
根據控制檯輸出打開localhost:8081
:
隨便輸入用戶名與密碼,不存在則建立,存在則登陸:
註冊的同時後端數據庫會生成一條記錄:
再次輸入相同的用戶名和密碼會顯示登陸成功:
這樣就正式完成了一個簡單的先後端分離登陸Demo
。
本文由博客羣發一文多發等運營工具平臺 OpenWrite 發佈