Autojs 利用OpenCV识别棋子之天天象棋你马没了
本例子通过代码像你介绍利用OpenCV实现霍尔找圆的方法定位棋子位置
通过autojs脚本实现自动点击棋子 开源地址
https://github.com/Liberations/TtxqYourHorseIsGone/blob/master/main.js
AutoXJs
https://github.com/kkevsekk1/AutoX/releases
auto()
//安卓版本高于Android 9
if (device.sdkInt > 28) {//等待截屏权限申请并同意threads.start(function () {packageName('com.android.systemui').text('立即开始').waitFor()text('立即开始').click()});
}
//申请截屏权限
if (!requestScreenCapture()) {toast("请求截图失败")exit()
}
http.__okhttp__.setTimeout(40 * 1000)//低版本需要加这个
// 创建空的棋盘
// 创建空的棋盘
function createEmptyChessboard() {var temp = [];// 创建一个大小为 10x9 的二维数组,并用空字符串初始化for (var i = 0; i < 10; i++) {temp[i] = Array(9).fill("O");}return temp;
}// 打印二维数组
function printChessboard(chessboard) {for (var i = 0; i < chessboard.length; i++) {var row = chessboard[i].join(" ");log(row);}
}//复制一份新的数组
function copyArray(array) {var newArray = [];for (var i = 0; i < array.length; i++) {var innerArray = array[i];var newInnerArray = innerArray.slice(); // 复制内部数组newArray.push(newInnerArray);}return newArray;
}//将[row,col] 转成字母
function coordinatesToAlgebraic(coordinates) {var row = coordinates[0]var col = coordinates[1]if (isRed) {row = 9 - row;col = col + 'a'.charCodeAt(0);return String.fromCharCode(col) + row.toString();} else {col = 8 - col + 'a'.charCodeAt(0);return String.fromCharCode(col) + row.toString();}}//将字母转成 [row,col]
function algebraicToCoordinates(algebraic) {var col = algebraic.charCodeAt(0) - 'a'.charCodeAt(0);var row = 9 - parseInt(algebraic.substring(1));if (isRed) {//b9 c7 (0,1) (2,2)return [row, col];} else {//b9 c7 (9,7) (7,6)row = 9 - rowcol = 8 - colreturn [row, col];}
}//二维数组旋转180度
function rotateArray180(array) {var rotated = [];for (var i = array.length - 1; i >= 0; i--) {var row = array[i];var rotatedRow = row.slice().reverse();rotated.push(rotatedRow);}return rotated;
}//大小写翻转
function transformString(str) {let transformedString = "";for (let i = 0; i < str.length; i++) {let char = str.charAt(i);if (char === char.toUpperCase()) {transformedString += char.toLowerCase();} else {transformedString += char.toUpperCase();}}return transformedString;
}//释放图片资源
function releaseSouce() {// if (img != null) {// img.recycle()// img = null// }if (board_img != null) {board_img.recycle()img = null}
}//通过row col去点击相应的坐标
function clickChess(row, col) {var clickStartX = (2 * col + 1) * chess_piece_radius + boardLeft;var clickStartY = (2 * row + 1) * chess_piece_radius + boardTop;//log("点击位置clickStartX " + clickStartX + " clickStartY " + clickStartY)click(clickStartX, clickStartY);
}//所有的操作都用这个控制
//初始的象棋布局
var defaultChessBoard = [['r', 'n', 'b', 'a', 'k', 'a', 'b', 'n', 'r'],['O', 'O', 'O', 'O', 'O', 'O', 'O', 'O', 'O'],['O', 'c', 'O', 'O', 'O', 'O', 'O', 'c', 'O'],['p', 'O', 'p', 'O', 'p', 'O', 'p', 'O', 'p'],['O', 'O', 'O', 'O', 'O', 'O', 'O', 'O', 'O'],['O', 'O', 'O', 'O', 'O', 'O', 'O', 'O', 'O'],['P', 'O', 'P', 'O', 'P', 'O', 'P', 'O', 'P'],['O', 'C', 'O', 'O', 'O', 'O', 'O', 'C', 'O'],['O', 'O', 'O', 'O', 'O', 'O', 'O', 'O', 'O'],['R', 'N', 'B', 'A', 'K', 'A', 'B', 'N', 'R']
]//初始化圆的位置
var defaultCircleBoard = [['B', 'B', 'B', 'B', 'B', 'B', 'B', 'B', 'B'],['O', 'O', 'O', 'O', 'O', 'O', 'O', 'O', 'O'],['O', 'B', 'O', 'O', 'O', 'O', 'O', 'B', 'O'],['B', 'O', 'B', 'O', 'B', 'O', 'B', 'O', 'B'],['O', 'O', 'O', 'O', 'O', 'O', 'O', 'O', 'O'],['O', 'O', 'O', 'O', 'O', 'O', 'O', 'O', 'O'],['R', 'O', 'R', 'O', 'R', 'O', 'R', 'O', 'R'],['O', 'R', 'O', 'O', 'O', 'O', 'O', 'R', 'O'],['O', 'O', 'O', 'O', 'O', 'O', 'O', 'O', 'O'],['R', 'R', 'R', 'R', 'R', 'R', 'R', 'R', 'R']
]var defaultBlackCircleBoard = [['R', 'R', 'R', 'R', 'R', 'R', 'R', 'R', 'R'],['O', 'O', 'O', 'O', 'O', 'O', 'O', 'O', 'O'],['O', 'R', 'O', 'O', 'O', 'O', 'O', 'R', 'O'],['R', 'O', 'R', 'O', 'R', 'O', 'R', 'O', 'R'],['O', 'O', 'O', 'O', 'O', 'O', 'O', 'O', 'O'],['O', 'O', 'O', 'O', 'O', 'O', 'O', 'O', 'O'],['B', 'O', 'B', 'O', 'B', 'O', 'B', 'O', 'B'],['O', 'B', 'O', 'O', 'O', 'O', 'O', 'B', 'O'],['O', 'O', 'O', 'O', 'O', 'O', 'O', 'O', 'O'],['B', 'B', 'B', 'B', 'B', 'B', 'B', 'B', 'B']
]var isRed = null
var isSart = true
var isMyStep = false //标志是否我方下棋
var stepCount = 0 //累计步数
var fen = ""
var img = null //每次的截图
var board_img = null //裁剪出来的棋盘
var storage = storages.create("chessStorage");
var boardWidth = 1010 //棋 盘宽度
var boardHeight = 1140 //棋盘高度
var boardTop = 626 //棋盘上边距
var boardLeft = 34 //棋盘左边距
var speekOn = false //棋盘左边距
var chess_piece_radius = 0 //棋子半径
var chess_piece_width = 0 //棋子宽度
var moves = [] //当前移动的fen码
var chessBoard = null
var circleBoard = []
var intervalId = 0
var 开始 = images.read("开始.png")
var 关闭 = images.read("关闭.png")
var 确定 = images.read("确定.png")
var 叉叉 = images.read("叉叉.jpg")
var 对战助手 = images.read("对战助手.jpg")
var 确定退出 = images.read("确定退出.jpg")//删除截图
var imgDir = "img";
files.removeDir(imgDir)
files.create(imgDir + "/")function main() {initStart()isSart = truelog('开始新的对局')while (isSart) {try {sleep(2000)findChess()} catch (e) {log("发生异常" + e)}//isSart = false}}function initStart() {chessBoard = copyArray(defaultChessBoard)circleBoard = copyArray(defaultCircleBoard)isRed = nullisSart = trueisMyStep = false //标志是否我方下棋stepCount = 0 //累计步数fen = ""moves = []lastMove = ""if (intervalId != 0) {clearInterval(intervalId)intervalId = 0}
}function findChess() {img = images.captureScreen()if (!img) {toastLog('截图失败')return}var pos4 = images.findImage(img, 对战助手)if (pos4) {speek("发现对战助手按钮")click(pos4.x, pos4.y)//重新初始化initStart()return}//log("开始findChess")var pos = images.findImage(img, 关闭)if (pos) {speek("发现关闭对局结束")click(pos.x, pos.y)sleep(1000)click(device.width / 2, device.height / 2)//重新初始化initStart()return}var pos1 = images.findImage(img, 开始)if (pos1) {speek("发现开始按钮")click(pos1.x, pos1.y)//重新初始化initStart()return}var pos2 = images.findImage(img, 确定)if (pos2) {speek("发现确定按钮")click(pos2.x, pos2.y)//重新初始化initStart()return}var pos3 = images.findImage(img, 叉叉)if (pos3) {speek("发现叉叉按钮")click(pos3.x, pos3.y)//重新初始化initStart()return}chess_piece_width = boardWidth / 9chess_piece_radius = chess_piece_width / 2board_img = images.clip(img, boardLeft, boardTop, boardWidth, boardHeight)if (isRed == null) {var col = 4var row = 0var centerX = (2 * col + 1) * chess_piece_radius;var centerY = (2 * row + 1) * chess_piece_radius;var left = centerX - chess_piece_radiusvar top = centerY - chess_piece_radiuslet cutImg = images.clip(board_img, left, top, chess_piece_radius * 2, chess_piece_radius * 2)let containRed = images.findColor(cutImg, "#953c28", {threshold: 4})let containBlack = images.findColor(cutImg, "#3a3938", {threshold: 4})if (containRed) {circleBoard = nullcircleBoard = copyArray(defaultBlackCircleBoard)isRed = false} else if (containBlack) {circleBoard = nullcircleBoard = copyArray(defaultCircleBoard)isRed = true} else {log("没找到帅啥也没有")}speek("我方为" + (isRed ? "红方" : "黑方"))}var tempchessboard = createEmptyChessboard()var count = 0;// 找圆min_radius = chess_piece_width / 4max_radius = chess_piece_width / 3// 灰度化图片let gray = images.grayscale(board_img);// 找圆let arr = images.findCircles(gray, {dp: 1,minDst: 30,param1: 50,param2: 30,minRadius: parseInt(min_radius),maxRadius: parseInt(max_radius),});gray.recycle()for (let i = 0; i < arr.length; i++) {let circle = arr[i];let x = circle.x; // 圆心 x 坐标let y = circle.y; // 圆心 y 坐标let radius = circle.radius; // 圆的半径// 裁剪图像let row = parseInt(y / chess_piece_width)let col = parseInt(x / chess_piece_width)var left = x - radius;var top = y - radius;let cutImg = images.clip(board_img, left, top, radius * 2, radius * 2)let containRed = images.findColor(cutImg, "#953c28", {threshold: 4})let containBlack = images.findColor(cutImg, "#3a3938", {threshold: 4})if (containRed) {count++tempchessboard[row][col] = "R"} else if (containBlack) {count++tempchessboard[row][col] = "B"} else {tempchessboard[row][col] = "O"}cutImg.recycle()}//log('总共找到圆' + count)if (stepCount == 0 && count != 32) {log(arr.length + '数量不够' + count)initStart()printChessboard(tempchessboard)releaseSouce()return}if (stepCount < 999) {var pao1start = tempchessboard[7][1]var pao1Mid = tempchessboard[2][1]var pao1end = tempchessboard[0][1]var pao2start = tempchessboard[7][7]var pao2Mid = tempchessboard[2][7]var pao2end = tempchessboard[0][7]if (pao1start != "O" && pao1end != "O" && pao1Mid != "O") {speek("发现炮1")clickChess(7, 1)sleep(50)clickChess(0, 1)speek("你马没了")sleep(2000)stepCount++return}if (pao2start != "O" && pao2end != "O" && pao2Mid != "O") {speek("发现炮2")clickChess(7, 7)sleep(50)clickChess(0, 7)speek("你马没了")sleep(2000)stepCount++return}return}
}function speek(str) {if (true) {TTS.speak(str, TextToSpeech.QUEUE_ADD, null);//自定义语音合成内容} else {log(str)}}//调用系统语音合成
importPackage(android.speech.tts);
importClass(java.util.Locale);
if (TTS != undefined) { TTS.stop(); TTS.shutdown(); }
var TTS = new TextToSpeech(context, function (status) {if (status != TextToSpeech.SUCCESS) {toast("初始化TTS失败");}var r = TTS.setLanguage(Locale.CHINA);if (r < 0) {toast("不支持该语言: " + r);exit();}log("TTS初始化成功");
});
sleep(500);
speek("开始辅助")
// 创建悬浮窗
var window = floaty.window(<vertical><text text="在棋力评测或者积分场快速按钮界面点击开始运行,需要辅助功能模拟点击" textColor="red" w="auto" ></text><button id="toggleButton" text="展开" w="auto" h="auto" /><vertical id="menu" visibility="gone"><button id="startButton" text="开始" w="auto" h="auto" /><button id="stopButton" text="结束" w="auto" h="auto" /></vertical></vertical>
);
var isOpen = false
// 设置悬浮窗位置
window.setPosition(0, device.height / 20);log("我的引擎" + engines.myEngine())
engines.all().forEach(item => {if (item.id != engines.myEngine().id) {log("停止引擎" + item)item.forceStop()}})
function hideMenu() {isOpen = falsewindow.menu.setVisibility(8);window.toggleButton.setText("展开");
}
function expandOrHideMenu() {isOpen = !isOpenif (isOpen) {window.menu.setVisibility(0);window.toggleButton.setText("收起");} else {window.menu.setVisibility(8);window.toggleButton.setText("展开");}
}
// 切换按钮点击事件
window.toggleButton.click(() => {expandOrHideMenu()});// 开始按钮点击事件
window.startButton.click(() => {hideMenu()threads.start(function () {isSart = falsespeek("开始")main()});});// 结束按钮点击事件
window.stopButton.click(() => {isSart = falsethreads.shutDownAll()toastLog("已停止");hideMenu()
});//保持脚本运行
setInterval(() => { }, 1000);
主要函数利用autojs提供的 findCircles函数
images.findCircles(gray, options)
gray {Image} 灰度图片
options {Object} 选项包括:
region {Array} 找圆区域。是一个两个或四个元素的数组。(region[0], region[1])表示找圆区域的左上角;region[2]*region[3]表示找圆区域的宽高。如果只有region只有两个元素,则找圆区域为(region[0], region[1])到图片右下角。如果不指定region选项,则找圆区域为整张图片。
dp {number} dp是累加面与原始图像相比的分辨率的反比参数,dp=2时累计面分辨率是元素图像的一半,宽高都缩减为原来的一半,dp=1时,两者相同。默认为1。
minDst {number} minDist定义了两个圆心之间的最小距离。默认为图片高度的八分之一。
param1 {number} param1是Canny边缘检测的高阈值,低阈值被自动置为高阈值的一半。默认为100,范围为0-255。
param2 {number} param2是累加平面对是否是圆的判定阈值,默认为100。
minRadius {number} 定义了检测到的圆的半径的最小值,默认为0。
maxRadius {number} 定义了检测到的圆的半径的最大值,0为不限制最大值,默认为0。
return {Array}
在图片中寻找圆(做霍夫圆变换)。找到时返回找到的所有圆{x,y,radius}的数组,找不到时返回null。
一个寻找圆的例子:
// 请求截图
requestScreenCapture();
// 截图
let img = captureScreen();
// 灰度化图片
let gray = images.grayscale(img);
// 找圆
let arr = findCircles(gray, {dp: 1,minDst: 80,param1: 100,param2: 100,minRadius: 50,maxRadius: 80,
});
// 回收图片
gray.recycle();
具体看代码 可以在这个基础上实现手机版自动连线功能!
相关文章:

Autojs 利用OpenCV识别棋子之天天象棋你马没了
本例子通过代码像你介绍利用OpenCV实现霍尔找圆的方法定位棋子位置 通过autojs脚本实现自动点击棋子 开源地址 https://github.com/Liberations/TtxqYourHorseIsGone/blob/master/main.js AutoXJs https://github.com/kkevsekk1/AutoX/releasesauto() //安卓版本高于Android 9…...

好数组——尺取法
好数组 给定一个长度为 n 的数组 a,计算数组 a 中所有子数组中好数组的数目。 好数组定义如下: 对于数组 al ,al1, ⋯ ,ar ,若数组中所有数的质因数种类数不超过 k,则称为好数组。 Input 输入的第一行包含两个正整数 n,k (1≤…...

【Linux】Ubuntu升级nodejs版本
在下载nvm对nodejs版本进行管理时,由于网络因素一直下载失败,于是采用了新的方法对nodejs版本进行升级。 首先我们先查询一下现存的nodejs版本号,发现是12 我们下载一个名为n的软件包,n 是一个非常方便的 Node.js 版本管理工具&am…...

二维码智慧门牌管理系统升级解决方案:一级属性 二级属性
文章目录 前言一、什么是智慧门牌管理系统?二、一级属性 vs. 二级属性三、升级中的实践意义 前言 在本文中,我们将深入探讨二维码智慧门牌管理系统的升级解决方案,特别聚焦于一级属性和二级属性的关键概念。我们将详细解释这些概念ÿ…...

input改造文件上传,el-table的改造,点击上传,拖拽上传,多选上传
第一个input标签效果 第二个input标签的效果 el-table的改造效果 <template><div class"outerBox"><div class"analyze" v-if"status"><div class"unFile"><div class"mainBox"><img clas…...

申请实用新型专利需要的时间
1、咨询 确定发明创造的内容是否属于可以申请专利的内容;对此咨询,建议多咨询几家专利代理机构后对比确定正确的结论。因为当前很多的专利代理机构的资讯接待员是的工资都是提成制的,为了业务量,有时对咨询会有不恰当的回复。确定…...

Redis 主从复制和哨兵监控,实现Redis高可用配置
文章目录 一、概述二、主从复制模拟说明三、准备配置文件四、启动Redis实例五、主从复制配置5.1 命令方式启用和取消主从复制5.2 配置文件方式启用和取消主从复制5.3 测试主从复制5.4 有其主从复制的其他参数配置 六、Sentinel 配置6.1 Sentinel 的作用6.2 Sentinel 监控说明6.…...

虹科直播 | CDS网络与数据安全专题技术直播重磅来袭,11.2起与您精彩相约
文章来源:虹科网络安全 阅读原文:https://mp.weixin.qq.com/s/T-CgU28hmYy4YV5SV9QGhg 虹科数据加密解决方案 虹科终端安全防护方案 虹科是在各细分专业技术领域内的资源整合及技术服务落地供应商,虹科网络安全事业部的宗旨是:让…...

nginx加权轮询,upstream,Keepalive,负载均衡实现案例
1. nginx 加权轮询, weight是权重配置。 #配置上游服务器 upstream tomcats {server 192.168.1.173:8080 weight=1;server 192.168.1.174:8080 weight=2;server 192.168.1.175:8080 weight=5; } server{liste...

java代理示例
以上代码通过Apache HttpComponents库,使用Java其中,proxy_host参数为代理服务器的主机名,proxy_port参数为服务器的端口号。程序首先创建了一个HttpGet对象,然后创建了一个HttpClient对象。接着,设置了HttpGet对象的U…...

51单片机汽车胎压大气气压测量仪仿真设计_数码管显示(代码+仿真+设计报告+讲解)
51单片机汽车胎压大气气压测量仪仿真设计_数码管显示 (代码仿真设计报告讲解) 仿真原版本:proteus 7.8 程序编译器:keil 4/keil 5 编程语言:C语言 设计编号:S0018 目录 51单片机汽车胎压大气气压测量仪仿真设计_数码管显示功…...

mac idea 解决0% classes 0% lines covered不显示,非快捷键办法
问题如下 网上说了一堆快捷键,冲突了用不了,页面按下面这样点就可以了点击no coverage就行了...

Fabric.js 复制粘贴元素
本文简介 点赞 关注 收藏 学会了 当你要复制一个 fabric 的元素时,你考虑到的是什么?是深拷贝当前选中对象再添加到画布中? 其实,fabric.js 提供了一个克隆方法,在 fabric.js 官网的案例里也有这个demo:…...

rstudio server 服务器卡死了怎么办
欢迎关注weixin:生信小博士 #rstudio 卡死了怎么办 cd ~/.local/share/ ls rm -fr rstudio.old mv ~/.rstudio ~/.rstudio.oldcd ~/.config/ rm -fr .rstudio.old mv ~/.config/rstudio/ ~/.config/rstudio.oldps -ef|grep t040413 |grep rsession |awk {print $2}| xarg…...

贪心算法学习——加油站
目录 一,题目 二,题目接口 三,解题思路及其代码 一,题目 在一条环路上有 n 个加油站,其中第 i 个加油站有汽油 gas[i] 升。 你有一辆油箱容量无限的的汽车,从第 i 个加油站开往第 i1 个加油站需要消耗汽油…...

Android 字符串工具类
Java基础大佬勿喷,小白专属! public class StringUtils {// 判断字符串是否为空public static boolean isEmpty(String str) {return str null || str.trim().isEmpty();}// 判断字符串是否为非空public static boolean isNotEmpty(String str) {retur…...

有了InheritableThreadLocal为啥还需要TransmittableThreadLocal?
有了InheritableThreadLocal为啥还需要TransmittableThreadLocal? 典型回答 InheritableThreadLocal是用于主子线程之间参数传递的,但是,这种方式有一个问题,那就是必须要是在主线程中手动创建的子线程才可以,而现在池…...

结构伪类选择器
伪类选择器:用来描述一个元素的特殊状态!比如第一个元素、某个元素的子元素、鼠标点击的元素 1 first-child/last-child /*ul的第一个子元素*/ ul li:first-child{ background: #0f35ad; } /*ul的最后一个子元素*/ ul li:last-child{ background: #0f3…...

java-- 静态数组
1.静态初始化数组 定义数组的时候直接给数组赋值。 2.静态初始化数组的格式: 注意: 1."数据类型[] 数组名"也可以写成"数据类型 数组名[]"。 2.什么类型的数组只能存放什么类型的数据 3.数组在计算机中的基本原理 当计算机遇到…...

世界经济论坛:ChatGPT等生成式AI,对全球23%岗位产生巨大影响
世界经济论坛与全球最大上市咨询公司之一埃森哲合作,联合发布了《未来工作:大语言模型与就业》白皮书。 世界经济论坛表示,随着ChatGPT、Midjourney、Github Copilot等生成式AI的飞速发展,对全球经济和劳动市场产生巨大影响。未来…...

myTracks for Mac:GPS轨迹记录器的强大与便捷
你是否曾经在户外活动或旅行中,希望能够记录下你的移动轨迹?或者在工作中,需要跟踪你的行程路线?myTracks for Mac 是一款强大的 GPS 轨迹记录器,它可以帮助你实现这些愿望。 myTracks 是一款专门为 Mac 设计的 GPS 轨…...

Macos视频增强修复工具:Topaz Video AI for mac
Topaz Video AI是一款使用人工智能技术对视频进行增强和修复的软件。它可以自动降噪、去除锐化、减少压缩失真、提高清晰度等等。Topaz Video AI可以处理各种类型的视频,包括低分辨率视频、老旧影片、手机录制的视频等等。 使用Topaz Video AI非常简单,…...

如何在IDEA中配置指定JDK版本?轻松解决!!!
有时候我们在导入项目,如果手动在IDEA中指定jdk版本,往往启动项目会报错误。 因此当我们新引入项目启动不了时可以检查一下自己IDEA中的jdk版本是否正确。 下面以配置jdk版本为11显示步骤: 1、配置 Project Structure 1.1、通过快捷键&qu…...

思维导图软件 ConceptDraw MINDMAP mac中文特色介绍
ConceptDraw MINDMAP mac是一款思维导图绘制软件,它可以帮助用户快速创建各种类型的思维导图,如组织结构图、流程图、概念图和UML图等。该软件具有直观的界面和简单易用的操作方式,使得用户能够轻松地创建复杂的思维导图。此外,它…...

PDF编辑工具Acrobat Pro DC 2023中文
Acrobat Pro DC 2023是一款全面、高效的PDF编辑和管理软件。它提供了丰富的PDF编辑功能,如创建、编辑、合并、分割、压缩、旋转、裁剪等,让用户可以轻松处理各种PDF文档。同时,该软件还具有智能的PDF处理技术,可以自动识别和修复P…...

如何开通 Medium会员
1 开通 WildCard 卡 首先你需要一张可以支付的外国卡 选择开通 WildCard 卡,优点: 1 无需上传身份证件,支付宝认证即可 2 可以使用国内手机号注册 3 可以使用支付宝、微信充值 开通地址: https://bewildcard.com/card 一步一步…...

CDN是如何一步步壮大到现在这样的
当我们浏览网页、观看在线视频或下载文件时,CDN(内容分发网络)已经成为网络世界中不可或缺的一部分。本文将探讨CDN的发展历程,其工作原理,以及它如何利用不同地区来提供更快速、可靠的内容交付服务。 CDN的发展历程 过…...

【Java】电子病历编辑器源码(云端SaaS服务)
电子病历编辑器极具灵活性,它既可嵌入到医院HIS系统中,作为内置编辑工具供多个模块使用,也可以独立拿出来,与第三方业务厂商展开合作,为他们提供病历书写功能,充分发挥编辑器的功能。 电子病历基于云端SaaS…...

解决netty作为web,post请求体过大导致413 Request Entity Too Largew问题
问题 项目中使用netty作为web服务,postman请求体内容超出5mb请求netty时,返回413 Request Entity Too Large 解决 查询了一下资料:https://netty.io/4.0/api/io/netty/handler/codec/http/HttpObjectAggregator.html ChannelPipeline p .…...

【Linux】rpm和yum的使用
不知道是不是有和我一样的宝子们,在rpm上卡了老久老久,但其实搞通了,理解了原理之后,不难的,所以不管你现在遇到的困难是什么,都不要放弃,一定要坚持,加油。 一、rpm 1.rpm rpm的…...