百木园-与人分享,
就是让自己快乐。

彩虹女神跃长空,Go语言进阶之Go语言高性能Web框架Iris项目实战-用户系统EP03

前文再续,之前一篇我们已经配置好了数据库以及模板引擎,现在可以在逻辑层编写具体业务代码了,博客平台和大多数在线平台一样,都是基于用户账号体系来进行操作,所以我们需要针对用户表完成用户账号的CURD操作。

用户后台模板

首先用户操作逻辑主要在后台展现,所以模板应该单独生成admin文件夹,和前台模板进行逻辑隔离:

cd views  
mkdir admin

随后创建用户管理页面模板user.html:

<!DOCTYPE html>  
<html lang=\"zh-CN\">  
  <head>  
    <meta http-equiv=\"Content-Type\" content=\"text/html;charset=utf-8\">  
    <meta http-equiv=\"X-UA-Compatible\" content=\"IE=edge\">  
    <meta name=\"viewport\" content=\"width=device-width, initial-scale=1\">  
    <meta name=\"applicable-device\" content=\"pc,mobile\" />  
  <title>用户管理</title>  
<meta content=\"index,follow\" name=\"robots\">  
<meta content=\"index,follow\" name=\"GOOGLEBOT\">  
<meta content=\"刘悦\"  name=\"Author\">  
  
  <meta http-equiv=\"expires\" content=\"4500\"/>  
  
   <link rel=\"stylesheet\" href=\"../assets/css/style.css\"  />  
  
   <script src=\"../assets/js/axios.js\"></script>  
    <script src=\"../assets/js/vue.js\"></script>  
  
  
  </head>  
  <body >  
  
    <div id=\"app\">  
  
    <nav class=\"navbar navbar-inverse navbar-fixed-top\">  
      <div class=\"container\">  
        <div class=\"navbar-header\">  
  
            <div class=\"switch_a nav_swich\">  
              
                <div class=\"react-toggle\">  
              <div class=\"react-toggle-track\"><div class=\"react-toggle-track-check\"><img src=\"\" width=\"16\" height=\"16\" role=\"presentation\" style=\"pointer-events: none;\"></div>  
            <div class=\"react-toggle-track-x\"><img src=\"\" width=\"16\" height=\"16\" role=\"presentation\" style=\"pointer-events: none;\"></div></div><div class=\"react-toggle-thumb\"></div></div>  
    
              </div>  
  
          <button type=\"button\" class=\"navbar-toggle collapsed\" data-toggle=\"collapse\" data-target=\"#navbar\" aria-expanded=\"false\" aria-controls=\"navbar\">  
          <span class=\"sr-only\">菜单</span>  
          <span class=\"icon-bar\"></span>  
          <span class=\"icon-bar\"></span>  
          <span class=\"icon-bar\"></span>  
          </button>  
        </div>  
        <div id=\"navbar\" class=\"collapse navbar-collapse\">  
          <ul class=\"nav navbar-nav\">  
            <li  class=\"index_nav index_index\"><a href=\"/\" title=\'用户管理\'>用户管理</a></li>  
            <li class=\"index_nav index_1\"><a href=\"/l_id_1\" title=\'文章管理\'></a></li>  
          </ul>  
  
  
            <div class=\"react-toggle bigtoggle\">  
              <div class=\"react-toggle-track\"><div class=\"react-toggle-track-check\"><img src=\"\" width=\"16\" height=\"16\" role=\"presentation\" style=\"pointer-events: none;\"></div>  
            <div class=\"react-toggle-track-x\"><img src=\"\" width=\"16\" height=\"16\" role=\"presentation\" style=\"pointer-events: none;\"></div></div><div class=\"react-toggle-thumb\"></div></div>  
  
          <div class=\"search navbar-right\" >  
            <form action=\"/Index_search\" method =\"GET\" class=\"search_form\" >  
              <input type=\"search\" name=\"text\" class=\"search_input\" placeholder=\"Search\" required=\"required\" >  
            </form>  
          </div>  
  
  
        </div>  
        
      </div>  
    </nav>  
     
    <div class=\"container\">  
    <header>  
      
      
    </header>  
      
    <section>  
      <div class=\"row\">  
        <div class=\"col-md-8\">  
           
            
            <ul class=\"g-formlist form_li\">  
                <li>  
                    <label class=\"mark\">用户名</label>  
                    <div class=\"write\">  
                        <input type=\"text\" id=\"form-name\" class=\"g-text-entry\" placeholder=\"请输入4-10字符\" />  
                        <span class=\"tip\" data-initial=\"请输入4-10字符\"></span>  
                    </div>  
                </li>  
                <li>  
                    <label class=\"mark\">密 码</label>  
                    <div class=\"write\">  
                        <input type=\"password\" id=\"form-psw\" class=\"g-text-entry\" placeholder=\"请输入6-30字符\" />  
                        <span class=\"tip\" data-initial=\"请输入6-30字符\"></span>  
                    </div>  
                </li>  
               
            </ul>  
  
            <button>提交</button>  
  
     
  
       
            
        </div>  
        </div>  
        </section>  
  
    </div>  
  
    </div>  
  
  </body>

模板目录架构如下:

└── views  
    ├── admin  
    │ └── user.html  
    ├── index.html  
    └── test.html

views根目录模板为前台模板,而admin目录下模板是为后台模板。

同时前端声明username和password变量,分别绑定用户名和密码:

const App = {  
            data() {  
                return {  
                    //用户名  
                    username: \"\",  
                    //密码  
                    password:\"\"  
                };  
            },  
            created: function() {  
  
                console.log(\"你好,女神\");  
  
            },  
            methods: {  
            },  
        };

接着构造用户添加表单,绑定表单字段:

<ul class=\"g-formlist form_li\">  
                <li>  
                    <label class=\"mark\">用户名</label>  
                    <div class=\"write\">  
                        <input v-model=\"username\" type=\"text\" id=\"form-name\" class=\"g-text-entry\" placeholder=\"请输入4-10字符\" />  
                        <span class=\"tip\" data-initial=\"请输入4-10字符\"></span>  
                    </div>  
                </li>  
                <li>  
                    <label class=\"mark\">密 码</label>  
                    <div class=\"write\">  
                        <input v-model=\"password\" type=\"password\" id=\"form-psw\" class=\"g-text-entry\" placeholder=\"请输入6-30字符\" />  
                        <span class=\"tip\" data-initial=\"请输入6-30字符\"></span>  
                    </div>  
                </li>  
               
            </ul>  
  
            <button @click=\"submit\">提交</button>

这里通过v-model关键字将表单和变量做双向绑定,同时为按钮绑定submit提交方法。

如果愿意,我们也可以针对前端的axios库进行二次封装,增加异步请求方法的复用性:

const myaxios = function (url, type, data = {}) {  
  
return new  
  
    Promise((resolve) => {  
  
        if (type === \"get\" || type === \"delete\") {  
  
  
            axios({  
  
                method: type,  
                url: url,  
                params: data  
            }).then((result) => {  
  
                resolve(result.data);  
  
            });  
  
  
        } else {  
  
            const params = new URLSearchParams();  
            for (var key in data) {  
            params.append(key,data[key]);  
            }  
            axios({  
  
                method: type,  
                url: url,  
                data:params  
            }).then((result) => {  
  
                resolve(result.data);  
  
            });  
  
        }  
  
    });  
  
}  
  
app.config.globalProperties.myaxios = myaxios;

这样,我们就可以随时使用this关键字来向后台发起异步请求了。

接着,编写后台视图,将用户后台模板渲染出来:

app.Get(\"/admin/user/\", func(ctx iris.Context) {  
  
		ctx.View(\"/admin/user.html\")  
  
	})

编译后,访问 http://localhost:5000/admin/user/,如图所示:

用户后台接口

后台接口主要负责接收前端请求的参数,然后根据请求方式类型来决定用户表的操作动作,首先构建添加接口:

app.Post(\"/admin/user_action/\", func(ctx iris.Context) {  
  
		username := ctx.PostValue(\"username\")  
		password := ctx.PostValue(\"password\")  
  
		fmt.Println(username, password)  
  
		ret := map[string]string{  
			\"errcode\": \"0\",  
			\"msg\":     \"ok\",  
		}  
		ctx.JSON(ret)  
  
	})

这里使用Post方式匹配路由/admin/user_action/,随后通过ctx结构体的PostValue函数来获取具体的参数key,然后利用ctx.JSON函数将字典序列化为Json,再返回给前端。

前端则使用之前封装好的myaxios内置方法向后端发起异步请求:

submit:function(){  
  
  
                    this.myaxios(\"http://localhost:5000/admin/user_action/\",\"post\",{\"username\":this.username,\"password\":this.password}).then(data => {  
        console.log(data)  
      });  
  
                }

后台返回:

Now listening on: http://localhost:5000  
Application started. Press CTRL+C to shut down.  
19:30:58 app         | admin admin

可以看到,后端打印出了前端请求的用户名和密码,接着就是入库操作:

app.Post(\"/admin/user_action/\", func(ctx iris.Context) {  
  
		username := ctx.PostValue(\"username\")  
		password := ctx.PostValue(\"password\")  
  
		fmt.Println(username, password)  
  
		user := &model.User{Username: username, Password: password}  
		res := db.Create(user)  
  
		fmt.Println(res.Error)  
  
		ret := map[string]string{  
			\"errcode\": \"0\",  
			\"msg\":     \"ok\",  
		}  
		ctx.JSON(ret)  
  
	})

这里初始化结构体变量user后,利用db.Create函数进行入库操作。

随后检查入库结果:

MySQL [irisblog]> select * from user;  
+----+---------------------+---------------------+------------+----------+----------+  
| id | created_at          | updated_at          | deleted_at | username | password |  
+----+---------------------+---------------------+------------+----------+----------+  
| 13 | 2022-08-22 19:33:16 | 2022-08-22 19:33:16 | NULL       | admin    | admin    |  
+----+---------------------+---------------------+------------+----------+----------+  
1 row in set (0.00 sec)

入库操作虽然成功了,但显然,密码不能使用明文,否则不就步CSDN的后尘贻笑大方了吗?

添加md5加密逻辑:

w := md5.New()  
io.WriteString(w, password) //将str写入到w中  
md5str := fmt.Sprintf(\"%x\", w.Sum(nil))

注意导入\"crypto/md5\"和\"io\"两个标准库包。

完成代码:

app.Post(\"/admin/user_action/\", func(ctx iris.Context) {  
  
		username := ctx.PostValue(\"username\")  
		password := ctx.PostValue(\"password\")  
  
		fmt.Println(username, password)  
  
		w := md5.New()  
		io.WriteString(w, password) //将str写入到w中  
		md5str := fmt.Sprintf(\"%x\", w.Sum(nil))  
  
		user := &model.User{Username: username, Password: md5str}  
		res := db.Create(user)  
  
		fmt.Println(res.Error)  
  
		ret := map[string]string{  
			\"errcode\": \"0\",  
			\"msg\":     \"ok\",  
		}  
		ctx.JSON(ret)  
  
	})

重新编译后,再次发起请求,检查入库结果:

MySQL [irisblog]> select * from user;  
+----+---------------------+---------------------+------------+----------+----------------------------------+  
| id | created_at          | updated_at          | deleted_at | username | password                         |  
+----+---------------------+---------------------+------------+----------+----------------------------------+  
| 16 | 2022-08-22 19:41:40 | 2022-08-22 19:41:40 | NULL       | admin    | 21232f297a57a5a743894a0e4a801fc3 |  
+----+---------------------+---------------------+------------+----------+----------------------------------+  
1 row in set (0.00 sec)

完成添加逻辑后,可以将用户列表批量查询出来:

app.Get(\"/admin/userlist/\", func(ctx iris.Context) {  
  
		var users []model.User  
		res := db.Find(&users)  
  
		ctx.JSON(res)  
  
	})

注意这里声明一个切片嵌套结构users,切片的每一个元素是用户结构体,接口返回:

{  
Value: [  
{  
ID: 16,  
CreatedAt: \"2022-08-22T19:41:40+08:00\",  
UpdatedAt: \"2022-08-22T19:41:40+08:00\",  
DeletedAt: null,  
Username: \"admin\",  
Password: \"21232f297a57a5a743894a0e4a801fc3\"  
},  
{  
ID: 17,  
CreatedAt: \"2022-08-22T19:48:25+08:00\",  
UpdatedAt: \"2022-08-22T19:48:25+08:00\",  
DeletedAt: null,  
Username: \"888123\",  
Password: \"202cb962ac59075b964b07152d234b70\"  
},  
{  
ID: 18,  
CreatedAt: \"2022-08-22T19:48:28+08:00\",  
UpdatedAt: \"2022-08-22T19:48:28+08:00\",  
DeletedAt: null,  
Username: \"admin123\",  
Password: \"21232f297a57a5a743894a0e4a801fc3\"  
}  
],  
Error: null,  
RowsAffected: 3  
}

随后,前端可以通过异步请求回调赋值将用户列表展示在页面中:

const App = {  
            data() {  
                return {  
                    //用户名  
                    username: \"\",  
                    //密码  
                    password:\"\",  
                    //用户列表  
                    userlist:[]  
                };  
            },  
            created: function() {  
  
                console.log(\"你好,女神\");  
  
                this.get_userlist();  
  
            },  
            methods: {  
  
                get_userlist:function(){  
  
  
                    this.myaxios(\"http://localhost:5000/admin/userlist/\",\"get\",).then(data => {  
        console.log(data)  
        this.userlist = data.Value  
      });  
  
  
                },  
                submit:function(){  
  
  
                    this.myaxios(\"http://localhost:5000/admin/user_action/\",\"post\",{\"username\":this.username,\"password\":this.password}).then(data => {  
        console.log(data)  
      });  
  
                }  
  
            },  
        };

随后,在页面中渲染userlist变量:

<table class=\"gridtable\">  
  
                <tr>  
  
                    <th>用户id</th>  
                    <th>用户名</th>  
                    <th>添加时间</th>  
                </tr>  
  
                <tr v-for=\"(item,index) in userlist\">  
                    <td>{{ item.ID }}</td>  
                    <td>{{ item.Username }}</td>  
                    <td>{{ item.CreatedAt }}</td>  
                </tr>  
  
  
  
            </table>

请求 http://localhost:5000/admin/user/ 如图所示:

Vue.js+Iris的前后端联调流程就跑通了,当然有些地方还需要封装,比如md5加密环节,后续登录模块也依然会依赖md5包,项目根目录下建立mytool目录:

mkdir mytool  
cd mytool

将md5加密封装为函数:

package mytool  
  
import (  
	\"crypto/md5\"  
	\"fmt\"  
	\"io\"  
)  
  
func Make_password(password string) string {  
  
	w := md5.New()  
	io.WriteString(w, password) //将str写入到w中  
	md5str := fmt.Sprintf(\"%x\", w.Sum(nil))  
  
	return md5str  
  
}

随后通过包名进行调用:

md5str := mytool.Make_password(password)

结语

至此,前后端分离的用户系统就构建好了,开发效率层面,基于Go lang的Iris框架并不逊色于任何动态语言框架,语法的简明程度有过之而无不及,性能层面更是不遑多让,该项目已开源在Github:https://github.com/zcxey2911/IrisBlog ,与君共觞,和君共勉。


来源:https://www.cnblogs.com/v3ucn/p/16650685.html
本站部分图文来源于网络,如有侵权请联系删除。

未经允许不得转载:百木园 » 彩虹女神跃长空,Go语言进阶之Go语言高性能Web框架Iris项目实战-用户系统EP03

相关推荐

  • 暂无文章