C#上位机与三菱PLC的通信08---开发自己的通讯库(A-1E版)
1、A-1E报文回顾
具体细节请看:
C#上位机与三菱PLC的通信03--MC协议之A-1E报文解析
C#上位机与三菱PLC的通信04--MC协议之A-1E报文测试
2、为何要开发自己的通讯库
前面使用了第3方的通讯库实现了与三菱PLC的通讯,实现了数据的读写,对于通讯库,我们只要引用并调用相关的方法即可实现目的,为什么别人可以封装通讯库dll文件,自己能不能做到?当然可以,但写一个通讯库需要非凡的技术,需要考虑的东西很多,比如扩展性,通用性,等等之类的。通过封装通讯库达到更高的层次,想想,别人使用自己的东西,说明自己牛XXXX啊,大师就是这样锻造出来的,接下来马上安排,鸿鹄之志从小事做起,振兴工业自动化,匹夫有责。
3、空谈误国,实干兴邦
1、创建vs项目
2、添加类库项目
3、创建目录及基础类 
AreaCode.cs代码
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;namespace Mitsubishi.Communication.MC.Mitsubishi.Base
{/// <summary>/// 存储区枚举/// </summary>public enum AreaCode{D = 0xA8,X = 0x9C,Y = 0x9D,M = 0x90,R = 0xAF,S = 0x98,TS = 0xC1,CN = 0xC5}
}
MelsecBase.cs代码
using System;
using System.Collections.Generic;
using System.Linq;
using System.Net.Sockets;
using System.Reflection;
using System.Runtime.InteropServices;
using System.Text;
using System.Threading;
using System.Threading.Tasks;namespace Mitsubishi.Communication.MC.Mitsubishi.Base
{/// <summary>/// mc协议基类/// </summary>public class MelsecBase{/// <summary>/// plc的ip地址/// </summary>public string _ip;/// <summary>/// plc的端口号/// </summary>public int _port;/// <summary>/// socket对象/// </summary>public Socket socket = null;/// <summary>/// 超时事件/// </summary>ManualResetEvent TimeoutObject = new ManualResetEvent(false);/// <summary>/// 连接状态/// </summary>bool connectState = false;/// <summary>/// 构造方法/// </summary>/// <param name="ip"></param>/// <param name="port"></param>public MelsecBase(string ip, short port){_ip = ip;_port = port;// 初始化一个通信对象socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);}/// <summary>/// 连接PLC/// </summary>/// <param name="timeout">超时时间</param>/// <returns></returns>public Result Connect(int timeout = 50){TimeoutObject.Reset();Result result = new Result();try{if (socket == null){socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);}int count = 0;while (count < timeout){if (!(!socket.Connected || (socket.Poll(200, SelectMode.SelectRead) && (socket.Available == 0)))){return result;}try{socket?.Close();socket.Dispose();socket = null;socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);//异步连接 socket.BeginConnect(_ip, _port, callback =>{connectState = false;var cbSocket = callback.AsyncState as Socket;if (cbSocket != null){connectState = cbSocket.Connected;if (cbSocket.Connected){cbSocket.EndConnect(callback);}}TimeoutObject.Set();}, socket);TimeoutObject.WaitOne(2000, false);if (!connectState){throw new Exception();}else{break;}}catch (SocketException ex){if (ex.ErrorCode == 10060){throw new Exception(ex.Message);}}catch (Exception ex){throw new Exception(ex.Message);}finally{count++;}}if (socket == null || !socket.Connected || ((socket.Poll(200, SelectMode.SelectRead) && (socket.Available == 0)))){throw new Exception("网络连接失败");}}catch (Exception ex){result.IsSuccessed = false;result.Message = ex.Message;}return result;}/// <summary>/// 构建开始地址/// </summary>/// <param name="areaCode">存储区</param>/// <param name="startAddr">开始地址</param>/// <returns></returns>/// <exception cref="Exception"></exception>public List<byte> StartToBytes(AreaCode areaCode, string startAddr){List<byte> startBytes = new List<byte>();if (areaCode == AreaCode.X || areaCode == AreaCode.Y){string str = startAddr.ToString().PadLeft(8, '0');for (int i = str.Length - 2; i >= 0; i -= 2){string v = str[i].ToString() + str[i + 1].ToString();startBytes.Add(Convert.ToByte(v, 16));}}else{int addr = 0;if (!int.TryParse(startAddr, out addr)){throw new Exception("软元件地址不支持!");}startBytes.Add((byte)(addr % 256));startBytes.Add((byte)(addr / 256 % 256));startBytes.Add((byte)(addr / 256 / 256 % 256));startBytes.Add((byte)(addr / 256 / 256 / 256 % 256));}return startBytes;}/// <summary>/// 发送报文/// </summary>/// <param name="reqBytes">字节集合</param>/// <param name="count">字节长度</param>/// <returns></returns>public virtual List<byte> Send(List<byte> reqBytes, int count){return null;}/// <summary>/// 数据解析/// </summary>/// <typeparam name="T">读取的数据类型</typeparam>/// <param name="datas">数据列表</param>/// <param name="typeLen">类型长度</param>/// <returns></returns>/// <exception cref="Exception"></exception>public List<T> AnalysisDatas<T>(List<byte> datas, int typeLen){List<T> resultDatas = new List<T>();if (typeof(T) == typeof(bool))//bool类型{for (int i = 0; i < datas.Count; i++){// 10 10 10 10 10string binaryStr = Convert.ToString(datas[i], 2).PadLeft(8, '0');dynamic state = binaryStr.Substring(0, 4) == "0001";resultDatas.Add(state);state = binaryStr.Substring(4) == "0001";resultDatas.Add(state);}}else//其他类型:ushort,short,float{for (int i = 0; i < datas.Count;){List<byte> valueByte = new List<byte>();for (int sit = 0; sit < typeLen * 2; sit++){valueByte.Add(datas[i++]);}Type tBitConverter = typeof(BitConverter);MethodInfo method = tBitConverter.GetMethods(BindingFlags.Public | BindingFlags.Static).FirstOrDefault(mi => mi.ReturnType == typeof(T)) as MethodInfo;if (method == null){throw new Exception("未找到匹配的数据类型转换方法");}resultDatas.Add((T)method?.Invoke(tBitConverter, new object[] { valueByte.ToArray(), 0 }));}}return resultDatas;}/// <summary>/// 计算长度/// </summary>/// <typeparam name="T">读取的数据类型</typeparam>/// <returns></returns>public int CalculatLength<T>(){int typeLen = 1;if (!typeof(T).Equals(typeof(bool))){typeLen = Marshal.SizeOf<T>() / 2;// 每一个数据需要多少个寄存器}return typeLen;}/// <summary>/// 获取数据的字节列表/// </summary>/// <typeparam name="T">数据类型</typeparam>/// <param name="values">数据列表</param>/// <returns></returns>public List<byte> GetDataBytes<T>(List<T> values){List<byte> datas = new List<byte>();int count = values.Count;if (typeof(T) == typeof(bool))//bool类型的数据{dynamic value = false;// 添加一个填充数据,保存一个完整字节if (values.Count % 2 > 0){values.Add(value);}for (int i = 0; i < values.Count; i += 2){byte valueByte = 0;if (bool.Parse(values[i].ToString())){valueByte |= 16;}if (bool.Parse(values[i + 1].ToString())){valueByte |= 1;}datas.Add(valueByte);}}else //其他类型:float,short,int16{for (int i = 0; i < values.Count; i++){dynamic value = values[i];datas.AddRange(BitConverter.GetBytes(value)); // MC不需要字节的颠倒}}return datas;}}
}
Result.cs代码
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;namespace Mitsubishi.Communication.MC.Mitsubishi.Base
{/// <summary>/// 结果类/// </summary>/// <typeparam name="T"></typeparam>public class Result<T>{/// <summary>/// 状态/// </summary>public bool IsSuccessed { get; set; }/// <summary>/// 对应的消息/// </summary>public string Message { get; set; }/// <summary>/// 数据列表/// </summary>public List<T> Datas { get; set; }public Result() : this(true, "OK") { }public Result(bool state, string msg) : this(state, msg, new List<T>()) { }public Result(bool state, string msg, List<T> datas){this.IsSuccessed = state; Message = msg; Datas = datas;}}public class Result : Result<bool> { }
}
确保上面的三个类文件编译成功,继续干
4、编写核心的通信类A1E.cs 
A1E.cs完整代码:
using Mitsubishi.Communication.MC.Mitsubishi.Base;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Net.Sockets;
using System.Text;
using System.Threading.Tasks;namespace Mitsubishi.Communication.MC.Mitsubishi
{/// <summary>/// A1E报文通讯库/// </summary>public class A1E : MelsecBase{/// <summary>/// 构造方法/// </summary>/// <param name="ip"></param>/// <param name="port"></param>public A1E(string ip, short port) : base(ip, port){}#region 读取数据/// <summary>/// 读取数据/// </summary>/// <typeparam name="T">读取的数据类型</typeparam>/// <param name="address">开始地址</param>/// <param name="count">读取长度</param>/// <returns></returns>public Result<T> Read<T>(string address, short count){AreaCode areaCode;string start;(areaCode, start) = this.AnalysisAddress(address);return Read<T>(areaCode, start, count);}/// <summary>/// 读取数据/// </summary>/// <typeparam name="T">读取的数据类型</typeparam>/// <param name="areaCode">存储区代码</param>/// <param name="startAddr">开始地址</param>/// <param name="count">读取长度</param>/// <returns></returns>public Result<T> Read<T>(AreaCode areaCode, string startAddr, short count){Result<T> result = new Result<T>();try{var connectState = this.Connect();if (!connectState.IsSuccessed){throw new Exception(connectState.Message);}//读取类型byte readCode = (byte)(typeof(T) == typeof(bool) ? 0x00 : 0x01);//起始地址List<byte> startBytes = this.StartToBytes(areaCode, startAddr);//存储区代码List<byte> areaBytes = this.AreaToBytes(areaCode);//读取长度int typeLen = this.CalculatLength<T>();//组装报文List<byte> command = new List<byte> {readCode,///读取类型0xFF,0x0A,0x00,//0xFF指PLC编号,0x0A,0x00指超时时间,超时时间是以250ms为单位 startBytes[0],startBytes[1],startBytes[2],startBytes[3], // 起始地址,占4个字节areaBytes[0],areaBytes[1], // 存储区,占2个字节(byte)(typeLen*count%256),// 读取长度,低位(byte)(typeLen*count/256%256) // 读取长度,高位};//计算响应报文的长度int respLen = typeLen * 2 * count;if (typeof(T) == typeof(bool)){respLen = (int)Math.Ceiling(typeLen * count * 1.0 / 2);}//发送报文List<byte> respBytes = this.Send(command, respLen);//数据解析result.Datas = this.AnalysisDatas<T>(respBytes, typeLen);}catch (Exception ex){result = new Result<T>(false, ex.Message);}return result;}#endregion #region 写入数据/// <summary>/// 写数据/// </summary>/// <typeparam name="T">写入的数据类型</typeparam>/// <param name="values">数据值列表</param>/// <param name="addr">开始地址</param>/// <returns></returns>public Result Write<T>(List<T> values, string addr){AreaCode areaCode; string start;(areaCode, start) = this.AnalysisAddress(addr);return this.Write<T>(values, areaCode, start);}/// <summary>/// 写数据/// </summary>/// <typeparam name="T">写入的数据类型</typeparam>/// <param name="values">数据值列表</param>/// <param name="areaCode">存储区代码</param>/// <param name="startAddr">开始地址</param>/// <returns></returns>public Result Write<T>(List<T> values, AreaCode areaCode, string startAddr){Result result = new Result();try{var connectState = this.Connect();if (!connectState.IsSuccessed){throw new Exception(connectState.Message);}// 写操作的类型 //0x00 批量位读取 //0x01 批量字读取 //0x02 批量位写入 //0x03 批量字写入 //0x04 随机位写入 //0x05 随机字写入byte writeCode = (byte)(typeof(T) == typeof(bool) ? 0x02 : 0x03);//开始地址List<byte> startBytes = this.StartToBytes(areaCode, startAddr);//存储区代码List<byte> areaBytes = this.AreaToBytes(areaCode);//构建数据的字节列表int count = values.Count;List<byte> datas = this.GetDataBytes<T>(values);//判断写入的长度,如果是float类型则长度要扩大2倍int length = count;//长度等于值的个数 if (typeof(T) == typeof(float)){length = length * 2;}//拼装报文List<byte> command = new List<byte> {writeCode,0xFF, 0x0A, 0x00,//0xFF指PLC编号,0x0A,0x00指超时时间,超时时间是以250ms为单位 startBytes[0], startBytes[1], startBytes[2], startBytes[3], // 起始地址areaBytes[0], areaBytes[1], // 存储区 //写入的长度的低位和高位 (byte)(length % 256),(byte)(length / 256 % 256),};command.AddRange(datas);//写入的具体数据//发送报文socket.Send(command.ToArray());// 判断写入的结果 byte[] respBytes = new byte[2];socket.Receive(respBytes);if (respBytes[0] != (writeCode |= 0x80)){throw new Exception("响应报文结构异常。" + respBytes[0].ToString());}if (respBytes[1] != 0x00){throw new Exception("响应异常。" + respBytes[1].ToString());}}catch (Exception ex){result.IsSuccessed = false;result.Message = ex.Message;}return result;}#endregion#region PLC启停,区别功能码 0x13,0x14public Result Run(){return PlcState(0x13);}public Result Stop(){return PlcState(0x14);}private Result PlcState(byte commandCode){Result result = new Result();try{var connectState = this.Connect();if (!connectState.IsSuccessed){throw new Exception(connectState.Message);}List<byte> commandBytes = new List<byte>{commandCode,0xFF,0x0A,0x00};socket.Send(commandBytes.ToArray());// 先判断响应状态byte[] respBytes = new byte[2];socket.Receive(respBytes);if (respBytes[0] != (commandCode |= 0x80)){throw new Exception("响应报文结构异常。" + respBytes[0].ToString());}if (respBytes[1] != 0x00){throw new Exception("响应异常。" + respBytes[1].ToString());}}catch (Exception ex){result.IsSuccessed = false;result.Message = ex.Message;}return result;}#endregion#region 内部方法/// <summary>/// 构建存储区代码/// </summary>/// <param name="areaCode">存储区代码</param>/// <returns></returns>private List<byte> AreaToBytes(AreaCode areaCode){List<byte> areaBytes = new List<byte>();string areaStr = areaCode.ToString();areaBytes.AddRange(Encoding.ASCII.GetBytes(areaStr));if (areaBytes.Count == 1){areaBytes.Add(0x20);}areaBytes.Reverse(); //字节反转return areaBytes;}/// <summary>/// 发送报文/// </summary>/// <param name="reqBytes">报文字节集合</param>/// <param name="len">报文字节长度</param>/// <returns></returns>/// <exception cref="Exception"></exception>public override List<byte> Send(List<byte> reqBytes, int len){socket.Send(reqBytes.ToArray()); //发送报文// 先判断响应状态byte[] respBytes = new byte[2];socket.Receive(respBytes);if (respBytes[0] != (reqBytes[0] |= 0x80)){throw new Exception("响应报文结构异常。" + respBytes[0].ToString());}if (respBytes[1] != 0x00){throw new Exception("响应异常。" + respBytes[1].ToString());}respBytes = new byte[len];socket.Receive(respBytes, 0, len, SocketFlags.None);return new List<byte>(respBytes);}/// <summary>/// 地址解析,输入的地址:X100 X1A0 M100 D100 TN10/// </summary>/// <param name="address">地址字符串</param>/// <returns>返回元组</returns>public Tuple<AreaCode, string> AnalysisAddress(string address){// 取两个字符string area = address.Substring(0, 2);if (!new string[] { "TN", "TS", "CS", "CN" }.Contains(area)){area = address.Substring(0, 1);}string start = address.Substring(area.Length);// 返回元组(一个对象,该对象包括编号和地址)var obj = new Tuple<AreaCode, string>((AreaCode)Enum.Parse(typeof(AreaCode), area), start);return obj;}#endregion }
}
确保项目编译成功,可以进行下一步
4、测试通讯库
1、添加项目引用 

2、启动MC服务器
3、利用通讯库读写数据
1 读取X区100开始的4个bool数据
2、读取D区100开始的5个float数据
3、读取M区200开始的2个short数据
4、写入M区200开始的2个short数据
5、写入D区200开始的5个float数据
4、完整代码
using Mitsubishi.Communication.MC.Mitsubishi;
using Mitsubishi.Communication.MC.Mitsubishi.Base;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;namespace Mitsubishi.Communication.Test
{internal class Program{static void Main(string[] args){MCLibTestA1E(); Console.WriteLine("执行完成!");Console.ReadKey();}/// <summary>/// 测试A-1E通讯库/// </summary>static void MCLibTestA1E(){A1E a1E = new A1E("192.168.1.7", 6000);#region 读数据Console.WriteLine("读取X区100开始的4个bool数据");var result1 = a1E.Read<bool>(AreaCode.X, "100", 4);if (result1.IsSuccessed){result1.Datas.ForEach(d => Console.WriteLine(d));}else{Console.WriteLine(result1.Message);}Console.WriteLine("读取D区200开始的5个float数据");var result2 = a1E.Read<float>(AreaCode.D, "200", 5);if (result2.IsSuccessed){result2.Datas.ForEach(d => Console.WriteLine(d));}else{Console.WriteLine(result2.Message);}Console.WriteLine("读取M区200开始的2个short数据");var result3 = a1E.Read<short>(AreaCode.M, "200", 2);if (result3.IsSuccessed){result3.Datas.ForEach(d => Console.WriteLine(d));}else{Console.WriteLine(result3.Message);}#endregion#region 写数据Console.WriteLine("写入M区200开始的2个short数据");var result4 = a1E.Write<short>(new List<short> { 61, 72 }, "M200");if (result4.IsSuccessed){Console.WriteLine(result4.Message);}Console.WriteLine("写入D区200开始的5个float数据");var result5 = a1E.Write<float>(new List<float> { 3.2f, -2.5f, 0, 35, -98 }, "D200");if (result5.IsSuccessed){Console.WriteLine(result5.Message);}#endregion}}
}
5、小结
原创真的不容易,走过路过不要错过,点赞关注收藏又圈粉,共同致富。
原创真的不容易,走过路过不要错过,点赞关注收藏又圈粉,共同致富。
原创真的不容易,走过路过不要错过,点赞关注收藏又圈粉,共同致富
相关文章:

C#上位机与三菱PLC的通信08---开发自己的通讯库(A-1E版)
1、A-1E报文回顾 具体细节请看: C#上位机与三菱PLC的通信03--MC协议之A-1E报文解析 C#上位机与三菱PLC的通信04--MC协议之A-1E报文测试 2、为何要开发自己的通讯库 前面使用了第3方的通讯库实现了与三菱PLC的通讯,实现了数据的读写,对于通…...

ABAQUS应用04——集中质量的添加方法
文章目录 0. 背景1. 集中质量的编辑2. 约束的设置3. 总结 0. 背景 混塔ABAQUS模型中,机头、法兰等集中质量的设置是模型建立过程中的一部分,需要研究集中质量的添加。 1. 集中质量的编辑 集中质量本身的编辑没什么难度,我已经用Python代码…...

[嵌入式系统-24]:RT-Thread -11- 内核组件编程接口 - 网络组件 - TCP/UDP Socket编程
目录 一、RT-Thread网络组件 1.1 概述 1.2 RT-Thread支持的网络协议栈 1.3 RT-Thread如何选择不同的网络协议栈 二、Socket编程 2.1 概述 2.2 UDP socket编程 2.3 TCP socket编程 2.4 TCP socket收发数据 一、RT-Thread网络组件 1.1 概述 RT-Thread 是一个开源的嵌入…...

【ansible】认识ansible,了解常用的模块
目录 一、ansible是什么? 二、ansible的特点? 三、ansible与其他运维工具的对比 四、ansible的环境部署 第一步:配置主机清单 第二步:完成密钥对免密登录 五、ansible基于命令行完成常用的模块学习 模块1:comma…...

【LeetCode】升级打怪之路 Day 01:二分法
今日题目: 704. 二分查找35. 搜索插入位置34. 在排序数组中查找元素的第一个和最后一个位置 目录 今日总结Problem 1: 二分法LeetCode 704. 二分查找 【easy】LeetCode 35. 搜索插入位置 ⭐⭐⭐⭐⭐LeetCode 34. 在排序数组中查找元素的第一个和最后一个位置 【medi…...

单片机stm32智能鱼缸
随着我国经济的快速发展而给人们带来了富足的生活,也有越来越多的人们开始养鱼,通过养各种鱼类来美化居住环境和缓解压力。但是在鱼类饲养过程中,常常由于鱼类对水质、水位及光照强度有着很高的要求,而人们也由于工作的方面而无法…...

面试经典150题——生命游戏
"Push yourself, because no one else is going to do it for you." - Unknown 1. 题目描述 2. 题目分析与解析 2.1 思路一——暴力求解 之所以先暴力求解,是因为我开始也没什么更好的思路,所以就先写一种解决方案,没准写着写…...

【C++】C++11下线程库
C11下线程库 1. thread类的简单介绍2.线程函数参数3.原子性操作库(atomic)4.mutex的种类5. RAII风格加锁解锁5.1Lock_guard5.2unique_lock 6.condition_variable 1. thread类的简单介绍 在C11之前,涉及到多线程问题,都是和平台相关的,比如wi…...

面试经典150题——矩阵置零
"Dream it. Wish it. Do it." - Unknown 1. 题目描述 2. 题目分析与解析 2.1 思路一——暴力求解 思路一很简单,就是尝试遍历矩阵的所有元素,如果发现值等于0,就把当前行与当前列的值分别置为0。同时我们需要注意,…...

多端开发围炉夜话
文章目录 一、多端开发 一、多端开发 uni-app 官网 UNI-APP中的UI框架:介绍常用的UI框架及其特点 uView UIVant WeappColor UIMint UI uniapp嵌入android原生开发的功能 uniapp使用安卓原生sdk uni-app中的uni.requireNativePlugin...

分治算法总结(Java)
目录 分治算法概述 快速排序 练习1:排序数组 练习2:数组中的第K个最大元素 练习3:最小k个数 归并排序 练习4:排序数组 练习5:交易逆序对的总数 练习6:计算右侧小于当前元素的个数 练习7࿱…...

【云原生系列之kubernetes】--Ingress使用
service的缺点: 不支持基于URL等机制对HTTP/HTTPS协议进行高级路由、超时、重试、基于流量的灰度等高级流量治理机制难以将多个service流量统一管理 1.1ingress的概念 ingress是k8s中的一个对象,作用是如何将请求转发到service的规则ingress controlle…...
练习:鼠标类设计之2_类和接口
前言 续鼠标类设计之1,前面解决了鼠标信号问题,这里解决显示问题 引入 鼠标伴随操作系统而生,考虑在屏幕上怎样显示 思路 1>鼠标显示是一个动态效果,所以需要一个“动态效果类”对象,添加进鼠标类的属性里。 在面…...

【程序员英语】【美语从头学】初级篇(入门)(笔记)Lesson 15 At the Department Store 在百货商店
《美语从头学初级入门篇》 注意:被 删除线 划掉的不一定不正确,只是不是标准答案。 文章目录 Lesson 15 At the Department Store 在百货商店会话A会话B笔记 Lesson 15 At the Department Store 在百货商店 会话A A: Can you help me, please? B: Sur…...

linux 安装、删除 JTAG驱动
安装 安装驱动需要sudo访问权限,所以得手动安装。 在petalinux安装目录下: 文件的路径。 cd tools/xsct/data/xicom/cable_drivers/lin64/install_script/install_drivers 然后执行文件 install_drivers。 sudo ./install_drivers安装成功。 删除 …...
CSS的伪类选择器:nth-child()
CSS的伪类选择器:nth-child() CSS的伪类选择器 :nth-child() 是一个非常强大的工具,它允许你根据元素在其父元素中的位置(序数)来选择特定的子元素。这个选择器可以应用于任何元素,并且可以与类型选择器、类选择器或ID选择器结合…...

python celery使用队列
在celery的配置方法中有个参数叫task_routes,是用来设置不同的任务 消费不同的队列(也就是路由)。 格式如下: { ‘task name’: { ‘queue’: ‘queue name’ }}直接上代码,简单明了,目录格式如下&#x…...

四非保研之旅
大家好,我是工藤学编程,虽有万分感概,但是话不多说,先直接进入正题,抒情环节最后再说,哈哈哈 写在开头 我的分享是来给大家涨信心的,网上的大佬们都太强了,大家拿我涨涨信心&#…...

基于Java+SpringBoot的旅游路线规划系统(源码+论文)
文章目录 目录 文章目录 前言 一、功能设计 二、功能实现 1.1 前端首页模块的实现 1.2 景点新闻 1.3 景点在线预订 1.4 酒店在线预订 1.5 管理员景点管理 1.6 管理员旅游线路管理 1.7 酒店信息管理 三、库表设计 前言 随着我国的经济的不断发展,现在的一些热门的景…...
AI与测试自动化:未来已来
AI与测试自动化注定融合。软件开发的速度和准确性要求已经远远超出了预期。测试自动化通过重复、详细和数据密集型测试来解决这个问题,确保敏捷和持续交付环境中的软件质量。AI的学习、适应和预测能力以完美的效率和准确性增强了测试自动化。复杂的算法现在充当质量…...

使用VSCode开发Django指南
使用VSCode开发Django指南 一、概述 Django 是一个高级 Python 框架,专为快速、安全和可扩展的 Web 开发而设计。Django 包含对 URL 路由、页面模板和数据处理的丰富支持。 本文将创建一个简单的 Django 应用,其中包含三个使用通用基本模板的页面。在此…...
golang循环变量捕获问题
在 Go 语言中,当在循环中启动协程(goroutine)时,如果在协程闭包中直接引用循环变量,可能会遇到一个常见的陷阱 - 循环变量捕获问题。让我详细解释一下: 问题背景 看这个代码片段: fo…...
IGP(Interior Gateway Protocol,内部网关协议)
IGP(Interior Gateway Protocol,内部网关协议) 是一种用于在一个自治系统(AS)内部传递路由信息的路由协议,主要用于在一个组织或机构的内部网络中决定数据包的最佳路径。与用于自治系统之间通信的 EGP&…...
大语言模型如何处理长文本?常用文本分割技术详解
为什么需要文本分割? 引言:为什么需要文本分割?一、基础文本分割方法1. 按段落分割(Paragraph Splitting)2. 按句子分割(Sentence Splitting)二、高级文本分割策略3. 重叠分割(Sliding Window)4. 递归分割(Recursive Splitting)三、生产级工具推荐5. 使用LangChain的…...

1.3 VSCode安装与环境配置
进入网址Visual Studio Code - Code Editing. Redefined下载.deb文件,然后打开终端,进入下载文件夹,键入命令 sudo dpkg -i code_1.100.3-1748872405_amd64.deb 在终端键入命令code即启动vscode 需要安装插件列表 1.Chinese简化 2.ros …...

华为OD机试-食堂供餐-二分法
import java.util.Arrays; import java.util.Scanner;public class DemoTest3 {public static void main(String[] args) {Scanner in new Scanner(System.in);// 注意 hasNext 和 hasNextLine 的区别while (in.hasNextLine()) { // 注意 while 处理多个 caseint a in.nextIn…...

Psychopy音频的使用
Psychopy音频的使用 本文主要解决以下问题: 指定音频引擎与设备;播放音频文件 本文所使用的环境: Python3.10 numpy2.2.6 psychopy2025.1.1 psychtoolbox3.0.19.14 一、音频配置 Psychopy文档链接为Sound - for audio playback — Psy…...

令牌桶 滑动窗口->限流 分布式信号量->限并发的原理 lua脚本分析介绍
文章目录 前言限流限制并发的实际理解限流令牌桶代码实现结果分析令牌桶lua的模拟实现原理总结: 滑动窗口代码实现结果分析lua脚本原理解析 限并发分布式信号量代码实现结果分析lua脚本实现原理 双注解去实现限流 并发结果分析: 实际业务去理解体会统一注…...
【HTML-16】深入理解HTML中的块元素与行内元素
HTML元素根据其显示特性可以分为两大类:块元素(Block-level Elements)和行内元素(Inline Elements)。理解这两者的区别对于构建良好的网页布局至关重要。本文将全面解析这两种元素的特性、区别以及实际应用场景。 1. 块元素(Block-level Elements) 1.1 基本特性 …...

12.找到字符串中所有字母异位词
🧠 题目解析 题目描述: 给定两个字符串 s 和 p,找出 s 中所有 p 的字母异位词的起始索引。 返回的答案以数组形式表示。 字母异位词定义: 若两个字符串包含的字符种类和出现次数完全相同,顺序无所谓,则互为…...