概况
项目有controller、dao、model、utils、views、以及sql等包。
项目说明
作为教程中的练手项目,项目中使用到了包括函数、切片、接口、指针在内的go基础知识,以及action、cookie等web内容。
基于MySQL项目将使用人群分为了两块,用户和管理员。
在用户端,项目实现了登录、注册、使用购物车、结算等功能。
而管理员不仅可以使用上述功能,同时也能使用管理购物项功能。
功能相对简单但是页面简洁使用丝滑。
代码
1.主函数
package main
import (
"bookstore0612/controller"
"net/http"
)
func main() {
//设置处理静态资源,如css和js文件
http.Handle("/static/", http.StripPrefix("/static/", http.FileServer(http.Dir("/bookstore0612/views/static"))))
//直接去html页面
http.Handle("/pages/", http.StripPrefix("/pages/", http.FileServer(http.Dir("/bookstore0612/views/pages"))))
//去首页
http.HandleFunc("/main", controller.GetPageBooksByPrice)
//去登录
http.HandleFunc("/login", controller.Login)
//去注销
http.HandleFunc("/logout", controller.Logout)
//去注册
http.HandleFunc("/regist", controller.Regist)
//通过Ajax请求验证用户名是否可用
http.HandleFunc("/checkUserName", controller.CheckUserName)
//获取所有图书
// http.HandleFunc("/getBooks", controller.GetBooks)
//获取带分页的图书信息
http.HandleFunc("/getPageBooks", controller.GetPageBooks)
http.HandleFunc("/getPageBooksByPrice", controller.GetPageBooksByPrice)
//添加图书
// http.HandleFunc("/addBook", controller.AddBook)
//删除图书
http.HandleFunc("/deleteBook", controller.DeleteBook)
//去更新图书的页面
http.HandleFunc("/toUpdateBookPage", controller.ToUpdateBookPage)
//更新或添加图书
http.HandleFunc("/updateOraddBook", controller.UpdateOrAddBook)
//添加图书到购物车中
http.HandleFunc("/addBook2Cart", controller.AddBook2Cart)
//获取购物车信息
http.HandleFunc("/getCartInfo", controller.GetCartInfo)
//清空购物车
http.HandleFunc("/deleteCart", controller.DeleteCart)
//删除购物项
http.HandleFunc("/deleteCartItem", controller.DeleteCartItem)
//更新购物项
http.HandleFunc("/updateCartItem", controller.UpdateCartItem)
//去结账
http.HandleFunc("/checkout", controller.Checkout)
//获取所有订单
http.HandleFunc("/getOrders", controller.GetOrders)
//获取订单详情,即订单所对应的所有的订单项
http.HandleFunc("/getOrderInfo", controller.GetOrderInfo)
//获取我的订单
http.HandleFunc("/getMyOrder", controller.GetMyOrders)
//发货
http.HandleFunc("/sendOrder", controller.SendOrder)
//确认收货
http.HandleFunc("/takeOrder", controller.TakeOrder)
http.ListenAndServe(":8090", nil)
}
2. controller包(控制层)
我认为这个包应该是项目中最重要也是最核心的部分,了解了这个部分应该就能够对项目有一个大概的认知
2.1 bookhandle(图书管理)
图书是拥有很多属性的。管理起来也是很麻烦所以用到了四个函数。分别为GetPageBooksByPrice、GetPageBooks、DeleteBook、ToUpdateBookPage、UpdateOrAddBook。顾名思义,GetPageBooksByPrice作用是 获取带分页和价格范围的图书、GetPageBooks 的作用获取带分页的图书、DeleteBook 的作用是删除图书、ToUpdateBookPage 的作用是去更新或者添加图书的页面、UpdateOrAddBook 的作用是更新或添加图书
package controller
import (
"bookstore0612/dao"
"bookstore0612/model"
"html/template"
"net/http"
"strconv"
)
func GetPageBooksByPrice(w http.ResponseWriter, r *http.Request) {
//获取页码
pageNo := r.FormValue("pageNo")
//获取价格范围
minPrice := r.FormValue("min")
maxPrice := r.FormValue("max")
if pageNo == "" {
pageNo = "1"
}
var page *model.Page
if minPrice == "" && maxPrice == "" {
//调用bookdao中获取带分页的图书的函数
page, _ = dao.GetPageBooks(pageNo)
} else {
//调用bookdao中获取带分页和价格范围的图书的函数
page, _ = dao.GetPageBooksByPrice(pageNo, minPrice, maxPrice)
//将价格范围设置到page中
page.MinPrice = minPrice
page.MaxPrice = maxPrice
}
//调用IsLogin函数判断是否已经登录
flag, session := dao.IsLogin(r)
if flag {
//已经登录,设置page中的IsLogin字段和Username的字段值
page.IsLogin = true
page.Username = session.UserName
}
//解析模板文件
t := template.Must(template.ParseFiles("views/index.html"))
//执行
t.Execute(w, page)
}
func GetPageBooks(w http.ResponseWriter, r *http.Request) {
//获取页码
pageNo := r.FormValue("pageNo")
if pageNo == "" {
pageNo = "1"
}
//调用bookdao中获取带分页的图书的函数
page, _ := dao.GetPageBooks(pageNo)
//解析模板文件
t := template.Must(template.ParseFiles("views/pages/manager/book_manager.html"))
//执行
t.Execute(w, page)
}
func DeleteBook(w http.ResponseWriter, r *http.Request) {
//获取要删除的图书的id
bookID := r.FormValue("bookId")
//调用bookdao中删除图书的函数
dao.DeleteBook(bookID)
//调用GetBooks处理器函数再次查询一次数据库
GetPageBooks(w, r)
}
func ToUpdateBookPage(w http.ResponseWriter, r *http.Request) {
//获取要更新的图书的id
bookID := r.FormValue("bookId")
//调用bookdao中获取图书的函数
book, _ := dao.GetBookByID(bookID)
if book.ID > 0 {
//在更新图书
//解析模板
t := template.Must(template.ParseFiles("views/pages/manager/book_edit.html"))
//执行
t.Execute(w, book)
} else {
//在添加图书
//解析模板
t := template.Must(template.ParseFiles("views/pages/manager/book_edit.html"))
//执行
t.Execute(w, "")
}
}
func UpdateOrAddBook(w http.ResponseWriter, r *http.Request) {
//获取图书信息
bookID := r.PostFormValue("bookId")
title := r.PostFormValue("title")
author := r.PostFormValue("author")
price := r.PostFormValue("price")
sales := r.PostFormValue("sales")
stock := r.PostFormValue("stock")
//将价格、销量和库存进行转换
fPrice, _ := strconv.ParseFloat(price, 64)
iSales, _ := strconv.ParseInt(sales, 10, 0)
iStock, _ := strconv.ParseInt(stock, 10, 0)
ibookID, _ := strconv.ParseInt(bookID, 10, 0)
//创建Book
book := &model.Book{
ID: int(ibookID),
Title: title,
Author: author,
Price: fPrice,
Sales: int(iSales),
Stock: int(iStock),
ImgPath: "/static/img/default.jpg",
}
if book.ID > 0 {
//在更新图书
//调用bookdao中更新图书的函数
dao.UpdateBook(book)
} else {
//在添加图书
//调用bookdao中添加图书的函数
dao.AddBook(book)
}
//调用GetBooks处理器函数再次查询一次数据库
GetPageBooks(w, r)
}
2.2cartthandler(购物车部分)
购物车部分实现的功能有:添加图书、获取购物车信息、清空购物车、删除购物项、更新购物项。同样使用到了四个函数。
AddBook2Cart添加图书到购物车、GetCartInfo 根据用户的id获取购物车信息、DeleteCart 清空购物车、DeleteCartItem 删除购物项、UpdateCartItem 更新购物项、
package controller
import (
"bookstore0612/dao"
"bookstore0612/model"
"bookstore0612/utils"
"encoding/json"
"fmt"
"html/template"
"net/http"
"strconv"
)
func AddBook2Cart(w http.ResponseWriter, r *http.Request) {
//判断是否登录
flag, session := dao.IsLogin(r)
if flag {
//已经登录
//获取要添加的图书的id
bookID := r.FormValue("bookId")
//根据图书的id获取图书信息
book, _ := dao.GetBookByID(bookID)
//获取用户的id
userID := session.UserID
//判断数据库中是否有当前用户的购物车
cart, _ := dao.GetCartByUserID(userID)
if cart != nil {
//当前用户已经有购物车,此时需要判断购物车中是否有当前这本图书
carItem, _ := dao.GetCartItemByBookIDAndCartID(bookID, cart.CartID)
if carItem != nil {
//购物车的购物项中已经有该图书,只需要将该图书所对应的购物项中的数量加1即可
//1.获取购物车切片中的所有的购物项
cts := cart.CartItems
//2.遍历得到每一个购物项
for _, v := range cts {
fmt.Println("当前购物项中是否有Book:", v)
fmt.Println("查询到的Book是:", carItem.Book)
//3.找到当前的购物项
if v.Book.ID == carItem.Book.ID {
//将购物项中的图书的数量加1
v.Count = v.Count + 1
//更新数据库中该购物项的图书的数量
dao.UpdateBookCount(v)
}
}
} else {
//购物车的购物项中还没有该图书,此时需要创建一个购物项并添加到数据库中
//创建购物车中的购物项
cartItem := &model.CartItem{
Book: book,
Count: 1,
CartID: cart.CartID,
}
//将购物项添加到当前cart的切片中
cart.CartItems = append(cart.CartItems, cartItem)
//将新创建的购物项添加到数据库中
dao.AddCartItem(cartItem)
}
//不管之前购物车中是否有当前图书对应的购物项,都需要更新购物车中的图书的总数量和总金额
dao.UpdateCart(cart)
} else {
//证明当前用户还没有购物车,需要创建一个购物车并添加到数据库中
//1.创建购物车
//生成购物车的id
cartID := utils.CreateUUID()
cart := &model.Cart{
CartID: cartID,
UserID: userID,
}
//2.创建购物车中的购物项
//声明一个CartItem类型的切片
var cartItems []*model.CartItem
cartItem := &model.CartItem{
Book: book,
Count: 1,
CartID: cartID,
}
//将购物项添加到切片中
cartItems = append(cartItems, cartItem)
//3将切片设置到cart中
cart.CartItems = cartItems
//4.将购物车cart保存到数据库中
dao.AddCart(cart)
}
w.Write([]byte("您刚刚将" + book.Title + "添加到了购物车!"))
} else {
//没有登录
w.Write([]byte("请先登录!"))
}
}
func GetCartInfo(w http.ResponseWriter, r *http.Request) {
_, session := dao.IsLogin(r)
//获取用户的id
userID := session.UserID
//根据用户的id从数据库中获取对应的购物车
cart, _ := dao.GetCartByUserID(userID)
if cart != nil {
//将购物车设置到session中
session.Cart = cart
//解析模板文件
t := template.Must(template.ParseFiles("views/pages/cart/cart.html"))
//执行
t.Execute(w, session)
} else {
//该用户还没有购物车
//解析模板文件
t := template.Must(template.ParseFiles("views/pages/cart/cart.html"))
//执行
t.Execute(w, session)
}
}
func DeleteCart(w http.ResponseWriter, r *http.Request) {
//获取要删除的购物车的id
cartID := r.FormValue("cartId")
//清空购物车
dao.DeleteCartByCartID(cartID)
//调用GetCartInfo函数再次查询购物车信息
GetCartInfo(w, r)
}
func DeleteCartItem(w http.ResponseWriter, r *http.Request) {
//获取要删除的购物项的id
cartItemID := r.FormValue("cartItemId")
//将购物项的id转换为int64
iCartItemID, _ := strconv.ParseInt(cartItemID, 10, 64)
//获取session
_, session := dao.IsLogin(r)
//获取用户的id
userID := session.UserID
//获取该用户的购物车
cart, _ := dao.GetCartByUserID(userID)
//获取购物车中的所有的购物项
cartItems := cart.CartItems
//遍历得到每一个购物项
for k, v := range cartItems {
//寻找要删除的购物项
if v.CartItemID == iCartItemID {
//这个就是我们要删除的购物项
//将当前购物项从切片中移出
cartItems = append(cartItems[:k], cartItems[k+1:]...)
//将删除购物项之后的切片再次赋给购物车中的切片
cart.CartItems = cartItems
//将当前购物项从数据库中删除
dao.DeleteCartItemByID(cartItemID)
}
}
//更新购物车中的图书的总数量和总金额
dao.UpdateCart(cart)
//调用获取购物项信息的函数再次查询购物车信息
GetCartInfo(w, r)
}
func UpdateCartItem(w http.ResponseWriter, r *http.Request) {
//获取要更新的购物项的id
cartItemID := r.FormValue("cartItemId")
//将购物项的id转换为int64
iCartItemID, _ := strconv.ParseInt(cartItemID, 10, 64)
//获取用户输入的图书的数量
bookCount := r.FormValue("bookCount")
iBookCount, _ := strconv.ParseInt(bookCount, 10, 64)
//获取session
_, session := dao.IsLogin(r)
//获取用户的id
userID := session.UserID
//获取该用户的购物车
cart, _ := dao.GetCartByUserID(userID)
//获取购物车中的所有的购物项
cartItems := cart.CartItems
//遍历得到每一个购物项
for _, v := range cartItems {
//寻找要更新的购物项
if v.CartItemID == iCartItemID {
//这个就是我们要更新的购物项
//将当前购物项中的图书的数量设置为用户输入的值
v.Count = iBookCount
//更新数据库中该购物项的图书的数量和金额小计
dao.UpdateBookCount(v)
}
}
//更新购物车中的图书的总数量和总金额
dao.UpdateCart(cart)
//调用获取购物项信息的函数再次查询购物车信息
cart, _ = dao.GetCartByUserID(userID)
// GetCartInfo(w, r)
//获取购物车中图书的总数量
totalCount := cart.TotalCount
//获取购物车中图书的总金额
totalAmount := cart.TotalAmount
var amount float64
//获取购物车中更新的购物项中的金额小计
cIs := cart.CartItems
for _, v := range cIs {
if iCartItemID == v.CartItemID {
//这个就是我们寻找的购物项,此时获取当前购物项中的金额小计
amount = v.Amount
}
}
//创建Data结构
data := model.Data{
Amount: amount,
TotalAmount: totalAmount,
TotalCount: totalCount,
}
//将data转换为json字符串
json, _ := json.Marshal(data)
//响应到浏览器
w.Write(json)
}
2.3 orderhandler(处理订单)
这一部分实现的功能有:结账、获取所有订单、获取订单对应的订单项、获取个人订单、收货、发货。
package controller
import (
"bookstore0612/dao"
"bookstore0612/model"
"bookstore0612/utils"
"html/template"
"net/http"
"time"
)
//Checkout //去结账
func Checkout(w http.ResponseWriter, r *http.Request) {
//获取session
_, session := dao.IsLogin(r)
//获取用户的id
userID := session.UserID
//获取购物车
cart, _ := dao.GetCartByUserID(userID)
//生成订单号
orderID := utils.CreateUUID()
//创建生成订单的时间
timeStr := time.Now().Format("2021-07-02 15:04:05")
//创建Order
order := &model.Order{
OrderID: orderID,
CreateTime: timeStr,
TotalCount: cart.TotalCount,
TotalAmount: cart.TotalAmount,
State: 0,
UserID: int64(userID),
}
//将订单保存到数据库中
dao.AddOrder(order)
//保存订单项
//获取购物车中的购物项
cartItems := cart.CartItems
//遍历得到每一个购物项
for _, v := range cartItems {
//创建OrderItem
orderItem := &model.OrderItem{
Count: v.Count,
Amount: v.Amount,
Title: v.Book.Title,
Author: v.Book.Author,
Price: v.Book.Price,
ImgPath: v.Book.ImgPath,
OrderID: orderID,
}
//将购物项保存到数据库中
dao.AddOrderItem(orderItem)
//更新当前购物项中图书的库存和销量
book := v.Book
book.Sales = book.Sales + int(v.Count)
book.Stock = book.Stock - int(v.Count)
//更新图书的信息
dao.UpdateBook(book)
}
//清空购物车
dao.DeleteCartByCartID(cart.CartID)
//将订单号设置到session中
session.OrderID = orderID
//解析模板
t := template.Must(template.ParseFiles("views/pages/cart/checkout.html"))
//执行
t.Execute(w, session)
}
//GetOrders 获取所有订单
func GetOrders(w http.ResponseWriter, r *http.Request) {
//调用dao中获取所有订单的函数
orders, _ := dao.GetOrders()
//解析模板
t := template.Must(template.ParseFiles("views/pages/order/order_manager.html"))
//执行
t.Execute(w, orders)
}
//GetOrderInfo 获取订单对应的订单项
func GetOrderInfo(w http.ResponseWriter, r *http.Request) {
//获取订单号
orderID := r.FormValue("orderId")
//根据订单号调用dao中获取所有订单项的函数
orderItems, _ := dao.GetOrderItemsByOrderID(orderID)
//解析模板
t := template.Must(template.ParseFiles("views/pages/order/order_info.html"))
//执行
t.Execute(w, orderItems)
}
//GetMyOrders 获取我的订单
func GetMyOrders(w http.ResponseWriter, r *http.Request) {
//获取session
_, session := dao.IsLogin(r)
//获取用户的id
userID := session.UserID
//调用dao中获取用户的所有订单的函数
orders, _ := dao.GetMyOrders(userID)
//将订单设置到session中
session.Orders = orders
//解析模板
t := template.Must(template.ParseFiles("views/pages/order/order.html"))
//执行
t.Execute(w, session)
}
//SendOrder 发货
func SendOrder(w http.ResponseWriter, r *http.Request) {
//获取要发货的订单号
orderID := r.FormValue("orderId")
//调用dao中的更新订单状态的函数
dao.UpdateOrderState(orderID, 1)
//调用GetOrders函数再次查询一下所有的订单
GetOrders(w, r)
}
//TakeOrder 收货
func TakeOrder(w http.ResponseWriter, r *http.Request) {
//获取要收货的订单号
orderID := r.FormValue("orderId")
//调用dao中的更新订单状态的函数
dao.UpdateOrderState(orderID, 2)
//调用获取我的订单的函数再次查询我的订单
GetMyOrders(w, r)
}
2.4userhandler(登录注册)
package controller
import (
"bookstore0612/dao"
"bookstore0612/model"
"bookstore0612/utils"
"html/template"
"net/http"
"time"
)
//Checkout //去结账
func Checkout(w http.ResponseWriter, r *http.Request) {
//获取session
_, session := dao.IsLogin(r)
//获取用户的id
userID := session.UserID
//获取购物车
cart, _ := dao.GetCartByUserID(userID)
//生成订单号
orderID := utils.CreateUUID()
//创建生成订单的时间
timeStr := time.Now().Format("2021-07-02 15:04:05")
//创建Order
order := &model.Order{
OrderID: orderID,
CreateTime: timeStr,
TotalCount: cart.TotalCount,
TotalAmount: cart.TotalAmount,
State: 0,
UserID: int64(userID),
}
//将订单保存到数据库中
dao.AddOrder(order)
//保存订单项
//获取购物车中的购物项
cartItems := cart.CartItems
//遍历得到每一个购物项
for _, v := range cartItems {
//创建OrderItem
orderItem := &model.OrderItem{
Count: v.Count,
Amount: v.Amount,
Title: v.Book.Title,
Author: v.Book.Author,
Price: v.Book.Price,
ImgPath: v.Book.ImgPath,
OrderID: orderID,
}
//将购物项保存到数据库中
dao.AddOrderItem(orderItem)
//更新当前购物项中图书的库存和销量
book := v.Book
book.Sales = book.Sales + int(v.Count)
book.Stock = book.Stock - int(v.Count)
//更新图书的信息
dao.UpdateBook(book)
}
//清空购物车
dao.DeleteCartByCartID(cart.CartID)
//将订单号设置到session中
session.OrderID = orderID
//解析模板
t := template.Must(template.ParseFiles("views/pages/cart/checkout.html"))
//执行
t.Execute(w, session)
}
//GetOrders 获取所有订单
func GetOrders(w http.ResponseWriter, r *http.Request) {
//调用dao中获取所有订单的函数
orders, _ := dao.GetOrders()
//解析模板
t := template.Must(template.ParseFiles("views/pages/order/order_manager.html"))
//执行
t.Execute(w, orders)
}
//GetOrderInfo 获取订单对应的订单项
func GetOrderInfo(w http.ResponseWriter, r *http.Request) {
//获取订单号
orderID := r.FormValue("orderId")
//根据订单号调用dao中获取所有订单项的函数
orderItems, _ := dao.GetOrderItemsByOrderID(orderID)
//解析模板
t := template.Must(template.ParseFiles("views/pages/order/order_info.html"))
//执行
t.Execute(w, orderItems)
}
//GetMyOrders 获取我的订单
func GetMyOrders(w http.ResponseWriter, r *http.Request) {
//获取session
_, session := dao.IsLogin(r)
//获取用户的id
userID := session.UserID
//调用dao中获取用户的所有订单的函数
orders, _ := dao.GetMyOrders(userID)
//将订单设置到session中
session.Orders = orders
//解析模板
t := template.Must(template.ParseFiles("views/pages/order/order.html"))
//执行
t.Execute(w, session)
}
//SendOrder 发货
func SendOrder(w http.ResponseWriter, r *http.Request) {
//获取要发货的订单号
orderID := r.FormValue("orderId")
//调用dao中的更新订单状态的函数
dao.UpdateOrderState(orderID, 1)
//调用GetOrders函数再次查询一下所有的订单
GetOrders(w, r)
}
//TakeOrder 收货
func TakeOrder(w http.ResponseWriter, r *http.Request) {
//获取要收货的订单号
orderID := r.FormValue("orderId")
//调用dao中的更新订单状态的函数
dao.UpdateOrderState(orderID, 2)
//调用获取我的订单的函数再次查询我的订单
GetMyOrders(w, r)
}
因代码量过大,仅展示主函数以及controller包的部分。见谅。
项目总结
本次项目的实践对于初学者具有重要的意义。通过学习和完善该项目,巩固了go基础知识、学习到了web中的基础知,同时对项目有了一个基础的认知。