js实现一个简单的扫雷
目录
- 先看下最终的效果:
- 首先来分析一个扫雷游戏具有哪些功能
- 分析完成后我们就开始一步步的实现
- 1. 相关html和css
- 2. 我们使用类来完成相应功能
- 3. 之后我们则是要定义一个地图
- 4. 对地图进行渲染
- 5. 对开始按钮添加点击事件
- 6. 现在我们可以实现鼠标左击扫雷的功能
- 7. 给单元格添加右键点击事件
- 8. 定义一个生成指定范围随机数的函数
- 9. 限制执行次数
- 10. 在布雷时需要进行校验
- 11. 布雷的校验规则
- 12. 限制布雷功能执行次数
- 13. 统计周围地雷的数量
- 14. 再次将数据渲染到html中
- 15. 设置点击样式并进行递归
- 16. 给reset按钮添加事件
- 17. 定义游戏结束功能
- 18. 定义游戏胜利功能
- 以下是完整js代码
- 结语(碎碎念
先看下最终的效果:
首先来分析一个扫雷游戏具有哪些功能
- 需要一个地图来表示扫雷
- 游戏会有不同的难度
- 第一下点击是不会触发雷的
- 每个非雷格子都会显示与它相近的单元格中雷的个数
- 点击一个非雷格子会自动将上下左右的非雷格子点开
- 右击单元格可以进行标注,再次右击可以取消
- 当点击到地雷单元格时就会游戏结束
- 当所有非雷单元格都点开时则游戏胜利
- …
分析完成后我们就开始一步步的实现
1. 相关html和css
<!DOCTYPE html>
<html lang="en"><head><meta charset="UTF-8"><meta http-equiv="X-UA-Compatible" content="IE=edge"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Document</title><style>* {box-sizing: border-box;margin: 0;padding: 0;}article {text-align: center;}main {margin: auto;flex-wrap: wrap;display: flex;width: 500px;height: 500px;}main>div {margin: 5px;font-size: 0px;background-color: antiquewhite;border: 1px solid blue;}/*不同难度下单元格不同的样式*/.width5 {width: 90px;height: 90px;}.width10 {width: 40px;height: 40px;}.width20 {margin: 2.5px;width: 20px;height: 20px;}/*select鼠标左键点击*/.select {font-size: 12px;background-color: aqua;}/*select2鼠标右键点击*/.select2 {font-size: 0px;background-color: blue;}</style>
</head><body><!-- 整个扫雷内容区 --><article id="mine"><button id="start">开始</button><select id="select"><!--value是不同难度下的地图大小,data-level是不同难度下一个非雷单元格周围一圈最多能有多少雷的规定--><option value="5" data-level="4">简单</option><option value="10" data-level="6">普通</option><option value="20" data-level="9">困难</option></select><button id="reset">reset</button><main></main></article><script src="./javascript/4.js"></script><script>var main = document.querySelector('#mine');//实例化一个扫雷游戏的类var mines = new MineSweeping(main);</script>
</body></html>
2. 我们使用类来完成相应功能
class MineSweeping {constructor(root) {this.btn = root.querySelector('#start');this.reset = root.querySelector('#reset')this.main = root.querySelector('main');this.select = root.querySelector('#select');}
}
3. 之后我们则是要定义一个地图
这里我们使用二维数组来实现,其中有一点需要注意,在扫雷当中对于边角,边缘,内部的布雷方式是不同,即在内部单元格周围一圈最多可以有8个雷,但在边缘或者边角的话最多就只有5个甚至是3个雷,为了以后在布雷时更方便的对此单元格进行校验(判断周围一圈的雷的数量是否合理,在边角,在边缘,在内部三种情况都需要进行单独判断)我们在定义地图时需要额外再扩大一层,比如我们界面当中的地图是5 * 5的,但我们在定义时的二维数组是6 * 6的,这么做的话我们巧妙地将原本地图边缘区域的单元格变成了内部单元格,进行校验的时候也会更加方便了
this.row = [];
this.col = [];
//传入数组的长度,因为我们定义的地图是一个边长相等的正方形,所以只传一个值就可以了
init = (num) => {for (let i = 0; i < num; i++) {for (let j = 0; j < num; j++) {this.col.push(0);}this.row.push(this.col);this.col = [];}
}
4. 对地图进行渲染
即把二维数组在html中写出来
addMap = (map) => {for (let i = 1; i < map[0].length - 1; i++) {for (let j = 1; j < map[0].length - 1; j++) {let div = document.createElement('div');div.className = 'width' + (this.num - 2);//给div添加自定义属性,表示该div在二维数组中对应的位置div.setAttribute('data-row', i - 1);div.setAttribute('data-col', j - 1)this.main.appendChild(div);}}}
5. 对开始按钮添加点击事件
因为是当用户点击了开始按钮之后我们才进行的初始化地图和渲染地图,所以需要对开始按钮加一个点击事件
constructor(root) {//num为地雷数量,level为不同难度下单元格周围最大的地雷数量this.num;this.level;}
this.btn.addEventListener('click', this.startInit)startInit = () => {this.num = parseInt(this.select.value) + 2;let index = this.select.selectedIndex;this.level = parseInt(this.select[index].getAttribute('data-level'));this.init(this.num)this.addMap(this.row)}
6. 现在我们可以实现鼠标左击扫雷的功能
因为如果给每个单元格都添加点击事件的话性能开销就会比较大,我们这里就使用事件委托的形式进行,事件委托即把原本需要添加给子节点的事件委托到父节点中,核心原理就是DOM元素中的事件冒泡
this.main.addEventListener('click', this.gameStart)gameStart = (e) => {//这其中的涉及的函数下文会讲到this.flag && this.initMine(this.row, (this.num - 2) * (this.num - 2) / 2.5, e.target);this.flag && this.countAroundMines(this.row);this.flag && this.addMapMine(this.row);this.flag = false;this.selectAround(this.row, parseInt(e.target.getAttribute('data-row')) + 1, parseInt(e.target.getAttribute('data-col')) + 1);this.gameWin();}
7. 给单元格添加右键点击事件
constructor(root) {this.main.addEventListener('mousedown', (e) => {document.oncontextmenu = function (e) {e.preventDefault();};if (e.button == 2) {e.target.classList.toggle('select2')}})}
8. 定义一个生成指定范围随机数的函数
方便下面布雷功能的展开
//获取一个大于等于min并且小于等于max的随机值getRandomIntInclusive = (min, max) => {min = Math.ceil(min);max = Math.floor(max);return Math.floor(Math.random() * (max - min + 1)) + min;}
9. 限制执行次数
在扫雷游戏中,用户的第一次点击是不会触发雷的,所以我们布雷的功能需要在用户第一次点击之后运行
//传入了三个参数,map为地图,num为地雷的个数,div为当前点击的单元格
initMine = (map, num, div) => {num = parseInt(num);let x;let y;while (num > 0) {x = this.getRandomIntInclusive(1, this.num - 2);y = this.getRandomIntInclusive(1, this.num - 2);while (map[x][y] == -1 || (div.getAttribute('data-row') == x - 1 && div.getAttribute('data-col') == y - 1) || this.isNumberMines(map, x, y)) {x = this.getRandomIntInclusive(1, this.num - 2);y = this.getRandomIntInclusive(1, this.num - 2);}//可以布雷时就将二维数组中对应下标的值赋值为-1map[x][y] = -1;num--;}}
10. 在布雷时需要进行校验
我们首先判断当前单元格是否已经有地雷,然后还需判断这个单元格是否为当前点击的单元格,最后还需判断单元格周围的地雷数量是否合理
while (map[x][y] == -1 || (div.getAttribute('data-row') == x - 1 && div.getAttribute('data-col') == y - 1) || this.isNumberMines(map, x, y)) {x = this.getRandomIntInclusive(1, this.num - 2);y = this.getRandomIntInclusive(1, this.num - 2);}
11. 布雷的校验规则
判断此单元格周围的雷数是否合理,不合理就不能布雷
//传入三个参数,map为地图,x,y为当前单元格坐标isNumberMines = (map, x, y) => {//count即周围地雷的数量let count = 0;for (let i = x - 1; i < x + 1; i++) {for (let j = y - 1; j < y + 1; j++) {if (map[i][j] == -1) {count++;}}}//不同难度等级有不同的限制if (this.level == 9) {if ((x == 1 && y == 1) || (x == 0 && y == this.num - 2) || (x == this.num - 2 && y == 0) || (x == this.num - 2 && y == this.num - 2)) {if (count >= 4) {return true;}} else if (x == 1 || y == 0 || x == this.num - 2 || y == this.num - 2) {if (count >= 6) {return true;}} else {if (count >= 9) {return true;}}}if (this.level == 6) {if ((x == 1 && y == 1) || (x == 0 && y == this.num - 2) || (x == this.num - 2 && y == 0) || (x == this.num - 2 && y == this.num - 2)) {if (count >= 4) {return true;}} else if (x == 1 || y == 0 || x == this.num - 2 || y == this.num - 2) {if (count >= 6) {return true;}} else {if (count >= 7) {return true;}}}if (this.level == 4) {if ((x == 1 && y == 1) || (x == 0 && y == this.num - 2) || (x == this.num - 2 && y == 0) || (x == this.num - 2 && y == this.num - 2)) {if (count >= 4) {return true;}} else {if (count >= 5) {return true;}}}}
12. 限制布雷功能执行次数
因为布雷功能只会在第一次点击之后执行一次,以后所有点击都将不会再次执行布雷,所以我们需要一个标志来限制布雷功能的执行次数
class MineSweeping {constructor(root) {this.flag = true;}gameStart = (e) => {this.flag && this.initMine(this.row, (this.num - 2) * (this.num - 2) / 2.5, e.target);this.flag = false;}
}
13. 统计周围地雷的数量
布完雷之后我们还需要对没有雷的单元格进行统计周围地雷数量,并把统计结果赋值到对应的数组元素当中,同样这个功能也会只执行一次
gameStart = (e) => {this.flag && this.countAroundMines(this.row);this.flag = false;}
//传入一个参数,map为地图
countAroundMines = (map) => {let count;for (let i = 1; i < map[0].length - 1; i++) {for (let j = 1; j < map[0].length - 1; j++) {count = 0;if (map[i][j] != -1) {for (let ii = i - 1; ii <= i + 1; ii++) {for (let jj = j - 1; jj <= j + 1; jj++) {if (map[ii][jj] == -1) {count++;}}}map[i][j] = count;}}}}
14. 再次将数据渲染到html中
我们在统计完之后就需要把这些结果,包括地雷,地雷数量等数据渲染到html中,这个功能同样只会执行一次
gameStart = (e) => {this.flag && this.addMapMine(this.row);this.flag = false;}
//传入一个参数,map为地图
addMapMine = (map) => {//获取到当前html中所有的单元格let div = this.main.querySelectorAll('div');let t = 0;for (let i = 1; i < map[0].length - 1; i++) {for (let j = 1; j < map[0].length - 1; j++) {if (map[i][j] == -1) {div[t++].innerHTML = '雷';} else {div[t++].innerHTML = map[i][j];}}}}
15. 设置点击样式并进行递归
以上做完之后,我们还需对当前单元格设置点击样式,并实现如果点击了非雷单元格则要同时点开其上下左右四个格子,直到遇到地雷为止
selectAround = (map, x, y) => {let div = this.main.querySelectorAll('div');//对此刻传入的单元格进行判断,如果不在规定的范围内(x与y的范围,以及单元格本身是否被点击了)则终止函数执行(因为有递归)if (x < 1 || y < 1 || x > this.num - 2 || y > this.num - 2 || div[(x - 1) * (this.num - 2) + y - 1].classList.contains('select')) {return;//如果此时单元格为地雷} else if (map[x][y] == -1) {//判断此单元格是否被右键标记,有则移除标记,添加左键点击样式if (div[(x - 1) * (this.num - 2) + y - 1].classList.contains('select2')) {div[(x - 1) * (this.num - 2) + y - 1].classList.remove('select2');}div[(x - 1) * (this.num - 2) + y - 1].classList.add('select');//触发游戏结束this.gameOver();} else {if (div[(x - 1) * (this.num - 2) + y - 1].classList.contains('select2')) {div[(x - 1) * (this.num - 2) + y - 1].classList.remove('select2');}div[(x - 1) * (this.num - 2) + y - 1].classList.add('select');//开始对该单元格上下左右进行递归if (map[x][y + 1] != -1) {this.selectAround(map, x, y + 1);}if (map[x][y - 1] != -1) {this.selectAround(map, x, y - 1);}if (map[x - 1][y] != -1) {this.selectAround(map, x - 1, y);}if (map[x + 1][y] != -1) {this.selectAround(map, x + 1, y);}}}
16. 给reset按钮添加事件
constructor(root) {this.reset.addEventListener('click', this.gameReset)}gameReset = () => {this.flag = true;this.main.innerHTML = '';this.row = [];this.col = [];}
17. 定义游戏结束功能
gameOver = () => {setTimeout(() => {alert('over');this.gameReset();}, 100)}
18. 定义游戏胜利功能
this.main.addEventListener('click', this.gameStart)gameStart = (e) => {//每点击一次就判断一次是否胜利this.gameWin();}
gameWin = () => {let flag = true;let div = this.main.querySelectorAll('div');for (let i = 0; i < (this.num - 2) * (this.num - 2); i++) {if (div[i].innerHTML != '雷') {if (!div[i].classList.contains('select')) {flag = false;break;}}}if (flag) {setTimeout(() => {alert('win');this.gameReset();}, 100)}}
至此,整个扫雷游戏就基本写完了
以下是完整js代码
class MineSweeping {constructor(root) {this.btn = root.querySelector('#start');this.reset = root.querySelector('#reset')this.main = root.querySelector('main');this.select = root.querySelector('#select');this.num;this.level;this.flag = true;this.row = [];this.col = [];this.btn.addEventListener('click', this.startInit)this.main.addEventListener('click', this.gameStart)this.main.addEventListener('mousedown', (e) => {document.oncontextmenu = function (e) {e.preventDefault();};if (e.button == 2) {e.target.classList.toggle('select2')}})this.reset.addEventListener('click', this.gameReset)}//因为用户第一次点击时单元格不为雷,所以布雷放在触发点击事件之后并且保证布雷只会执行一次gameStart = (e) => {this.flag && this.initMine(this.row, (this.num - 2) * (this.num - 2) / 2.5, e.target);this.flag && this.countAroundMines(this.row);this.flag && this.addMapMine(this.row);this.flag = false;this.selectAround(this.row, parseInt(e.target.getAttribute('data-row')) + 1, parseInt(e.target.getAttribute('data-col')) + 1);this.gameWin();}//开始初始化地图startInit = () => {this.num = parseInt(this.select.value) + 2;let index = this.select.selectedIndex;this.level = parseInt(this.select[index].getAttribute('data-level'));this.init(this.num)this.addMap(this.row)}//定义地图init = (num) => {for (let i = 0; i < num; i++) {for (let j = 0; j < num; j++) {this.col.push(0);}this.row.push(this.col);this.col = [];}}//开始在二维数组中布雷initMine = (map, num, div) => {num = parseInt(num);let x;let y;while (num > 0) {x = this.getRandomIntInclusive(1, this.num - 2);y = this.getRandomIntInclusive(1, this.num - 2);while (map[x][y] == -1 || (div.getAttribute('data-row') == x - 1 && div.getAttribute('data-col') == y - 1) || this.isNumberMines(map, x, y)) {x = this.getRandomIntInclusive(1, this.num - 2);y = this.getRandomIntInclusive(1, this.num - 2);}map[x][y] = -1;num--;}}//布雷的校验规则,判断此单元格能否布雷isNumberMines = (map, x, y) => {let count = 0;for (let i = x - 1; i < x + 1; i++) {for (let j = y - 1; j < y + 1; j++) {if (map[i][j] == -1) {count++;}}}if (this.level == 9) {if ((x == 1 && y == 1) || (x == 0 && y == this.num - 2) || (x == this.num - 2 && y == 0) || (x == this.num - 2 && y == this.num - 2)) {if (count >= 4) {return true;}} else if (x == 1 || y == 0 || x == this.num - 2 || y == this.num - 2) {if (count >= 6) {return true;}} else {if (count >= 9) {return true;}}}if (this.level == 6) {if ((x == 1 && y == 1) || (x == 0 && y == this.num - 2) || (x == this.num - 2 && y == 0) || (x == this.num - 2 && y == this.num - 2)) {if (count >= 4) {return true;}} else if (x == 1 || y == 0 || x == this.num - 2 || y == this.num - 2) {if (count >= 6) {return true;}} else {if (count >= 7) {return true;}}}if (this.level == 4) {if ((x == 1 && y == 1) || (x == 0 && y == this.num - 2) || (x == this.num - 2 && y == 0) || (x == this.num - 2 && y == this.num - 2)) {if (count >= 4) {return true;}} else {if (count >= 5) {return true;}}}}//获取一个大于等于min并且小于等于max的随机值getRandomIntInclusive = (min, max) => {min = Math.ceil(min);max = Math.floor(max);return Math.floor(Math.random() * (max - min + 1)) + min;}//渲染地图addMap = (map) => {//0-7for (let i = 1; i < map[0].length - 1; i++) {for (let j = 1; j < map[0].length - 1; j++) {let div = document.createElement('div');div.className = 'width' + (this.num - 2);div.setAttribute('data-row', i - 1);div.setAttribute('data-col', j - 1)this.main.appendChild(div);}}}//开始渲染地雷及非雷单元格周围地雷数量addMapMine = (map) => {let div = this.main.querySelectorAll('div');let t = 0;for (let i = 1; i < map[0].length - 1; i++) {for (let j = 1; j < map[0].length - 1; j++) {if (map[i][j] == -1) {div[t++].innerHTML = '雷';} else {div[t++].innerHTML = map[i][j];}}}}//在二维数组中统计非雷单元格周围的地雷数量并赋值到对应元素中countAroundMines = (map) => {let count;for (let i = 1; i < map[0].length - 1; i++) {for (let j = 1; j < map[0].length - 1; j++) {count = 0;if (map[i][j] != -1) {for (let ii = i - 1; ii <= i + 1; ii++) {for (let jj = j - 1; jj <= j + 1; jj++) {if (map[ii][jj] == -1) {count++;}}}map[i][j] = count;}}}}//给点击的单元格设置样式,如果用户点击了非雷单元格则程序会自动帮用户点击此单元格上下左右四个单元格中同样非雷的单元格,直到遇到地雷selectAround = (map, x, y) => {let div = this.main.querySelectorAll('div');if (x < 1 || y < 1 || x > this.num - 2 || y > this.num - 2 || div[(x - 1) * (this.num - 2) + y - 1].classList.contains('select')) {return;} else if (map[x][y] == -1) {if (div[(x - 1) * (this.num - 2) + y - 1].classList.contains('select2')) {div[(x - 1) * (this.num - 2) + y - 1].classList.remove('select2');}div[(x - 1) * (this.num - 2) + y - 1].classList.add('select');//如果点击了地雷则触发游戏结束this.gameOver();} else {if (div[(x - 1) * (this.num - 2) + y - 1].classList.contains('select2')) {div[(x - 1) * (this.num - 2) + y - 1].classList.remove('select2');}div[(x - 1) * (this.num - 2) + y - 1].classList.add('select');if (map[x][y + 1] != -1) {this.selectAround(map, x, y + 1);}if (map[x][y - 1] != -1) {this.selectAround(map, x, y - 1);}if (map[x - 1][y] != -1) {this.selectAround(map, x - 1, y);}if (map[x + 1][y] != -1) {this.selectAround(map, x + 1, y);}}}//重置游戏gameReset = () => {this.flag = true;this.main.innerHTML = '';this.row = [];this.col = [];}//游戏结束gameOver = () => {setTimeout(() => {alert('over');this.gameReset();}, 100)}//游戏胜利gameWin = () => {let flag = true;let div = this.main.querySelectorAll('div');for (let i = 0; i < (this.num - 2) * (this.num - 2); i++) {if (div[i].innerHTML != '雷') {if (!div[i].classList.contains('select')) {flag = false;break;}}}if (flag) {setTimeout(() => {alert('win');this.gameReset();}, 100)}}}
结语(碎碎念
第一次写这种教程也不知道怎么样写才能带来更好的阅读体验,如果觉得写的不行的话非常抱歉
这是我第一次进行知识输出,以前则是一直只有输入,感觉写完这篇文章之后自己确实有点感悟,但具体在哪还没感觉出来
其实代码写的还有很多问题的,就比如那个布雷功能,只能对当前已经布置下的地雷进行校验,未来要是在另一个位置布雷则会影响之前已经布下的地雷,具体来说就是简单难度最多一圈只有4个雷,但实际运行下来最多却有5个。另外样式也不美观,自己需要学习的地方还有很多。
如果这篇文章有帮到你的话我会很高兴,
最后,祝我们变得更强:)。
相关文章:

js实现一个简单的扫雷
目录先看下最终的效果:首先来分析一个扫雷游戏具有哪些功能分析完成后我们就开始一步步的实现1. 相关html和css2. 我们使用类来完成相应功能3. 之后我们则是要定义一个地图4. 对地图进行渲染5. 对开始按钮添加点击事件6. 现在我们可以实现鼠标左击扫雷的功能7. 给单…...

禁用非必需插件,让 IDEA 飞起
文章首发于个人博客,欢迎访问关注:https://www.lin2j.tech IDEA 为我们提供了众多的插件,但是这些插件并不都是必须的。如果电脑的性能不够强,反而会带来一些不必要的资源消耗。 因此这里整理了一些不常用的插件,可以…...

解决win10任何程序打开链接仍然为老旧IE的顽固问题[修改默认浏览器]
文章目录一、问题与修改原因1、着手修改吧2、弯路上探索3、发现祸根二、后话文章原出处: https://blog.csdn.net/haigear/article/details/129344503一、问题与修改原因 我们发现,很多程序默认的网页打开浏览器都是IE,这个很是郁闷ÿ…...

计算机网络体系结构——“计算机网络”
各位CSDN的uu们你们好呀,今天小雅兰来学习一个全新的知识点,就是计算机网络啦,下面,开始虚心学习。 计算机网络的概念 计算机网络的功能 计算机网络的组成 计算机网络的分类 标准化工作 计算机网络的性能 计算机网络的概念 …...

基于微信小程序的校园二手交易平台小程序
文末联系获取源码 开发语言:Java 框架:ssm JDK版本:JDK1.8 服务器:tomcat7 数据库:mysql 5.7/8.0 数据库工具:Navicat11 开发软件:eclipse/myeclipse/idea Maven包:Maven3.3.9 浏览器…...

Linux lvm管理讲解及命令
♥️作者:小刘在C站 ♥️个人主页:小刘主页 ♥️每天分享云计算网络运维课堂笔记,努力不一定有收获,但一定会有收获加油!一起努力,共赴美好人生! ♥️夕阳下,是最美的绽放࿰…...

GPT-4,终于来了!
就在昨天凌晨,OpenAI发布了多模态预训练大模型GPT-4。 这不昨天一觉醒来,GPT-4都快刷屏了,不管是在朋友圈还是网络上都看到了很多信息和文章。 GPT是Generative Pre-trained Transformer的缩写,也即生成型预训练变换模型的意思。…...

面部表情识别3:Android实现表情识别(含源码,可实时检测)
面部表情识别3:Android实现表情识别(含源码,可实时检测) 目录 面部表情识别3:Android实现表情识别(含源码,可实时检测) 1.面部表情识别方法 2.人脸检测方法 3.面部表情识别模型训练 4.面部表情识别模型Android部署 &#x…...

【IT女神勋章挑战赛名单公布】:看看谁获奖啦!
致敬女性开发者,为那些IT女神而战!「IT女神勋章挑战赛」已落下帷幕,博主们的比拼结果也已新鲜出炉,快来看看此次征文中,有哪些精彩文章不容错过!活动官网:https://activity.csdn.net/creatActiv…...

ThreadPool线程池源码解析
ThreadPool线程池源码解析 文章目录前言一、基本使用二、执行流程三、源码分析ThreadPoolExecutor 中重要属性ThreadPoolExecutor 内部类Workerexecute()方法addWorker(command, true)方法runWorker(worker )方法getTask()方法shutdown和shutdownNow四、…...

Python中 5个非常有用的单行代码
Python中 5个非常有用的单行代码1. 什么是单行代码?2. 了解和使用单行代码的好处3. 5个单行代码示例参考作为开发人员,你想提高生产力吗?掌握这 5 个 Python 一行代码,你就能写出简洁的代码。 这篇博客将介绍5个非常有用的初学者可…...

蓝牙模块各种工作模式
摘要:本文主要归纳总结蓝牙模块的不同工作模式,通过蓝牙模块不同的工作模式了解其扮演不同角色时工作的一个基本原理,为更深入的研究蓝牙模块底层的工作机制和技术方案进行铺垫。 1、主设备工作模式 主设备是能够搜索别人并主动建立连接的一…...

修剪灌木[蓝桥杯2022初赛]
题目描述 爱丽丝要完成一项修剪灌木的工作。 有 N 棵灌木整齐的从左到右排成一排。 爱丽丝在每天傍晚会修剪一棵灌木,让灌木的高度变为 0 厘米。 爱丽丝修剪灌木的顺序是从最左侧的灌木开始,每天向右修剪一棵灌木。 当修剪了最右侧的灌木后,…...

Python的知识点运用-1(日期转换)
问:如何将 星期一, 三月 13, 2023转换成2023-03-13看到这个问题,你的第一反应是什么???反正我是懵逼的。不过后面一想,时间模块可以。在这个问题后面,群友又问了一个问题,如何在本地…...

原理图制图规范详细说明
1、1 原理图必须使用公司统一原理图库 在原理图设计中,必须采用公司统一原理图库,以保证设计的一致性和打包后封装、料单等结果的一致性。不使用公司统一原理图库造成的连接、封装错误个人承担责任。 注意使cds.lib中的路径指向库服务器eda-svr1的路径…...

【Unity小知识】Editor编写常用方法汇总
汇总一些Unity Editor开发的常用方法和实现方式,会持续更新。 添加自定义菜单栏方法 using UnityEngine; using UnityEditor;public class EditorTools : EditorWindow {[MenuItem("EditorTools/自定义的编辑器方法")]public static void CustomEditroFu…...

【数据仓库-4】-- 提取、转换、装载(ETL)
1.数据抽取 1.1 逻辑抽取 1.1.1 全量数据抽取 一般发生在我们初始化时,需要一次性将源库(业务系统)的所有数据抽取到数据仓库的ODS层。 比如将一个全表导出数据文件或者查询源表所有数据的SQL语句, 都是全量抽取的例子。 select * from user.person; 1.1.2 增量数据抽取 对…...

【DBC专题】-12-不同类型报文(应用/诊断/网关/测量标定)在DBC中配置,以及在Autosar各模块间的信号数据流向
点击返回「Autosar从入门到精通-实战篇」总目录 案例背景(共18页精讲):该篇博文将告诉您: 1)Autosar中,不同类型报文(App应用,UDS/OBD诊断,NM网络管理报文,XCP测量标定)的信号数据流向; 2)CAN …...

【Linux】进程的基础概念 进程的相关操作 进程的状态
进程一、进程的基本知识1、基本概念2、进程的描述 —— PCB3、task_ struct内容分类二、进程的相关操作1、在Linux下查看进程2、通过系统调用在代码中获取进程标示符3、如何创建子进程4、关于fork()的一些深度理解三、进程的状态Linux中的进程的状态四、僵尸进程与孤儿进程僵尸…...

【小猫爪】AUTOSAR学习笔记06-Communication Stack之ComM模块
【小猫爪】AUTOSAR学习笔记06-Communication Stack之ComM模块前言1 ComM简介2 ComM功能介绍2.1 PNC 状态管理2.2 Channel状态管理2.3 通信禁止功能2.4 不同类型的NM2.5 User、PNC 与 Channel 的映射2.6 状态保存END前言 因为一个偶然的机会让我接触到了AUTOSAR,所以…...

BP插件开发(JAVA)
本文会包括BP插件开发流程及打包,API,javaswing(UI)部分的内容。阅读完本文后,读者将初步具有开发BP插件的能力。1 开始开发我们使用IDEA作为开发工具(使用其他IDE也绰绰有余)。引入依赖包&…...

【Zookeeper】介绍与配置
目录 概述 工作机制 特点 数据结构 应用场景 统一配置管理 统一集群管理 编辑 服务器动态上下线 软负载均衡 下载 启动 启动客户端 配置参数 集群配置 配置服务器编号 配置zoo.cfg 分发zoo.cfg配置文件 选举机制 第一次启动 非第一次启动 集群启动停止脚本…...

chrome快捷键
Ctrl T:打开新标签页。Ctrl W:关闭当前标签页。Ctrl Shift T:重新打开最近关闭的标签页。Ctrl Tab:在打开的标签页之间切换。Ctrl Shift Tab:在打开的标签页之间反向切换。Ctrl N:打开新窗口。Ctrl…...

手搓string类
手搓string类 文章目录手搓string类string的成员一.构造,拷贝构造,赋值重载和析构1.构造函数2.拷贝构造3.swap问题4.赋值重载5.析构函数二.简单接口1.c_str2.size(有效字符长度)3.capacity(有效字符容量)4.operator[]5.迭代器和范…...

小白学Pytorch系列--Torch API (7)
小白学Pytorch系列–Torch API (7) Comparison Ops allclose 此函数检查输入和其他是否满足条件: >>> torch.allclose(torch.tensor([10000., 1e-07]), torch.tensor([10000.1, 1e-08])) False >>> torch.allclose(torch.tensor([10000., 1e-…...

函数(上)——“Python”
各位CSDN的uu们你们好呀,今天小雅兰的内容是Python的函数呀,下面,就让我们进入函数的世界吧 首先可以选择性地看一下小雅兰很久之前写的C语言函数章节的知识: 函数——“C”_认真学习的小雅兰.的博客-CSDN博客 函数递归…...

ChatGPT说:如何利用ChatGPT变现?躺着赚钱不是梦。
您好,我是码农飞哥,感谢您阅读本文,欢迎一键三连哦。 💪🏻 1. Python基础专栏,基础知识一网打尽,9.9元买不了吃亏,买不了上当。 Python从入门到精通 😁 2. 毕业设计专栏&…...

4.网络爬虫—Post请求(实战演示)
网络爬虫—Post请求实战演示POST请求GET请求POST请求和GET请求的区别获取二进制数据爬[百度官网](https://www.baidu.com/)logo实战发送post请求百度翻译实战使用session发送请求模拟登录17k小说网常见问题前言: 📝📝此专栏文章是专门针对…...

【视频文稿】车载Android应用开发与分析 - 开发系统应用
本期视频地址:https://www.bilibili.com/video/BV1NY411z7TK/ 前言 Hello,大家好,我是林栩。 开发车载应用,其实主要都是在Android系统中编写各种系统应用,所以上期视频先介绍了Android系统源码的下载和编译流程&…...

Scala流程控制
目录 单分支 双分支 多分支 for 循环控制 循环守卫 循环步长 循环嵌套 循环返回值 While 和 do..While 循环控制 While循环控制 do..While 循环控制 循环中断 单分支 if (条件表达式) {执行代码块 }var age StdIn.readShort()if (age < 18){println("童年&quo…...