C#教程

关注公众号 jb51net

关闭
首页 > 软件编程 > C#教程 > C#通过HSLCommunication库操作PLC

C#通过HSLCommunication库操作PLC用法

作者:工业程序猿老赵

本文主要介绍了C#通过HSLCommunication库操作PLC用法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

你需要掌握 C# 通过 HSLCommunication 库操作 PLC 的完整流程和实用用法,下面将从环境搭建、核心操作、资源管理到完整示例进行详细讲解,以主流西门子 S7-1200/1500 PLC 为例,其他品牌(三菱、欧姆龙等)用法可类比替换。

一、环境准备(前提步骤)

1. 安装 HSLCommunication 库

HSLCommunication 以 NuGet 包形式分发,两种安装方式任选:

Install-Package HslCommunication

2. 引入核心命名空间

在 C# 代码文件头部引入必要命名空间,针对西门子 PLC 的核心命名空间如下:

using System;
using System.Collections.Generic;
using HslCommunication;          // 核心公共命名空间
using HslCommunication.Profinet;  // 工业总线公共命名空间
using HslCommunication.Profinet.Siemens;  // 西门子PLC专属命名空间
// 三菱PLC替换为:using HslCommunication.Profinet.Mitsubishi;
// 欧姆龙PLC替换为:using HslCommunication.Profinet.Omron;

二、核心操作流程

HSLCommunication 操作 PLC 的统一核心流程:创建 PLC 实例 → 建立连接 → 读写操作 → 释放资源

1. 创建 PLC 通信实例

针对不同品牌 PLC,HSLCommunication 提供了专属实例类,以西门子 S7-1200/1500 为例,创建 TCP/IP 通信实例:

// 构造函数参数说明:
// 1. PLC型号(S71200/S71500/S7300/S7400)
// 2. PLC的局域网IP地址(需与电脑在同一网段)
// 3. 机架号(默认0,无需修改)
// 4. 槽号(S71200/1500默认1;S7300默认2)
SiemensS7Net plcClient = new SiemensS7Net(SiemensPLCS.S71200, "192.168.1.100", 0, 1);

// 可选配置:设置连接超时和读写超时(单位:毫秒,默认3000)
plcClient.ConnectTimeOut = 5000;  // 连接超时时间
plcClient.ReceiveTimeOut = 5000;  // 读写超时时间

其他品牌 PLC 实例创建示例:

2. 建立 PLC 连接

使用ConnectServer()方法建立连接,该方法返回OperateResult对象(HSL 核心返回类型,用于判断操作是否成功):

// 建立与PLC的连接
OperateResult connectResult = plcClient.ConnectServer();

// 判断连接是否成功
if (connectResult.IsSuccess)
{
    Console.WriteLine("PLC连接成功!");
    // 连接成功后,执行读写操作
}
else
{
    // 连接失败,输出错误信息
    Console.WriteLine($"PLC连接失败!错误原因:{connectResult.Message}");
    // 连接失败时,建议释放资源
    plcClient.Dispose();
    return;
}

3. 核心读写操作(最常用)

HSLCommunication 提供了丰富的读写方法,核心分为单个地址读写批量地址读写,支持 bool、int16、int32、float 等常用数据类型。

(1)单个地址读写

① 布尔类型(位操作:M0.0、I0.1、Q0.2、DB1.DBX0.0)

布尔类型是 PLC 的位状态,对应输入、输出、中间继电器等,核心方法:

读取:ReadBool(string address)

写入:Write(string address, bool value)

try
{
    // 读取M0.0的状态
    OperateResult<bool> readBoolResult = plcClient.ReadBool("M0.0");
    if (readBoolResult.IsSuccess)
    {
        bool m00Value = readBoolResult.Content; // 获取读取结果
        Console.WriteLine($"M0.0 当前值:{m00Value}");
    }
    else
    {
        Console.WriteLine($"读取M0.0失败:{readBoolResult.Message}");
    }

    // 写入M0.0为true(置1)
    OperateResult writeBoolResult = plcClient.Write("M0.0", true);
    if (writeBoolResult.IsSuccess)
    {
        Console.WriteLine("M0.0 写入true成功!");
    }
    else
    {
        Console.WriteLine($"M0.0 写入失败:{writeBoolResult.Message}");
    }
}
catch (Exception ex)
{
    Console.WriteLine($"布尔类型读写异常:{ex.Message}");
}

② 数值类型(int32、float、int16 等)

以工业场景最常用的int32(双字,对应 PLC DInt 类型)和float(实数,对应 PLC Real 类型)为例:

读取:ReadInt32(string address)ReadFloat(string address)

写入:Write(string address, int value)Write(string address, float value)

try
{
    // 1. 读写int32类型(对应PLC DB1.DBD0,数据块双字地址)
    // 读取DB1.DBD0的值
    OperateResult<int> readInt32Result = plcClient.ReadInt32("DB1.DBD0");
    if (readInt32Result.IsSuccess)
    {
        int db1Dbd0Value = readInt32Result.Content;
        Console.WriteLine($"DB1.DBD0(int32):{db1Dbd0Value}");
    }
    else
    {
        Console.WriteLine($"读取DB1.DBD0失败:{readInt32Result.Message}");
    }

    // 写入DB1.DBD0为2025
    OperateResult writeInt32Result = plcClient.Write("DB1.DBD0", 2025);
    if (writeInt32Result.IsSuccess)
    {
        Console.WriteLine("DB1.DBD0 写入2025成功!");
    }
    else
    {
        Console.WriteLine($"DB1.DBD0 写入失败:{writeInt32Result.Message}");
    }

    // 2. 读写float类型(对应PLC DB1.DBD4,数据块实数地址)
    // 读取DB1.DBD4的值
    OperateResult<float> readFloatResult = plcClient.ReadFloat("DB1.DBD4");
    if (readFloatResult.IsSuccess)
    {
        float db1Dbd4Value = readFloatResult.Content;
        Console.WriteLine($"DB1.DBD4(float):{db1Dbd4Value:F2}"); // 保留2位小数输出
    }
    else
    {
        Console.WriteLine($"读取DB1.DBD4失败:{readFloatResult.Message}");
    }

    // 写入DB1.DBD4为6.66f
    OperateResult writeFloatResult = plcClient.Write("DB1.DBD4", 6.66f);
    if (writeFloatResult.IsSuccess)
    {
        Console.WriteLine("DB1.DBD4 写入6.66成功!");
    }
    else
    {
        Console.WriteLine($"DB1.DBD4 写入失败:{writeFloatResult.Message}");
    }
}
catch (Exception ex)
{
    Console.WriteLine($"数值类型读写异常:{ex.Message}");
}

(2)批量读写操作(高效推荐)

当需要读写多个 PLC 地址时,批量操作可减少网络通信次数,提升效率,核心用法如下:

批量读取:传入地址数组,返回结果数组

批量写入:传入键值对字典(地址为键,值为待写入数据)

try
{
    // 1. 批量读取布尔类型(M0.0、M0.1、I0.0)
    string[] boolAddresses = new string[] { "M0.0", "M0.1", "I0.0" };
    OperateResult<bool[]> readBoolBatchResult = plcClient.ReadBool(boolAddresses);
    if (readBoolBatchResult.IsSuccess)
    {
        bool[] boolValues = readBoolBatchResult.Content;
        for (int i = 0; i < boolAddresses.Length; i++)
        {
            Console.WriteLine($"{boolAddresses[i]}:{boolValues[i]}");
        }
    }
    else
    {
        Console.WriteLine($"批量读取布尔类型失败:{readBoolBatchResult.Message}");
    }

    // 2. 批量写入(混合布尔、int32、float类型)
    Dictionary<string, object> writeDict = new Dictionary<string, object>()
    {
        { "M0.1", false },        // 布尔类型
        { "DB1.DBD8", 1000 },     // int32类型
        { "DB1.DBD12", 9.99f }    // float类型
    };
    OperateResult writeBatchResult = plcClient.Write(writeDict);
    if (writeBatchResult.IsSuccess)
    {
        Console.WriteLine("所有地址批量写入成功!");
    }
    else
    {
        Console.WriteLine($"批量写入失败:{writeBatchResult.Message}");
    }
}
catch (Exception ex)
{
    Console.WriteLine($"批量读写异常:{ex.Message}");
}

4. 资源释放(关键步骤)

通信完成后,需关闭连接并释放资源,避免内存泄漏,推荐两种方式:

方式 1:使用using语句(推荐,自动释放)

using 语句会自动调用对象的Dispose()方法,无需手动释放资源:

// using包裹PLC实例,代码块执行完毕后自动释放资源
using (SiemensS7Net plcClient = new SiemensS7Net(SiemensPLCS.S71200, "192.168.1.100", 0, 1))
{
    // 建立连接、执行读写操作...
    OperateResult connectResult = plcClient.ConnectServer();
    if (connectResult.IsSuccess)
    {
        // 读写逻辑
    }
} // 此处自动释放plcClient资源,无需手动调用Dispose()

方式 2:手动释放资源

若不使用using语句,需在操作完成后手动关闭连接并释放:

SiemensS7Net plcClient = new SiemensS7Net(SiemensPLCS.S71200, "192.168.1.100", 0, 1);
try
{
    // 建立连接、读写操作...
}
finally
{
    // 关闭连接
    plcClient.DisconnectServer();
    // 释放资源
    plcClient.Dispose();
    Console.WriteLine("PLC连接已关闭,资源已释放!");
}

三、关键注意事项

四、完整可运行示例代码

using System;
using System.Collections.Generic;
using HslCommunication;
using HslCommunication.Profinet.Siemens;

namespace HslPlcOperationDemo
{
    class Program
    {
        static void Main(string[] args)
        {
            // 使用using语句自动释放PLC资源
            using (SiemensS7Net plcClient = new SiemensS7Net(SiemensPLCS.S71200, "192.168.1.100", 0, 1))
            {
                try
                {
                    // 1. 建立PLC连接
                    OperateResult connectResult = plcClient.ConnectServer();
                    if (!connectResult.IsSuccess)
                    {
                        Console.WriteLine($"PLC连接失败:{connectResult.Message}");
                        Console.ReadKey();
                        return;
                    }
                    Console.WriteLine("PLC连接成功!\n");

                    // 2. 单个读写布尔类型(M0.0)
                    SingleReadWriteBool(plcClient);

                    // 3. 单个读写数值类型(int32 + float)
                    SingleReadWriteNumber(plcClient);

                    // 4. 批量读写操作
                    BatchReadWrite(plcClient);
                }
                catch (Exception ex)
                {
                    Console.WriteLine($"程序异常:{ex.Message}");
                }
                finally
                {
                    // 手动关闭连接(可选,using会自动释放)
                    plcClient.DisconnectServer();
                    Console.WriteLine("\nPLC连接已关闭!");
                }
            }

            Console.ReadKey();
        }

        /// <summary>
        /// 单个读写布尔类型
        /// </summary>
        private static void SingleReadWriteBool(SiemensS7Net plcClient)
        {
            // 读取M0.0
            OperateResult<bool> readResult = plcClient.ReadBool("M0.0");
            if (readResult.IsSuccess)
            {
                Console.WriteLine($"M0.0 读取值:{readResult.Content}");
            }
            else
            {
                Console.WriteLine($"M0.0 读取失败:{readResult.Message}");
            }

            // 写入M0.0为true
            OperateResult writeResult = plcClient.Write("M0.0", true);
            if (writeResult.IsSuccess)
            {
                Console.WriteLine("M0.0 写入true成功!\n");
            }
            else
            {
                Console.WriteLine($"M0.0 写入失败:{writeResult.Message}\n");
            }
        }

        /// <summary>
        /// 单个读写数值类型(int32 + float)
        /// </summary>
        private static void SingleReadWriteNumber(SiemensS7Net plcClient)
        {
            // 读写int32(DB1.DBD0)
            OperateResult<int> readIntResult = plcClient.ReadInt32("DB1.DBD0");
            if (readIntResult.IsSuccess)
            {
                Console.WriteLine($"DB1.DBD0(int32)读取值:{readIntResult.Content}");
            }
            else
            {
                Console.WriteLine($"DB1.DBD0 读取失败:{readIntResult.Message}");
            }

            OperateResult writeIntResult = plcClient.Write("DB1.DBD0", 2025);
            if (writeIntResult.IsSuccess)
            {
                Console.WriteLine("DB1.DBD0 写入2025成功!");
            }
            else
            {
                Console.WriteLine($"DB1.DBD0 写入失败:{writeIntResult.Message}");
            }

            // 读写float(DB1.DBD4)
            OperateResult<float> readFloatResult = plcClient.ReadFloat("DB1.DBD4");
            if (readFloatResult.IsSuccess)
            {
                Console.WriteLine($"DB1.DBD4(float)读取值:{readFloatResult.Content:F2}");
            }
            else
            {
                Console.WriteLine($"DB1.DBD4 读取失败:{readFloatResult.Message}");
            }

            OperateResult writeFloatResult = plcClient.Write("DB1.DBD4", 6.66f);
            if (writeFloatResult.IsSuccess)
            {
                Console.WriteLine("DB1.DBD4 写入6.66成功!\n");
            }
            else
            {
                Console.WriteLine($"DB1.DBD4 写入失败:{writeFloatResult.Message}\n");
            }
        }

        /// <summary>
        /// 批量读写操作
        /// </summary>
        private static void BatchReadWrite(SiemensS7Net plcClient)
        {
            // 批量读取布尔类型
            string[] boolAddrs = new string[] { "M0.0", "M0.1", "I0.0" };
            OperateResult<bool[]> batchReadResult = plcClient.ReadBool(boolAddrs);
            if (batchReadResult.IsSuccess)
            {
                Console.WriteLine("批量读取布尔类型结果:");
                for (int i = 0; i < boolAddrs.Length; i++)
                {
                    Console.WriteLine($"{boolAddrs[i]}:{batchReadResult.Content[i]}");
                }
            }
            else
            {
                Console.WriteLine($"批量读取布尔类型失败:{batchReadResult.Message}");
            }

            // 批量写入混合类型
            Dictionary<string, object> writeDict = new Dictionary<string, object>()
            {
                { "M0.1", false },
                { "DB1.DBD8", 1000 },
                { "DB1.DBD12", 9.99f }
            };
            OperateResult batchWriteResult = plcClient.Write(writeDict);
            if (batchWriteResult.IsSuccess)
            {
                Console.WriteLine("批量写入所有地址成功!");
            }
            else
            {
                Console.WriteLine($"批量写入失败:{batchWriteResult.Message}");
            }
        }
    }
}

到此这篇关于C#通过HSLCommunication库操作PLC用法的文章就介绍到这了,更多相关C#通过HSLCommunication库操作PLC内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

您可能感兴趣的文章:
阅读全文