Beego 使用教程 8:Session 和 Cookie

beego 是一个用于Go编程语言的开源、高性能的 web 框架

beego 被用于在Go语言中企业应用程序的快速开发,包括RESTful API、web应用程序和后端服务。它的灵感来源于Tornado, Sinatra 和 Flask

beego 官网:http://beego.gocn.vip/

上面的 beego 官网如果访问不到,看这篇文章《beego 官网文档本地环境搭建》

注意:本文的 beego 文档使用的就是本地环境搭建的文档,因为官网文档已经不可用了

beego 官方 github 仓库:https://github.com/beego/beego

上一讲,讲了 beego 页面视图,需要的朋友可以查看《Beego 使用教程 7:Web 文件上传下载和错误处理》

这一讲,讲解 session 和 cookie。代码使用上一讲的代码

目录

1、Session 使用

1.1、基本使用

1.2、修改 cookies 名称

1.3、修改 session 存储位置

1.4、修改 session 存储在 redis

2、Cookie 使用

2.1、普通 Cookie 处理

2.2、加密 Cookie 处理


1、Session 使用

beego 内置了 session 模块,目前 session 模块支持的后端引擎包括 memory、cookie、file、mysql、redis、couchbase、memcache、postgres,用户也可以根据相应的接口实现自己的引擎

使用 session 前需要先开启,可通过代码设置或配置文件开启

web.BConfig.WebConfig.Session.SessionOn = true

配置文件配置,在 app.conf 中配置

sessionon = true

笔者使用在配置文件中配置的方式

1.1、基本使用

默认session 存储在 内存中

app.conf 配置文件开启 session 使用

在 controller 目录下新建 session.go ,代码是下面内容

package controller

import (
	"github.com/beego/beego/v2/server/web"
	"strconv"
)

type SessionController struct {
	web.Controller
}

func (this *SessionController) GetUserInfo() {
	user := this.GetSession("user")
	if user == nil {
		this.SetSession("user", int(1))
	} else {
		this.SetSession("user", user.(int)+1)
	}
	res := "ok"
	if user != nil {
		res = res + strconv.Itoa(user.(int))
	}
	this.Ctx.WriteString(res)
}

GetSession 获取session,SetSession 往session 中添加数据,更多的 session 相关方法看下图

在 main.go 中添加 GetUserInfo 的路由

package main

import (
	"beego-demo/controller"
	"beego-demo/filter"
	"fmt"
	"github.com/beego/beego/v2/core/config"
	"github.com/beego/beego/v2/server/web"
	"github.com/beego/beego/v2/server/web/context"
	"html/template"
	"net/http"
)

func main() {
	//通过config获取自定义配置
	workername, _ := config.String("workername")
	fmt.Println(workername)

	//执行定时任务
	//go job.DemoTask()

	//注册自动路由
	//web.AutoPrefix("api", &controller.UserController{})
	web.CtrlGet("/name", (*controller.UserController).Name)
	web.CtrlGet("/get/:id", (*controller.UserController).GetUserById)
	web.CtrlGet("/get/:id/:num", (*controller.UserController).GetUserByIdAndNum)
	//返回页面
	web.CtrlGet("/page/index", (*controller.PageController).Index)
	web.CtrlGet("/page/f", (*controller.PageController).F)
	web.CtrlGet("/page/main", (*controller.PageController).Main)
	//web输入参数
	web.CtrlGet("/pathparam/:name", (*controller.ParamController).PathParam)
	web.CtrlGet("/getparam", (*controller.ParamController).GetParam)
	web.CtrlPost("/postparam", (*controller.ParamController).PostParam)
	web.CtrlPost("/bindparam", (*controller.ParamController).BindParam)
	//上传文件
	web.CtrlPost("/upload", (*controller.FileController).Upload)
	//下载文件
	web.CtrlGet("/download", (*controller.FileController).Download)
	//错误处理
	web.CtrlGet("/getUserName", (*controller.ErrorHandlerController).GetUserName)
	web.CtrlGet("/getUserAge", (*controller.ErrorHandlerController).GetUserAge)
	web.CtrlGet("/getUserAddr", (*controller.ErrorHandlerController).GetUserAddr)
	web.CtrlGet("/getUserGender", (*controller.ErrorHandlerController).GetUserGender)
	//注册错误处理函数
	web.ErrorController(&controller.ErrorController{})

	//session
	web.CtrlGet("/getUserInfo", (*controller.SessionController).GetUserInfo)

	//注册函数式路由
	controller.RegisterFunctionalRoutes()
	//web命名空间
	controller.RegisterNamespaceRoutes()

	//过滤器
	filter.RegisterFilters()

	//开启 Admin 管理后台
	web.BConfig.Listen.EnableAdmin = true
	web.BConfig.Listen.AdminAddr = "localhost"
	web.BConfig.Listen.AdminPort = 8088

	//web.BConfig.WebConfig.ViewsPath = "pages"

	//开启post 请求 bind绑定请求体
	web.BConfig.CopyRequestBody = true

	//查看已注册路由
	tree := web.PrintTree()
	methods := tree["Data"].(web.M)
	for k, v := range methods {
		fmt.Printf("%s => %v\n", k, v)
	}

	//自定义模板函数
	web.AddFuncMap("bookName", bookName)

	//自定义401返回
	web.ErrorHandler("401", page401)
	//自定义404返回
	web.ErrorHandler("404", page404)

	web.ErrorHandler("dbError", dbError)

	web.BConfig.RecoverFunc = func(context *context.Context, config *web.Config) {
		if err := recover(); err != nil {
			context.WriteString(fmt.Sprintf("you panic, err: %v", err))
		}
	}

	web.Run()
}

// 自定义模板函数添加书名号
func bookName(in string) (out string) {
	out = "《" + in + "》"
	return
}

func page401(rw http.ResponseWriter, r *http.Request) {
	t, _ := template.New("401.html").ParseFiles(web.BConfig.WebConfig.ViewsPath + "/401.html")
	data := make(map[string]interface{})
	data["content"] = "没有访问权限"
	t.Execute(rw, data)
}

func page404(rw http.ResponseWriter, r *http.Request) {
	t, _ := template.New("404.html").ParseFiles(web.BConfig.WebConfig.ViewsPath + "/404.html")
	data := make(map[string]interface{})
	data["content"] = "页面没找到"
	t.Execute(rw, data)
}

func dbError(rw http.ResponseWriter, r *http.Request) {
	t, _ := template.New("dberror.html").ParseFiles(web.BConfig.WebConfig.ViewsPath + "/dberror.html")
	data := make(map[string]interface{})
	data["content"] = "我是自定义字符串错误类型处理函数"
	t.Execute(rw, data)
}

运行效果

浏览器访问:http://localhost:9090/getUserInfo

1.2、修改 cookies 名称

Session 默认是保存在用户的浏览器 cookies 里面的,默认名是 beegosessionID

通过代码 web.BConfig.WebConfig.Session.SessionName 设置,或配置 sessionname

笔者使用配置 sessionname 修改 cookie 名称,改成 JSESSIONID

默认的名称可以通过访问后,浏览器F12打开开发者工具查看,看下图

修改后重启项目,重新打开浏览器,访问:http://localhost:9090/getUserInfo

1.3、修改 session 存储位置

默认 session 存储在内存 memory 中,可以修改其存储在 file、mysql、redis 等

通过代码 web.BConfig.WebConfig.Session.SessionProvider 或配置文件参数 sessionprovider 修改

笔者下面讲解奖session 存储在 file 中,笔者使用配置文件的方式

# 设置 Session 的引擎,默认是 memory,目前支持还有 file、mysql、redis 等
sessionprovider = file
# 设置对应 file、mysql、redis 引擎的保存路径或者链接地址,默认值是空
sessionproviderconfig = E:\tmp\file\session

app.conf 文件内容看下图

重启项目,访问:http://localhost:9090/getUserInfo

1.4、修改 session 存储在 redis

将 session 存储在 redis 比较常见,下面说明

redis 相关配置

sessionprovider = redis
# Redis 配置信息如下所示 表示链接的地址,连接池,访问密码,没有保持为空
sessionproviderconfig = "127.0.0.1:6379,10,123456"

redis 密码配置在配置信息中 

添加 beego redis 依赖,在项目根目录执行下面命令

go get github.com/beego/beego/v2/server/web/session/redis

再执行下面命令

go mod tidy

在 main.go 中匿名引入 redis 引擎对应的包

_ "github.com/beego/beego/v2/server/web/session/redis"

main.go 代码

package main

import (
	"beego-demo/controller"
	"beego-demo/filter"
	"fmt"
	"github.com/beego/beego/v2/core/config"
	"github.com/beego/beego/v2/server/web"
	"github.com/beego/beego/v2/server/web/context"
	_ "github.com/beego/beego/v2/server/web/session/redis"
	"html/template"
	"net/http"
)

func main() {
	//通过config获取自定义配置
	workername, _ := config.String("workername")
	fmt.Println(workername)

	//执行定时任务
	//go job.DemoTask()

	//注册自动路由
	//web.AutoPrefix("api", &controller.UserController{})
	web.CtrlGet("/name", (*controller.UserController).Name)
	web.CtrlGet("/get/:id", (*controller.UserController).GetUserById)
	web.CtrlGet("/get/:id/:num", (*controller.UserController).GetUserByIdAndNum)
	//返回页面
	web.CtrlGet("/page/index", (*controller.PageController).Index)
	web.CtrlGet("/page/f", (*controller.PageController).F)
	web.CtrlGet("/page/main", (*controller.PageController).Main)
	//web输入参数
	web.CtrlGet("/pathparam/:name", (*controller.ParamController).PathParam)
	web.CtrlGet("/getparam", (*controller.ParamController).GetParam)
	web.CtrlPost("/postparam", (*controller.ParamController).PostParam)
	web.CtrlPost("/bindparam", (*controller.ParamController).BindParam)
	//上传文件
	web.CtrlPost("/upload", (*controller.FileController).Upload)
	//下载文件
	web.CtrlGet("/download", (*controller.FileController).Download)
	//错误处理
	web.CtrlGet("/getUserName", (*controller.ErrorHandlerController).GetUserName)
	web.CtrlGet("/getUserAge", (*controller.ErrorHandlerController).GetUserAge)
	web.CtrlGet("/getUserAddr", (*controller.ErrorHandlerController).GetUserAddr)
	web.CtrlGet("/getUserGender", (*controller.ErrorHandlerController).GetUserGender)
	//注册错误处理函数
	web.ErrorController(&controller.ErrorController{})

	//session
	web.CtrlGet("/getUserInfo", (*controller.SessionController).GetUserInfo)

	//注册函数式路由
	controller.RegisterFunctionalRoutes()
	//web命名空间
	controller.RegisterNamespaceRoutes()

	//过滤器
	filter.RegisterFilters()

	//开启 Admin 管理后台
	web.BConfig.Listen.EnableAdmin = true
	web.BConfig.Listen.AdminAddr = "localhost"
	web.BConfig.Listen.AdminPort = 8088

	//web.BConfig.WebConfig.ViewsPath = "pages"

	//开启post 请求 bind绑定请求体
	web.BConfig.CopyRequestBody = true

	//查看已注册路由
	tree := web.PrintTree()
	methods := tree["Data"].(web.M)
	for k, v := range methods {
		fmt.Printf("%s => %v\n", k, v)
	}

	//自定义模板函数
	web.AddFuncMap("bookName", bookName)

	//自定义401返回
	web.ErrorHandler("401", page401)
	//自定义404返回
	web.ErrorHandler("404", page404)

	web.ErrorHandler("dbError", dbError)

	web.BConfig.RecoverFunc = func(context *context.Context, config *web.Config) {
		if err := recover(); err != nil {
			context.WriteString(fmt.Sprintf("you panic, err: %v", err))
		}
	}

	web.Run()
}

// 自定义模板函数添加书名号
func bookName(in string) (out string) {
	out = "《" + in + "》"
	return
}

func page401(rw http.ResponseWriter, r *http.Request) {
	t, _ := template.New("401.html").ParseFiles(web.BConfig.WebConfig.ViewsPath + "/401.html")
	data := make(map[string]interface{})
	data["content"] = "没有访问权限"
	t.Execute(rw, data)
}

func page404(rw http.ResponseWriter, r *http.Request) {
	t, _ := template.New("404.html").ParseFiles(web.BConfig.WebConfig.ViewsPath + "/404.html")
	data := make(map[string]interface{})
	data["content"] = "页面没找到"
	t.Execute(rw, data)
}

func dbError(rw http.ResponseWriter, r *http.Request) {
	t, _ := template.New("dberror.html").ParseFiles(web.BConfig.WebConfig.ViewsPath + "/dberror.html")
	data := make(map[string]interface{})
	data["content"] = "我是自定义字符串错误类型处理函数"
	t.Execute(rw, data)
}

启动 redis 后,浏览器请求:http://localhost:9090/getUserInfo

可在redis 中查看session 信息,笔者使用 redis 工具查看

2、Cookie 使用

Beego 通过Context直接封装了对普通 Cookie 的处理方法,可以直接使用

2.1、普通 Cookie 处理

修改 session.go 为下面代码

package controller

import (
	"github.com/beego/beego/v2/server/web"
	"strconv"
)

type SessionController struct {
	web.Controller
}

func (this *SessionController) GetUserInfo() {
	user := this.GetSession("user")
	if user == nil {
		this.SetSession("user", int(1))
	} else {
		this.SetSession("user", user.(int)+1)
	}
	res := "ok"
	if user != nil {
		res = res + strconv.Itoa(user.(int))
	}
	this.Ctx.WriteString(res)
}

func (this *SessionController) PutCookie() {
	// 设置cookie 和 过期时间
	this.Ctx.SetCookie("name", "web cookie", 10)

	this.Ctx.WriteString("SetCookie ok")
}

func (this *SessionController) ReadCookie() {
	name := this.Ctx.GetCookie("name")
	this.Ctx.WriteString(name)
}

在 main.go 中添加路由

package main

import (
	"beego-demo/controller"
	"beego-demo/filter"
	"fmt"
	"github.com/beego/beego/v2/core/config"
	"github.com/beego/beego/v2/server/web"
	"github.com/beego/beego/v2/server/web/context"
	_ "github.com/beego/beego/v2/server/web/session/redis"
	"html/template"
	"net/http"
)

func main() {
	//通过config获取自定义配置
	workername, _ := config.String("workername")
	fmt.Println(workername)

	//执行定时任务
	//go job.DemoTask()

	//注册自动路由
	//web.AutoPrefix("api", &controller.UserController{})
	web.CtrlGet("/name", (*controller.UserController).Name)
	web.CtrlGet("/get/:id", (*controller.UserController).GetUserById)
	web.CtrlGet("/get/:id/:num", (*controller.UserController).GetUserByIdAndNum)
	//返回页面
	web.CtrlGet("/page/index", (*controller.PageController).Index)
	web.CtrlGet("/page/f", (*controller.PageController).F)
	web.CtrlGet("/page/main", (*controller.PageController).Main)
	//web输入参数
	web.CtrlGet("/pathparam/:name", (*controller.ParamController).PathParam)
	web.CtrlGet("/getparam", (*controller.ParamController).GetParam)
	web.CtrlPost("/postparam", (*controller.ParamController).PostParam)
	web.CtrlPost("/bindparam", (*controller.ParamController).BindParam)
	//上传文件
	web.CtrlPost("/upload", (*controller.FileController).Upload)
	//下载文件
	web.CtrlGet("/download", (*controller.FileController).Download)
	//错误处理
	web.CtrlGet("/getUserName", (*controller.ErrorHandlerController).GetUserName)
	web.CtrlGet("/getUserAge", (*controller.ErrorHandlerController).GetUserAge)
	web.CtrlGet("/getUserAddr", (*controller.ErrorHandlerController).GetUserAddr)
	web.CtrlGet("/getUserGender", (*controller.ErrorHandlerController).GetUserGender)
	//注册错误处理函数
	web.ErrorController(&controller.ErrorController{})

	//session
	web.CtrlGet("/getUserInfo", (*controller.SessionController).GetUserInfo)
	//cookie
	web.CtrlGet("/putCookie", (*controller.SessionController).PutCookie)
	web.CtrlGet("/readCookie", (*controller.SessionController).ReadCookie)

	//注册函数式路由
	controller.RegisterFunctionalRoutes()
	//web命名空间
	controller.RegisterNamespaceRoutes()

	//过滤器
	filter.RegisterFilters()

	//开启 Admin 管理后台
	web.BConfig.Listen.EnableAdmin = true
	web.BConfig.Listen.AdminAddr = "localhost"
	web.BConfig.Listen.AdminPort = 8088

	//web.BConfig.WebConfig.ViewsPath = "pages"

	//开启post 请求 bind绑定请求体
	web.BConfig.CopyRequestBody = true

	//查看已注册路由
	tree := web.PrintTree()
	methods := tree["Data"].(web.M)
	for k, v := range methods {
		fmt.Printf("%s => %v\n", k, v)
	}

	//自定义模板函数
	web.AddFuncMap("bookName", bookName)

	//自定义401返回
	web.ErrorHandler("401", page401)
	//自定义404返回
	web.ErrorHandler("404", page404)

	web.ErrorHandler("dbError", dbError)

	web.BConfig.RecoverFunc = func(context *context.Context, config *web.Config) {
		if err := recover(); err != nil {
			context.WriteString(fmt.Sprintf("you panic, err: %v", err))
		}
	}

	web.Run()
}

// 自定义模板函数添加书名号
func bookName(in string) (out string) {
	out = "《" + in + "》"
	return
}

func page401(rw http.ResponseWriter, r *http.Request) {
	t, _ := template.New("401.html").ParseFiles(web.BConfig.WebConfig.ViewsPath + "/401.html")
	data := make(map[string]interface{})
	data["content"] = "没有访问权限"
	t.Execute(rw, data)
}

func page404(rw http.ResponseWriter, r *http.Request) {
	t, _ := template.New("404.html").ParseFiles(web.BConfig.WebConfig.ViewsPath + "/404.html")
	data := make(map[string]interface{})
	data["content"] = "页面没找到"
	t.Execute(rw, data)
}

func dbError(rw http.ResponseWriter, r *http.Request) {
	t, _ := template.New("dberror.html").ParseFiles(web.BConfig.WebConfig.ViewsPath + "/dberror.html")
	data := make(map[string]interface{})
	data["content"] = "我是自定义字符串错误类型处理函数"
	t.Execute(rw, data)
}

运行效果

浏览器请求:http://localhost:9090/putCookie

和 http://localhost:9090/readCookie

存储 cookie 和 获取 cookie

2.2、加密 Cookie 处理

Beego 提供了两个方法用于辅助 Cookie 加密处理,它采用了sha256来作为加密算法,下面Secret则是加密的密钥

修改 session.go 为下面代码

package controller

import (
	"github.com/beego/beego/v2/server/web"
	"strconv"
)

type SessionController struct {
	web.Controller
}

func (this *SessionController) GetUserInfo() {
	user := this.GetSession("user")
	if user == nil {
		this.SetSession("user", int(1))
	} else {
		this.SetSession("user", user.(int)+1)
	}
	res := "ok"
	if user != nil {
		res = res + strconv.Itoa(user.(int))
	}
	this.Ctx.WriteString(res)
}

func (this *SessionController) PutCookie() {
	// 设置cookie 和 过期时间
	this.Ctx.SetCookie("name", "web cookie", 10)

	this.Ctx.WriteString("SetCookie ok")
}

func (this *SessionController) ReadCookie() {
	name := this.Ctx.GetCookie("name")
	this.Ctx.WriteString(name)
}

func (this *SessionController) PutSecureCookie() {
	//my-secret 是加密的密钥
	this.Ctx.SetSecureCookie("my-secret", "name", "web cookie")
	this.Ctx.WriteString("SetSecureCookie ok")
}

func (this *SessionController) ReadSecureCookie() {
	name, _ := this.Ctx.GetSecureCookie("my-secret", "name")
	this.Ctx.WriteString(name)
}

在 main.go 中添加路由

package main

import (
	"beego-demo/controller"
	"beego-demo/filter"
	"fmt"
	"github.com/beego/beego/v2/core/config"
	"github.com/beego/beego/v2/server/web"
	"github.com/beego/beego/v2/server/web/context"
	_ "github.com/beego/beego/v2/server/web/session/redis"
	"html/template"
	"net/http"
)

func main() {
	//通过config获取自定义配置
	workername, _ := config.String("workername")
	fmt.Println(workername)

	//执行定时任务
	//go job.DemoTask()

	//注册自动路由
	//web.AutoPrefix("api", &controller.UserController{})
	web.CtrlGet("/name", (*controller.UserController).Name)
	web.CtrlGet("/get/:id", (*controller.UserController).GetUserById)
	web.CtrlGet("/get/:id/:num", (*controller.UserController).GetUserByIdAndNum)
	//返回页面
	web.CtrlGet("/page/index", (*controller.PageController).Index)
	web.CtrlGet("/page/f", (*controller.PageController).F)
	web.CtrlGet("/page/main", (*controller.PageController).Main)
	//web输入参数
	web.CtrlGet("/pathparam/:name", (*controller.ParamController).PathParam)
	web.CtrlGet("/getparam", (*controller.ParamController).GetParam)
	web.CtrlPost("/postparam", (*controller.ParamController).PostParam)
	web.CtrlPost("/bindparam", (*controller.ParamController).BindParam)
	//上传文件
	web.CtrlPost("/upload", (*controller.FileController).Upload)
	//下载文件
	web.CtrlGet("/download", (*controller.FileController).Download)
	//错误处理
	web.CtrlGet("/getUserName", (*controller.ErrorHandlerController).GetUserName)
	web.CtrlGet("/getUserAge", (*controller.ErrorHandlerController).GetUserAge)
	web.CtrlGet("/getUserAddr", (*controller.ErrorHandlerController).GetUserAddr)
	web.CtrlGet("/getUserGender", (*controller.ErrorHandlerController).GetUserGender)
	//注册错误处理函数
	web.ErrorController(&controller.ErrorController{})

	//session
	web.CtrlGet("/getUserInfo", (*controller.SessionController).GetUserInfo)
	//cookie
	web.CtrlGet("/putCookie", (*controller.SessionController).PutCookie)
	web.CtrlGet("/readCookie", (*controller.SessionController).ReadCookie)
	web.CtrlGet("/putSecureCookie", (*controller.SessionController).PutSecureCookie)
	web.CtrlGet("/readSecureCookie", (*controller.SessionController).ReadSecureCookie)

	//注册函数式路由
	controller.RegisterFunctionalRoutes()
	//web命名空间
	controller.RegisterNamespaceRoutes()

	//过滤器
	filter.RegisterFilters()

	//开启 Admin 管理后台
	web.BConfig.Listen.EnableAdmin = true
	web.BConfig.Listen.AdminAddr = "localhost"
	web.BConfig.Listen.AdminPort = 8088

	//web.BConfig.WebConfig.ViewsPath = "pages"

	//开启post 请求 bind绑定请求体
	web.BConfig.CopyRequestBody = true

	//查看已注册路由
	tree := web.PrintTree()
	methods := tree["Data"].(web.M)
	for k, v := range methods {
		fmt.Printf("%s => %v\n", k, v)
	}

	//自定义模板函数
	web.AddFuncMap("bookName", bookName)

	//自定义401返回
	web.ErrorHandler("401", page401)
	//自定义404返回
	web.ErrorHandler("404", page404)

	web.ErrorHandler("dbError", dbError)

	web.BConfig.RecoverFunc = func(context *context.Context, config *web.Config) {
		if err := recover(); err != nil {
			context.WriteString(fmt.Sprintf("you panic, err: %v", err))
		}
	}

	web.Run()
}

// 自定义模板函数添加书名号
func bookName(in string) (out string) {
	out = "《" + in + "》"
	return
}

func page401(rw http.ResponseWriter, r *http.Request) {
	t, _ := template.New("401.html").ParseFiles(web.BConfig.WebConfig.ViewsPath + "/401.html")
	data := make(map[string]interface{})
	data["content"] = "没有访问权限"
	t.Execute(rw, data)
}

func page404(rw http.ResponseWriter, r *http.Request) {
	t, _ := template.New("404.html").ParseFiles(web.BConfig.WebConfig.ViewsPath + "/404.html")
	data := make(map[string]interface{})
	data["content"] = "页面没找到"
	t.Execute(rw, data)
}

func dbError(rw http.ResponseWriter, r *http.Request) {
	t, _ := template.New("dberror.html").ParseFiles(web.BConfig.WebConfig.ViewsPath + "/dberror.html")
	data := make(map[string]interface{})
	data["content"] = "我是自定义字符串错误类型处理函数"
	t.Execute(rw, data)
}

运行效果

浏览器请求:http://localhost:9090/putCookie

和 http://localhost:9090/readCookie

存储 cookie 和 获取 cookie

更多API用法可以查看官方文档


 

下一讲:《Beego 使用教程 9:ORM 操作数据库(上)》

至此完

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/656935.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

地下18米的科技守护:旗晟综合管廊巡检机器人

近日,安徽某业主的地下18米深的地下管廊处,一种先进的巡检机器人正活跃在管廊轨道上,执行着重要的巡检任务,只见机器人在管廊轨道上平稳前行,它搭载着先进的检测设备,对地下管廊内的各种设施进行监测巡检&a…

养猫久了才知道,为什么宠物空气净化器是养猫必备!效果惊人!

养猫是一件非常愉快的事情,猫咪的陪伴能带给我们无尽的欢乐和温暖。然而,随着时间的推移,许多养猫的朋友会发现一个问题,那就是家中的空气质量变差了,猫毛、异味等问题也随之而来。这时候,一款好的宠物空气…

【Python】 Python中的“命名元组”:简单而强大的数据结构

基本原理 在Python中,namedtuple是tuple的一个子类,它允许我们为元组的每个位置指定一个名字。这种数据结构非常适合用于需要固定字段和值的场景,例如数据库查询的结果或配置文件中的设置。 namedtuple提供了一种方便的方式来访问元组中的元…

mysql中单表查询的成本

大家好。我们知道MySQL在执行一个查询时,经常会有多个执行方案,然后从中选取成本最低或者说代价最低的方案去真正的执行查询。今天我们来聊一聊单表查询的成本。 那么到底什么是成本呢?这里我们说的成本或者代价是由两方面组成的&#xff1a…

Android性能优化方案

1.启动优化: application中不要做大量耗时操作,如果必须的话,建议异步做耗时操作2.布局优化:使用合理的控件选择,少嵌套。(合理使用include,merge,viewStub等使用)3.apk优化(资源文件优化&#…

opencv c++编程基础

1、图片的本质 图像在 OpenCV 中的本质 在 OpenCV 中,图像被表示为一个多维数组,其中每个元素对应于图像中的单个像素。图像的维度取决于其通道数和像素数。 **通道数:**图像可以有多个通道,每个通道存储图像的不同信息。例如&…

大学生选择算法向还是嵌入式向?

在开始前刚好我有一些资料,是我根据网友给的问题精心整理了一份「嵌入式的资料从专业入门到高级教程」, 点个关注在评论区回复“888”之后私信回复“888”,全部无偿共享给大家!!! 由于嵌入式的薪资待遇和…

【oracle】Oracle RAC中的GNS到底是什么?

本文为云贝教育 刘峰 原创,请尊重知识产权,转发请注明出处,不接受任何抄袭、演绎和未经注明出处的转载 一、概述 Oracle Grid Naming Service (GNS) 是Oracle Grid Infrastructure的一个重要组件,它提供了一种集中式的命名服务&…

33 mid 55. 跳跃游戏

贪心算法&#xff1a; class Solution {public boolean canJump(int[] nums) {int leftBorder 0;for (int i 0; i <nums.length; i) {if(i<leftBorder){leftBorderMath.max(leftBorder,inums[i]);}if(leftBorder>nums.length-1){return true;}}return false;} }

元宇宙vr科普馆场景制作引领行业潮流

在这个数字化高速发展的时代&#xff0c;北京3D元宇宙场景在线制作以其独特的优势&#xff0c;成为了行业内的创新引领者。它能够快速完成空间设计&#xff0c;根据您的个性化需求&#xff0c;轻松设置布局、灯光、音效以及互动元素等&#xff0c;为您打造出一个更加真实、丰富…

TCP—三次握手和四次挥手

目录 一、三次握手和四次挥手的目的 二、TCP可靠的方面 三、什么是三次握手 四、第三次握手的目的 五、什么是四次挥手 六、超时时间的目的 七、SYN包、ACK包、FIN包 八、解决丢包和乱序 九、参考资料 一、三次握手和四次挥手的目的 TCP三次握手的目的主要是为了确保两…

谷歌地图 | Google I/O ‘24 重磅发布助力企业拓展海外市场的新功能!

编者按&#xff1a;本文是 Google I/O 2024 系列的一部分&#xff0c;该系列分享了Google 年度开发者大会上最新的 Google Maps Platform 新闻。 距全球首个 Google Maps API 问世已近 20 年。它引领了网络和移动端地理空间体验的革命。从那时起&#xff0c;Google Maps Platf…

Common Lisp笔记

在计划学习函数式编程的时候&#xff0c;我一开始打算学习的是 F#。因为我朋友就是在 DTU 上的学&#xff0c;F# 就是 DTU&#xff08;丹麦理工&#xff09;开发的。但是由于 F# 和微软的 .NET 绑定&#xff0c;而在 macOS 上&#xff0c;目前版本的 .NET 的是有些问题的&#…

2023年平均工资公布!你是什么段位?

来源国家统计局&#xff08;下同&#xff09; 01 城镇非私营单位就业人员年平均工资情况 2023年&#xff0c;全国城镇非私营单位就业人员年平均工资为120698元&#xff0c;比上年增加6669元&#xff0c;名义增长5.8%&#xff0c;扣除价格因素实际增长5.5%。 2014-2023年城镇非私…

基于FPGA实现LED的闪烁——HLS

基于FPGA实现LED的闪烁——HLS 引言&#xff1a; ​ 随着电子技术的飞速发展&#xff0c;硬件设计和开发的速度与效率成为了衡量一个项目成功与否的关键因素。在传统的硬件开发流程中&#xff0c;工程师通常需要使用VHDL或Verilog等硬件描述语言来编写底层的硬件逻辑&#xff0…

Pytorch深度学习实践笔记12(b站刘二大人)

&#x1f3ac;个人简介&#xff1a;一个全栈工程师的升级之路&#xff01; &#x1f4cb;个人专栏&#xff1a;pytorch深度学习 &#x1f380;CSDN主页 发狂的小花 &#x1f304;人生秘诀&#xff1a;学习的本质就是极致重复! 《PyTorch深度学习实践》完结合集_哔哩哔哩_bilibi…

锐捷网络与您相约第七届数字中国建设峰会 共话数字未来

第七届数字中国建设峰会将于5月24日至25日在福建福州举办,本届峰会是国家数据工作体系优化调整后首次举办的数字中国建设峰会,主题是“释放数据要素价值,发展新质生产力”。作为行业领先的ICT基础设施及解决方案提供商,锐捷网络与福建省电子信息集团、星网锐捷,围绕“发展新质生…

newinit.sh挖矿攻击处理与规避方案

目录 攻击分析 恢复措施&#xff1a; 问题排查 攻击入口分析 预防 临时处理方案&#xff1a; 攻击分析 攻击者&#xff1a;职业黑客&#xff08;99%&#xff09; 攻击方式&#xff1a;挖矿病毒newinit.sh和蠕虫病毒pnscan 中毒现象: 服务器负载异常&#xff0c;具体表…

06_知识点总结(JS高级)

一、进程与线程 1. 进程(process)&#xff1a;程序的一次执行, 它占有一片独有的内存空间 2. 线程(thread)&#xff1a; 是进程内的一个独立执行单元&#xff0c;CPU的基本调度单元, 是程序执行的一个完整流程 3. 进程与线程 * 应用程序必须运行在某个进程的某个线程上 * 一个…

反射、类加载、静态代理,jdk动态代理,cglib代理

一、 反射 反射是在程序运行状态下&#xff0c;动态获取类的结构&#xff08;属性&#xff0c;构造器&#xff0c;方法&#xff0c;注解&#xff09;&#xff0c;动态的创建类对象然后调用类中的属性方法。反射的起源Class&#xff0c;Class中包含类反射要使用的API 获取Class的…