4

我正在使用 带有 C# .NET 的OPCSiemensDAAutomation dll 从 OPC Server 检索标签的值。我设法使用 QueryAvailableProperties()和 GetItemProperties()检索值,但目标是检索每个请求的 500k 个标签值。

我已经用 100 个标签进行了测试,代码在 45 秒内完成,多线程导致 100 个标签的 30 秒小幅改进。以当前速度达到目标标签量需要 4 个多小时。有什么方法可以批量检索标签值并具有更好的性能?谢谢。

var opcServer = new OPCSiemensDAAutomation.OPCServer();
opcServer.Connect("PCS7.OPCDAServer.1");
ConcurrentBag<DataRow> myBag = new ConcurrentBag<DataRow>(dt.AsEnumerable().ToList());
Parallel.ForEach(myBag, data =>
{
    if (count <= num)
    {
        int cnt;
        Array propertyIds, descriptions, dataTypes, errors, vals;
        try
        {
            opcServer.QueryAvailableProperties(data[0].ToString(), out cnt, out propertyIds, out descriptions, out dataTypes);
            opcServer.GetItemProperties(data[0].ToString(), cnt, propertyIds, out vals, out errors);
            Tags tag = new Tags();
            tag.Id = data[0].ToString();
            tag.Value = vals.GetValue(2).ToString();
            tags.Add(tag);
            Interlocked.Increment(ref count);
        }
        catch
        { }
    }
});
4

1 回答 1

1

您可以创建 OPC 组:

OPCGroup myGroup = myServer.addGroup(groupName, isActive, isSubscribed, updateRate);

然后您可以将标签添加到您的组:

myGroup.OPCItems.AddItem("FullAddress", ClientHandle) //a unique number inside the group

FullAddressOPCChannel 名称、连接名称和完整地址组成,即:S7:[MyPLCName]DB1.dbx4

当组完全填充后,您可以一次读取所有变量。

int itemCount = myGroup.OPCItems.Count;
object qualities = null;
object timeStamps = null;
object errors = null;
int serverHandles[itemCount];
Array values = Array.CreateInstance(TypeOf(object), {itemCount },{1})
for (int i = 0; i < itemCount; i++){
   serverHandles[i] = myGroup.OPCItems.Item(i + 1).ServerHandle;
   values.SetValue("", i);
}

myGroup.SyncRead(OPCSiemensDAAutomation.OPCDataSource.OPCDevice, itemCount + 1, ServerHandles, values, errors, qualities, timeStamps);

然后你将有四个与第一个相关的新数组serverHandles

qualities在使用数组中的数据之前检查数组是明智的values

于 2019-11-29T20:11:45.533 回答