golang小游戏:飞翔的小鸟

news2025/1/20 17:02:06

游戏开发总体思路

首先要选取一个合适的图形化界面进行开发。该项目选取的是 ebiten 一个用于创建2D游戏和图形应用程序的游戏引擎,提供了一些简单的GUI功能。

其次明确游戏设计思路。飞翔的小鸟共分为三个场景。

第一个场景就是游戏开始前的准备阶段,让玩家点击屏幕确定游戏开始。

第二个场景就是让游戏正式开始,玩家可以操控小鸟进行游戏。

第三个场景就是游戏结束,显示分数的阶段。

一、先让窗口和背景绘制出来

先简单的让窗口显示出来,通过ebiten.RunGame(&game)启动游戏引擎,会开始初始化游戏并且开始循环执行游戏的更新和渲染逻辑。

游戏运行后 会自动调用Update函数进行屏幕刷新,然后再进行游戏绘制。

一般情况下Update函数是写游戏逻辑的,Draw函数进行游戏回话。

func main() {
	// 设置窗口大小是
	ebiten.SetWindowSize(880, 520)
	// 设置窗口头部,显示 飞翔的小鸟
	ebiten.SetWindowTitle("飞翔的小鸟")

	// 运行游戏
	err := ebiten.RunGame(&game);
	if err != nil {
		log.Fatal(err)
	}
}

/*
	Layout()函数的返回值表示显示窗口里面逻辑上屏幕的大小
	官网上说参数outsideWidth和outsideHeight是显示在桌面的窗口大小
	这里是固定大小
*/
func (g *Game) Layout(outsideWidth, outsideHeight int) (screenWidth, screenHeight int) {
	return 880, 520
}

func (g *Game) Update(screen *ebiten.Image) error {
	g.DrawBegin(screen)
	return nil
}

func (g *Game) DrawBegin(screen *ebiten.Image) {
	DrawBackGround(screen)
}
}

这里是进行游戏背景的渲染的,screen.Fill 是将整个画面都填充完一种颜色。对于地板来说,进行一张图片通过for循环来进行反复绘制。

大概就是这个样子,for循环,每次只用让图片的x坐标进行增加就可以了,这样就把背景给渲染出来了。

在这里插入图片描述

func DrawBackGround(screen *ebiten.Image){
	//  背景颜色
	screen.Fill(color.RGBA{78, 192, 202,255})
	//  绘制地板
	f1, err := os.Open("imgs/ground.png")
	if err != nil {
		log.Fatal(err)
	}
	img1, err := png.Decode(f1)
	if err != nil {
		log.Fatal(err)
	}
	filter1 := ebiten.FilterNearest
	// 把Image文件转成ebiten.Image文件,用于展示
	eImg1, _ := ebiten.NewImageFromImage(img1,filter1)

	var groundX int = 0;
	var groundY int = 437;
	for i :=1;i<5;i++ {
		op1 := &ebiten.DrawImageOptions{}
		op1.GeoM.Translate(float64(groundX), float64(groundY))   // 图像坐标
		groundX+=250
		// 在屏幕上展示出图片
		screen.DrawImage(eImg1, op1)

	}

在这里插入图片描述

这样一个带有背景图片的窗口就绘制完成了。

二、游戏准备阶段

该阶段制作相对容易,因为此时只用基于上面那个画面,添加一个鼠标点击事件,让玩家点击后即可进入到下一个场景。

先传入两个照片,一个是显示游戏名,一个是点击进入到游戏的图片。

然后对第二个图片添加一个鼠标点击事件,让玩家点击图片进入到游戏中!

ebiten.IsMouseButtonPressed是鼠标点击事件,关键要确定好鼠标点击的范围!

func (g *Game) DrawReady(screen *ebiten.Image)  {
	g.DrawBegin(screen)

	imageObject1, _, _ := ebitenutil.NewImageFromFile("imgs/title.png",ebiten.FilterDefault)
	imageObject2, _, _ := ebitenutil.NewImageFromFile("imgs/start.png",ebiten.FilterDefault)

	gameReady := &GameReady{
		imgReady1: imageObject1,
		img1X: 370,
		img1Y: 140,
		imgReady2: imageObject2,
		img2X: 333,
		img2Y: 200,
	}

	op := &ebiten.DrawImageOptions{}
	op.GeoM.Translate(float64(gameReady.img1X),float64(gameReady.img1Y))
	screen.DrawImage(gameReady.imgReady1,op)

	op.GeoM.Reset()
	op.GeoM.Translate(float64(gameReady.img2X),float64(gameReady.img2Y))
	screen.DrawImage(gameReady.imgReady2,op)

	if ebiten.IsMouseButtonPressed(ebiten.MouseButtonLeft) {
		x,y := ebiten.CursorPosition()
		rect := image.Rect(gameReady.img2X,gameReady.img2Y,gameReady.img2X+200,gameReady.img2Y+30)
		if (x >=rect.Min.X && x<=rect.Max.X && y>=rect.Min.Y && y<= rect.Max.Y ){
			gameState = 1
		}
	}

}

在这里插入图片描述

三、游戏开始阶段

这个场景有两个重要的对象需要进行处理。小鸟和障碍物。

小鸟:需要进行绘制,并且给小鸟添加逻辑。每次跳跃修改坐标,并且碰到天空和地面进行over处理。

小鸟的制作

绘制

首先进行小鸟的绘制。绘制其实只用读取小鸟的图片,然后给小鸟设置一个初始坐标,然后展示在屏幕中。

f, err := os.Open(g.Bird)
	if err != nil {
		log.Fatal(err)
	}
	img, err := png.Decode(f)
	if err != nil {
		log.Fatal(err)
	}
	filter := ebiten.FilterNearest

	// 把Image文件转成ebiten.Image文件,用于展示
	eImg, _ := ebiten.NewImageFromImage(img,filter)

	op := &ebiten.DrawImageOptions{}
	op.GeoM.Translate(g.LocationX,g.LocationY)

	// 在屏幕上展示出图片
	screen.DrawImage(eImg, op)
逻辑处理

小鸟因为只用上下移动,它的逻辑处理起来还是比较简单的,只用通过按键响应监听空格,每按一下空格,小鸟的坐标就减少,松开空格后坐标就想加。(这是因为 这个图像左上角的坐标是0 0,向右和向下分别增加x和y的坐标值)。

同时也要对 小鸟状态进行判断,当触碰到天空和地面时,进行over处理,然后每次按

//  按空格 跳跃
	if ebiten.IsKeyPressed(ebiten.KeySpace) && g.BirdState != 0 && !isSpace && gameState == 1 &&!isstop {
		isSpace = true
		locationTemp = -5
		g.Bird = "imgs/up.png"
	}

	//  松开空格
	if !ebiten.IsKeyPressed(ebiten.KeySpace) && gameState == 1{
		locationTemp = 2
		g.Bird = "imgs/down.png"
		isSpace = false
	}

	//  判断是否暂停,然后判断 小鸟此时 能否移动
	if !isstop { // 暂停
		g.LocationY = g.LocationY + float64(locationTemp)
	}

	// 碰到下边界  和  上边界
	if (g.LocationY >= 409 || g.LocationY <= -4 ){
		g.Bird = "imgs/die.png"
		g.BirdState = 0
	}
	//  死亡不让超出下边界
	if (g.LocationY >= 479){
		g.LocationY = 479;
	}

	//  小鸟死亡
	if (g.BirdState == 0){
		g.Bird = "imgs/die.png"
		g.LocationY +=3
		locationTemp = 3
	}

在这里插入图片描述

障碍物的制作

障碍物的制作也分为两个方面,一个就是障碍物的绘制和逻辑处理。

障碍物的逻辑处理还是相对简单的,因为障碍物只需要一直向左边移动,所以只用改变障碍物的x坐标就可以了。

复杂的是障碍物的绘制,因为障碍物有三种形态,每种形态的绘制还是随机的,不能重复。

绘制

考虑到障碍物的长度不能一样,所以就想到了通过 随机数来确定障碍物的长度。并且也需要通过随机数来确定障碍物的哪一种形态的。

可以先获取一个随机数,来确定障碍物的长度。然后通过for循环来绘制障碍物。(通过for循环是因为障碍物 和 地方一样 它们的图片都是非常短的一截,确定长度后通过循环进行绘制)。

下面是三种障碍物的绘制方法,绘制后将其存入到切片中。

var obstacles []*NewBarrier // 切片存数组
// 下方障碍物
func (o *NewBarrier) DrawTop(screen *ebiten.Image) {
	for i := 1 ;i<o.TopLength;i++{
		op := &ebiten.DrawImageOptions{}
		op.GeoM.Translate(float64(o.X),float64(o.TopY+(i-1)*20))
		screen.DrawImage(o.Image,op)
	}
	op1 := &ebiten.DrawImageOptions{}
	op1.GeoM.Translate(float64(o.X-2),float64(o.TopLength*20-20))
	screen.DrawImage(o.BottomImage,op1)
}

// 上方障碍物
func (o *NewBarrier) DrawBottom(screen *ebiten.Image) {
	for i := 1 ;i<19-o.TopLength;i++{
		op := &ebiten.DrawImageOptions{}
		op.GeoM.Translate(float64(o.X),float64(o.BottomY-(i-1)*20))
		screen.DrawImage(o.Image,op)
	}
	op1 := &ebiten.DrawImageOptions{}
	op1.GeoM.Translate(float64(o.X-2),float64(500-(19-o.TopLength)*20+20))
	screen.DrawImage(o.UpImage,op1)
}

// 中间障碍物
func (o *NewBarrier) DrawMid(screen *ebiten.Image) {
	//fmt.Println(o.HoverUPY,o.TopLength,o.HoverBottomY)
	for i := 1 ;i<o.TopLength;i++{
		op := &ebiten.DrawImageOptions{}
		op.GeoM.Translate(float64(o.X),float64(o.HoverUPY+i*20))
		screen.DrawImage(o.Image,op)
	}
	op1 := &ebiten.DrawImageOptions{}
	op1.GeoM.Translate(float64(o.X-2),float64(o.HoverBottomY))
	screen.DrawImage(o.BottomImage,op1)
	op2 := &ebiten.DrawImageOptions{}
	op2.GeoM.Translate(float64(o.X-2),float64(o.HoverUPY))
	screen.DrawImage(o.UpImage,op2)
}

//  绘制障碍物
func (g *Game) DrawBarrier(screen *ebiten.Image){
	if (gameState != 0){
		for _, o := range obstacles {
			if (o.BarrierStates ==0 || o.BarrierStates==1){
				o.DrawTop(screen)
				o.DrawBottom(screen)
			}else{
				o.DrawMid(screen)
			}
			if (o.X<200){
				if (!o.isScore){
					o.isScore = true
					Score++
				}
			}
		}
	}
}
逻辑处理

障碍物的逻辑处理其实非常简单,只用遍历切片,让每个障碍物的坐标一直减小就可以了。

// 让障碍物移动起来
func obojsaasdf(obs []*NewBarrier){
	for i:=0;i< len(obs);i++ {
		obs[i].X --
	}
}
障碍物对象池

因为障碍物的数量考虑到非常多,并且一直有障碍物会移出屏幕,移出屏幕之后便看不见了,如果不对其进行妥善处理,我感觉会对内存有很大的消耗。所以就引入了类似于java对象池一样的池子。将超出屏幕的障碍物 修改其部分属性 然后重新添加到切片末尾,这样便只用初始化几个障碍物,然后这些障碍物反复利用,便可大大节省内存消耗。

func deal()  {
	if len(obstacles) > 0 {
		rand.Seed(time.Now().UnixNano())
		for i := 0; i < len(obstacles); i++{
			if obstacles[i].X < -50 {
				obstacles[i].TopLength = util.Random(15,2)
				// 移除超出屏幕的障碍物
				del := obstacles[0]
				del.X = obstacles[len(obstacles)-1].X+180
				ran := util.Random(15,2)
				ran1 := util.Random(180,80)
				del.TopLength = ran
				del.HoverUPY = ran1
				del.HoverBottomY = ran1 + 20*ran
				del.BarrierStates = util.Random(3,0)
				del.isScore = false
				obstacles = append(obstacles[:i], obstacles[i+1:]...)
				// 添加新的障碍物到切片末尾
				obstacles = append(obstacles, del)
			}
		}
	}
}

在这里插入图片描述

碰撞检测处理

当小鸟和障碍物都绘制完了,就需要考虑他们之间的碰撞检测了!

碰撞检测采用的是获取小鸟四个角的坐标和障碍物的坐标,让他们的坐标没有交集。

绿色的是障碍物,蓝色的是小鸟。有一点抽象了。。。图中这几种情况是不会发生碰撞的情况。

所以,对于上下型障碍物来说,当小鸟的右边 < 障碍物的左边 或者 小鸟的左边 > 障碍物的右边 或者 小鸟的上边 > 障碍物的下面 并且小鸟的下面 < 障碍物的上面

对于悬浮障碍物来说,x 坐标考虑和上面一样,对于 y 坐标。小鸟的下面 < 障碍物的上面 或者 小鸟的上面 > 障碍物的下面

在这里插入图片描述

// 碰撞检测
func IsColliding(birdX, birdY float64, birdWidth, birdHeight float64, barrierX, barrierTopY, barrierBottomY float64, barrierWidth int,HoverUp float64,HoverBottom float64) bool {
	birdLeft := birdX
	birdTop := birdY
	birdRight := birdLeft + birdWidth
	birdBottom := birdTop + birdHeight

	barrierLeft := barrierX
	barrierTop := barrierTopY
	barrierRight := barrierLeft + float64(barrierWidth)
	barrierBottom := barrierBottomY

	if birdRight < barrierLeft || birdLeft > barrierRight || ((birdBottom < barrierBottom) && (birdTop > barrierTop)) || ((birdBottom < HoverUp) || (birdTop > HoverBottom+20)) {
		// 没有碰撞
		return false
	}

	// 有碰撞
	return true
}

在这里插入图片描述

分数绘制

游戏中应该在添加一个记录成绩的文本。

小鸟每次过一个障碍物让其分数 +1,考虑到小鸟一直是原地上下移动,是障碍物在一直移动,所以可以考虑通过障碍物的坐标来记录分数。每当障碍物的 x 坐标 小于 小鸟的 x 坐标的时候,便让成绩+1即可。

func (g *Game) DrawBarrier(screen *ebiten.Image){
	if (gameState != 0){
		for _, o := range obstacles {
			if (o.X<200){
				if (!o.isScore){
					o.isScore = true
					Score++
				}
			}
		}
	}
}
func (g *Game) DrawScore(screen *ebiten.Image) {
	ebitenutil.DebugPrintAt(screen , Itoa(Score),100,100)
}

在这里插入图片描述

游戏暂停

这里是在游戏过程中,添加一个游戏暂停处理。

其实逻辑很简单,添加一个按键监听,当按下暂停键后,让 小鸟保持不动,障碍物不再移动即可。

那具体怎么处理呢? 小鸟的跳跃是通过按键响应来控制的,那么当调用这个按键响应的同时添加一个 判断即可,判断此时 是否暂停。

同样的,障碍物移动的时候,也添加一个判断,判断此时是否暂停。

var isstop bool = false // 游戏是否暂停
//  判断游戏是否暂停
	if ebiten.IsKeyPressed(ebiten.KeyS) && isPressStop && gameState ==1 && g.BirdState != 0{
		isPressStop = false
		if (!isstop){
			isstop = true
		}else {
			isstop = false
		}
	}
//  判断是否暂停,然后判断 小鸟此时 能否移动
	if !isstop { // 暂停
		g.LocationY = g.LocationY + float64(locationTemp)
	}
	// 判断游戏是否暂停
	if ((g.LocationY != 482 || g.BirdState !=0) && !isstop && gameState != 0){ // 没有暂停
		obojsaasdf(obstacles)
	}else if  ((g.LocationY != 482 || g.BirdState != 0 ) && isstop) { // 游戏暂停
			//fmt.Println("暂停了 ,障碍物不能动了")
	}

四、游戏结束阶段

游戏结束

界面绘制

游戏结束后,需要绘制的就是 “GameOver” 和 这一局的分数了。还有就是让障碍物不在移动,这个在下面的逻辑处理细讲。

其实很简单,就是获取字符串内容,然后显示在屏幕中间即可 - -

// 绘制 GameOver方法
func (g *Game) DrawGameOver(screen *ebiten.Image) {
	ebitenutil.DebugPrintAt(screen , "GameOver",400,210)
	ebitenutil.DebugPrintAt(screen , Itoa(Score),422,240)
}

// 绘制成绩
func (g *Game) DrawScore(screen *ebiten.Image) {
	ebitenutil.DebugPrintAt(screen , Itoa(Score),100,100)
}
逻辑处理

当游戏暂停后,停止调用障碍物移动的方法就行了。

// 判断游戏是否结束
	if ((g.LocationY != 482 || g.BirdState !=0) && !isstop && gameState != 0){ // 没有结束
		obojsaasdf(obstacles)
	}else if  ((g.LocationY != 482 || g.BirdState != 0 ) && isstop) { // 游戏暂停
			//fmt.Println("暂停了 ,障碍物不能动了")
	}else if(g.LocationY == 482 || g.BirdState == 0 ) {  // 游戏结束
		gameState = 2
		g.DrawBegin(screen)
		g.DrawGameOver(screen)
	}

在这里插入图片描述

游戏重开

逻辑处理

添加一个按键响应,当按下重开键后,重新开始游戏。

重点是:要将信息全部初始化:初始化障碍物的切片,初始化小鸟状态 和 坐标,初始化该局分数。

// 此时游戏结束 考虑是否重开
	if (gameState == 2){
		if ebiten.IsKeyPressed(ebiten.KeySpace) {   // 按压空格后 重开
			Score = 0
			gameState = 1
			g.BirdState = 1
			g.LocationY = 200
			obstacles = obstacles[:0] // 清空 原来切片中的障碍物
			makeBarrier()
		}
	}
界面绘制

重新调用游戏开始阶段的函数即可。

在这里插入图片描述

五、总结感想

做完这个游戏熟悉了 go 的基础语法,增加了对代码的手感。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

微信小程序:点击按钮出现右侧弹窗

效果 代码 wxml <!-- 弹窗信息 --> <view class"popup-container" wx:if"{{showPopup}}"><view class"popup-content"><!-- 弹窗内容 --><text>这是一个右侧弹窗</text></view> </view> <…

【Spring】使用aop切面编程时要给那些类加注解

&#x1f384;欢迎来到边境矢梦的csdn博文&#x1f384; &#x1f384;本文主要梳理 Spring 中使用aop切面编程时要给那些类加注解 &#x1f384; &#x1f308;我是边境矢梦&#xff0c;一个正在为秋招和算法竞赛做准备的学生&#x1f308; &#x1f386;喜欢的朋友可以关注一…

python 之运算符

文章目录 总的介绍代码例子 总的介绍 Python中有许多不同类型的运算符&#xff0c;它们用于执行各种操作&#xff0c;包括算术运算、比较、逻辑运算等。以下是Python中常用的运算符&#xff1a; 算术运算符&#xff1a; &#xff1a;加法&#xff0c;用于将两个数相加。-&…

PhpStorm快速注释与取消注释

ctrl / 单行注释 ctrl shift / 多行注释 重复以上操作&#xff0c;取消注释。

Slax Linux 获得增强的会话管理和启动参数选项

Slax Linux 的创建者和维护者托马斯-马特吉切克&#xff08;Tomas Matejicek&#xff09;在自己生日这天&#xff08;生日快乐&#xff01;&#xff09;发布了其小巧便携的 GNU/Linux 发行版的新版本&#xff0c;带来了各种增强功能和错误修复。 新发布的 Slax Linux 版本&…

优雅的使用String字符串处理各种类型转换

文章目录 &#x1f31f; 优雅的使用String字符串处理各种类型转换&#x1f34a; 基本类型转字符串&#x1f34a; 字符串转基本类型&#x1f34a; 字符串与字符数组的转换&#x1f34a; 字符串与字节数组的转换&#x1f34a; 其他类型转字符串&#x1f34a; 总结 &#x1f4d5;我…

AIGC实战——深度学习 (Deep Learning, DL)

AIGC实战——深度学习 0. 前言1. 深度学习基本概念1.1 基本定义1.2 非结构化数据 2. 深度神经网络2.1 神经网络2.2 学习高级特征 3. TensorFlow 和 Keras4. 多层感知器 (MLP)4.1 准备数据4.2 构建模型4.3 检查模型4.4 编译模型4.5 训练模型4.6 评估模型 小结系列链接 0. 前言 …

UE4 中可全局获取的变量(例如游戏实例、玩家控制器等) 详解

目录 0 引言1 全局对象&#xff08;全局变量&#xff09;1.1 游戏实例 GameInstance1.1.1 介绍1.1.2 使用 GameInstance 1.2 玩家控制器 PlayerController1.3 游戏世界类 UWorld &#x1f64b;‍♂️ 作者&#xff1a;海码007&#x1f4dc; 专栏&#xff1a;UE虚幻引擎专栏&…

Vue3.3指北(二)

Vue3.3指北 Vue31、组件基础1.1、全局组件1.2、局部组件1.3、组件的命名1.4、组件的数据存放1.5、组件标签化 2、父组件向子组件传递数据2.1、props2.2、动态props2.3、props传数组2.4、props传对象2.4.1、默认值和必传值 3、子组件向父组件传递数据4、父子组件互相访问4.1、父…

[SQL开发笔记]AND OR运算符复杂表达式开发实例

结合 AND & OR实例&#xff1a;通过圆括号使用and或or来组成复杂的表达式 目标数据库及表&#xff1a;使用 DRobot数据库&#xff0c;"T_Drobot" 表 假设我们需要查询"T_Drobot" 表&#xff0c;并从"T_Drobot"表中查询选取creator为 "…

腾讯云 AI 绘画:文生图、图生图、图审图 快速入门

腾讯云 AI 绘画是腾讯云推出的一款基于人工智能的图像生成和编辑产品&#xff0c;能够根据输入的图片或描述文本&#xff0c;智能生成与输入内容相关的图片&#xff0c;支持多样化的图片风格选择。 在本文中&#xff0c;我们将介绍如何使用腾讯云 AI 绘画的三项主要功能&#…

24装饰器

目录 1、一些案例导入&#xff1a;这个要求必须是&#xff0c;在发东西之前有那个登录验证 2、开始理装饰器的一些原理了 3、最后就是把代码再简洁一些了 1、一些案例导入&#xff1a;这个要求必须是&#xff0c;在发东西之前有那个登录验证 def fss():print(发说说) def ft…

Elasticsearch(十五)搜索---搜索匹配功能⑥--基于地理位置查询

一、前言 随着互联网的热门&#xff0c;越来越多的传统行业将全部或者部分业务转移到互联网上&#xff0c;其中不乏一些和地理位置强相关的行业。基于地理位置的搜索功能&#xff0c;大大提升了人们的生活和工作效率。例如&#xff0c;外出旅行时&#xff0c;只需要用手机打开…

cuda卸载

去查看你的电脑显卡对应的cuda版本&#xff0c;不然还是一整个用不到gpu的情况嘿嘿. 啊啊啊啊打开控制面板看一下&#xff0c;驱动不要乱卸载&#xff1a; 这些东西不能全部卸载了哦&#xff0c;只能卸载含有“CUDA”的那几个&#xff08;其实其他的可能也没有用 但是不懂的哇 …

基于MIMO+16QAM系统的VBLAST译码算法matlab仿真

目录 1.算法运行效果图预览 2.算法运行软件版本 3.部分核心程序 4.算法理论概述 5.算法完整程序工程 1.算法运行效果图预览 2.算法运行软件版本 matlab2022a 3.部分核心程序 ........................................................................ for SNR_dBSNRS…

CentOS 7.9.2009 数据盘挂载

一、linux版本&#xff1a; lsb_release -a 二、操作步骤 2.1&#xff0c;查看磁盘挂载情况&#xff0c;确认sdb是需挂载的硬盘 ## 查看磁盘挂载情况&#xff0c;确认sdb是需挂载的硬盘 lsblk 2.2&#xff0c;对硬盘sdb进行分区 ## 对硬盘sdb进行分区 fdisk /dev/sdb# 命令…

Git(三).git 文件夹详解

目录 一、初始化新仓库二、.git 目录2.1 hooks 文件夹2.2 info 文件夹2.3 logs 文件夹2.4 objects 文件夹【重要】2.5 refs 文件夹【重要】2.6 COMMIT_EDITMSG2.7 config2.8 description2.9 FETCH_HEAD2.10 HEAD【重要】2.11 index【重要】2.12 ORIG_HEAD2.13 packed-refs 官网…

服务器中了360后缀勒索病毒怎么解决,勒索病毒解密,数据恢复

近期&#xff0c;网络上的各种病毒都比较猖獗&#xff0c;而其中较为明显的就是360后缀勒索病毒&#xff0c;从这个月开始云天数据恢复中心接到很多企业的求助&#xff0c;企业的服务器遭到了360后缀勒索病毒的攻击&#xff0c;通过给用户的服务器检测与加密病毒的分析&#xf…

thinkphp5使用phpmail发送qq邮件

目录 1、使用composer 工具安装&#xff0c;在tp5根目录下执行 2、封装发送邮件方法 3、控制器中调用 4、运行后结果 1、使用composer 工具安装&#xff0c;在tp5根目录下执行 composer require phpmailer/phpmailer 安装成功后显示下面目录 2、封装发送邮件方法 function…

【嵌入式项目应用】__cJSON在单片机的使用

目录 前言 一、JSON和cJson 二、cJSON是如何表示JSON数据的 三、如何封装完整的JSON数据 1. 先将串口打通&#xff0c;方便电脑查看log日志。 2. 增加cjson.c文件&#xff0c;已经在main.c中 3. 准备打包如下的JSON包 4. 代码部分&#xff0c;先将几个部分初始化指针 …