C#上位机模板程序,使用的是台达AS228主机PLC,功能齐全,自动运行页面、切换页面、手动调试、参数设置页面都有。
最近在工业自动化项目里摸爬滚打,发现台达AS228这PLC真是经得起折腾的主儿。刚好手头有个自用的C#上位机模板,拿出来和大家唠唠怎么跟这铁疙瘩配合干活。这个模板不整花里胡哨的MVVM,直接WinForm硬刚,适合快速出活的场景。
通信模块是重头戏,先上核心代码:
// DeltaAS228通信协议实现 public class DeltaProtocol { private SerialPort _comPort; private byte[] _readBuffer = new byte[256]; // 关键寄存器地址映射 const int RUN_STATUS_ADDR = 0x1000; const int MANUAL_CTRL_ADDR = 0x2000; public bool Connect(string portName) { try { _comPort = new SerialPort(portName, 9600, Parity.Even, 8, StopBits.One); _comPort.DataReceived += DataReceivedHandler; _comPort.Open(); return true; } catch { return false; } } private void DataReceivedHandler(object sender, SerialDataReceivedEventArgs e) { _comPort.Read(_readBuffer, 0, _comPort.BytesToRead); // 协议解析逻辑... } public bool ReadDRegister(int address, out int value) { // 构造读取命令帧 byte[] cmd = new byte[] { 0x02, 0x30, (byte)(address >> 8), (byte)address }; _comPort.Write(cmd, 0, 4); // 等待响应... } }这段代码实现了基础通信框架,注意校验位用Even这个细节是台达协议的特殊要求。寄存器地址映射部分建议单独做成配置文件,方便现场调试时快速调整。遇到过最坑的是响应超时处理,建议加个重试机制,现场电磁干扰大时能救命。
手动调试页面最考验实时性,这里用了个骚操作——把按钮事件直接绑到IO操作:
// 手动控制气缸 private void btnCylinder_Click(object sender, EventArgs e) { // 0x2000是手动模式寄存器地址 if (!delta.WriteRegister(DeltaProtocol.MANUAL_CTRL_ADDR, 1)) { MessageBox.Show("切手动模式失败!"); return; } // 0x55是气缸启动指令 Task.Run(() => { delta.WriteCoil(0x3000, true); // 置位输出 Thread.Sleep(200); // 保持200ms delta.WriteCoil(0x3000, false); // 复位 }); }这里为什么要用Task.Run?现场测试发现直接操作SerialPort.Write会导致界面卡顿,特别在老旧工控机上更明显。异步执行后,按钮响应立马顺滑了。注意WriteCoil后要延时复位,很多新手会漏这个,导致PLC接收不到完整指令。
参数设置页面用了XML持久化,但加了个实用功能——参数版本控制:
// 参数保存逻辑 public void SaveParameters() { var paramSet = new XElement("Params", new XAttribute("Version", DateTime.Now.ToString("yyyyMMddHHmm")), new XElement("Speed", nudSpeed.Value), new XElement("Timeout", nudTimeout.Value) ); // 自动保留最近5个版本 var history = Directory.GetFiles("Params/") .OrderByDescending(f => f) .Skip(4); foreach (var file in history) File.Delete(file); paramSet.Save($"Params/{DateTime.Now:yyyyMMddHHmm}.xml"); }这个版本控制功能救了项目组好几次——设备参数被误改后能快速回退。用LINQ做文件筛选比传统方式简洁很多,Skip(4)配合OrderByDescending刚好保留最新5个版本。
页面切换用了个土法炼钢但好用的办法——控件可见性控制:
// 页面切换核心逻辑 private Dictionary<PageType, UserControl> _pages = new Dictionary<PageType, UserControl>(); private void SwitchPage(PageType target) { foreach (var page in _pages.Values) { page.Visible = false; } _pages[target].Dock = DockStyle.Fill; _pages[target].Visible = true; // 强制重绘避免残留 this.Refresh(); }为什么不直接用TabControl?现场反馈说物理按钮切换页面时TabControl的标签头太碍事。这种全屏切换模式虽然土,但操作工用着顺手。注意最后那个Refresh(),解决过某些显卡驱动下的画面残留问题。
调试时发现个坑:直接读写PLC寄存器容易产生竞争条件。后来加了操作队列:
// 串行化PLC操作 private BlockingCollection<Action> _plcQueue = new BlockingCollection<Action>(); // 初始化时启动消费者线程 Task.Factory.StartNew(() => { foreach (var action in _plcQueue.GetConsumingEnumerable()) { try { action(); } catch { /* 记录日志 */ } } }, TaskCreationOptions.LongRunning); // 提交操作请求 public void SafeWriteRegister(int addr, int value) { _plcQueue.Add(() => { WriteRegister(addr, value); }); }这个设计保证同一时间只有一个读写操作在进行,实测通信稳定性提升明显。BlockingCollection比锁更省心,特别是处理突发的大量操作时。
最后给模板加了个实用功能——运行日志的环形缓冲区:
// 固定大小的内存日志 public class RingLogger { private const int MAX_ENTRIES = 1000; private ConcurrentQueue<string> _logQueue = new ConcurrentQueue<string>(); public void Log(string message) { _logQueue.Enqueue($"{DateTime.Now:HH:mm:ss} {message}"); if (_logQueue.Count > MAX_ENTRIES) { _logQueue.TryDequeue(out _); } } public string GetRecentLogs() { return string.Join(Environment.NewLine, _logQueue.Reverse()); } }用ConcurrentQueue实现线程安全,Reverse()让最新日志显示在最前面。现场调试时通过这个能快速定位问题,比查文本日志高效得多。
这个模板在多个项目里打磨过,虽然界面不够炫,但胜在稳定可靠。下次可以聊聊怎么在这个基础上加远程监控——用WebSocket把实时数据抛到网页端,老师傅们抱着手机就能巡线了。