我正在开发一个 C# 自定义 OPC 客户端,我开始在控制台应用程序中编写以提高速度,一切都按照我的意愿完美运行。
然后我决定做一个windows窗体应用程序来获得视觉体验。
Windows 窗体应用程序只是简单地停止工作,大约一分钟后停止从 OPC 服务器读取数据。控制台应用程序不断读取和读取的位置。
我在 Debug模式下也找不到任何明显的东西。
我绝对是在抓紧救命稻草,希望有人能给出一些启示。
每个应用程序都使用 OPCFoundation 提供的 .dll 文件。
这是控制台应用程序
static void Main(string[] args)
{
Opc.URL url = new Opc.URL("opcda://localhost/RSLinx OPC Server");
Opc.Da.Server server = null;
OpcCom.Factory fact = new OpcCom.Factory();
server = new Opc.Da.Server(fact, null);
server.Connect(url, new Opc.ConnectData(new System.Net.NetworkCredential()));
// Create a group
Opc.Da.Subscription group;
Opc.Da.SubscriptionState groupState = new Opc.Da.SubscriptionState();
groupState.Name = "Group";
groupState.Active = true;
group = (Opc.Da.Subscription)server.CreateSubscription(groupState);
// add items to the group.
Opc.Da.Item[] items = new Opc.Da.Item[6];
items[0] = new Opc.Da.Item();
items[0].ItemName = "[UX1]F20:9";
items[1] = new Opc.Da.Item();
items[1].ItemName = "[UX1]F22:30";
items[2] = new Opc.Da.Item();
items[2].ItemName = "[UX1]F22:6";
items[3] = new Opc.Da.Item();
items[3].ItemName = "[UX1]F18:8";
items[4] = new Opc.Da.Item();
items[4].ItemName = "[UX1]F22:32";
items[5] = new Opc.Da.Item();
items[5].ItemName = "[UX1]F22:5";
items = group.AddItems(items);
group.DataChanged += new Opc.Da.DataChangedEventHandler(OnTransactionCompleted);
}
static void OnTransactionCompleted(object group, object hReq, Opc.Da.ItemValueResult[] items)
{
Console.WriteLine("------------------->");
Console.WriteLine("DataChanged ...");
for (int i = 0; i < items.GetLength(0); i++)
{
Console.WriteLine("Item DataChange - ItemId: {0}", items[i].ItemName);
Console.WriteLine(" Value: {0,-20}", items[i].Value);
Console.WriteLine(" TimeStamp: {0:00}:{1:00}:{2:00}.{3:000}",
items[i].Timestamp.Hour,
items[i].Timestamp.Minute,
items[i].Timestamp.Second,
items[i].Timestamp.Millisecond);
}
Console.WriteLine("-------------------<");
}
这是 WinForm 应用程序
public Form1()
{
InitializeComponent();
_Form1 = this;
}
public static Form1 _Form1;
public void update(string message)
{
this.richTextBox1.Text = message;
}
private void Form1_Load(object sender, EventArgs e)
{
readplc();
}
static void readplc()
{
Opc.URL url = new Opc.URL("opcda://localhost/RSLinx OPC Server");
Opc.Da.Server server = null;
OpcCom.Factory fact = new OpcCom.Factory();
server = new Opc.Da.Server(fact, null);
server.Connect(url, new Opc.ConnectData(new System.Net.NetworkCredential()));
// Create a group
Opc.Da.Subscription group;
Opc.Da.SubscriptionState groupState = new Opc.Da.SubscriptionState();
groupState.Name = "Group";
groupState.Active = true;
group = (Opc.Da.Subscription)server.CreateSubscription(groupState);
// add items to the group.
Opc.Da.Item[] items = new Opc.Da.Item[6];
items[0] = new Opc.Da.Item();
items[0].ItemName = "[UX1]F20:9";
items[1] = new Opc.Da.Item();
items[1].ItemName = "[UX1]F22:30";
items[2] = new Opc.Da.Item();
items[2].ItemName = "[UX1]F22:6";
items[3] = new Opc.Da.Item();
items[3].ItemName = "[UX1]F18:8";
items[4] = new Opc.Da.Item();
items[4].ItemName = "[UX1]F22:32";
items[5] = new Opc.Da.Item();
items[5].ItemName = "[UX1]F22:5";
items = group.AddItems(items);
group.DataChanged += new Opc.Da.DataChangedEventHandler(OnTransactionCompleted);
}
static void OnTransactionCompleted(object group, object hReq, Opc.Da.ItemValueResult[] items)
{
for (int i = 0; i < items.GetLength(0); i++)
{
UIUpdater TEXT = new UIUpdater();
TEXT.UpdateText(items.GetLength(0).ToString() + " t " + i.ToString() + "Item DataChange - ItemId:" + items[i].ItemName +
"Value: " + items[i].Value + " TimeStamp: " + items[i].Timestamp.Hour + ":" +
items[i].Timestamp.Minute + ":" + items[i].Timestamp.Second + ":" + items[i].Timestamp.Millisecond);
}
}
UIUpdate 类
class UIUpdater
{
public void UpdateText(string DATA)
{
Form1._Form1.update(DATA);
}
public class UpdateUI
{
public int updatedRows { get; set; }
public string Custom1 { get; set; }
public string Custom2 { get; set; }
public string Custom3 { get; set; }
public string exception { get; set; }
public plcTextStatus PLCStatus { get; set; }
}
有任何问题请提问!
最佳答案
正如所怀疑的那样,这是一个跨线程问题。问题是您不能从 UI 线程以外的任何其他线程更新 UI。交易完成的事件实际上是在一个单独的线程上调用的,因此它更新了 UI。
它可以工作一段时间,因为它相对容错,但是您可能会遇到死锁或抛出未被捕获(或报告)的异常。
虽然修复很简单。
在这个方法中:
public void update(string message)
{
this.richTextBox1.Text = message;
}
将其更改为:
public void update(string message)
{
richTextBox1.Invoke(
(MethodInvoker) delegate
{
richTextBox1.Text = message;
});
}
它的作用是告诉 richTextBox1 在其拥有的线程(又名 UI 线程)上“调用”或运行以下委托(delegate)(函数)。
您真的应该尽量避免在此代码中使用static 方法和引用。我看不出有任何理由表明您拥有的代码不应该是实例方法而不是静态方法。
作为旁注,我编写的 OPC 程序可以每秒处理数千个标签和数百个 UI 更新。您正在做的工作适用于小型演示程序,但不会很好地扩展。当体系结构增长时,您需要开始对 UI 更新进行批处理,这样您就不会忙于在更新内重复调用 UI 线程。
编辑
您遇到的另一个问题是使用本地引用(例如对 OPC 服务器和订阅)并演示了内存泄漏和僵尸对象。发生的事情是 readplc 方法超出范围,您已经创建了对保存在内存中的对象的引用。由于您无法取消订阅该事件,因此该事件一直在触发。这些变量应在 readplc 方法的范围之外声明,以便您可以正确取消订阅事件并关闭 OPC 服务器。否则您将离开僵尸订阅(查看 RSLinx OPC 诊断页面,您会看到所有订阅都在那里)。
关于C# OPC 应用程序相同的代码,但工作方式不同,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30259029/
我需要在客户计算机上运行Ruby应用程序。通常需要几天才能完成(复制大备份文件)。问题是如果启用sleep,它会中断应用程序。否则,计算机将持续运行数周,直到我下次访问为止。有什么方法可以防止执行期间休眠并让Windows在执行后休眠吗?欢迎任何疯狂的想法;-) 最佳答案 Here建议使用SetThreadExecutionStateWinAPI函数,使应用程序能够通知系统它正在使用中,从而防止系统在应用程序运行时进入休眠状态或关闭显示。像这样的东西:require'Win32API'ES_AWAYMODE_REQUIRED=0x0
我在从html页面生成PDF时遇到问题。我正在使用PDFkit。在安装它的过程中,我注意到我需要wkhtmltopdf。所以我也安装了它。我做了PDFkit的文档所说的一切......现在我在尝试加载PDF时遇到了这个错误。这里是错误:commandfailed:"/usr/local/bin/wkhtmltopdf""--margin-right""0.75in""--page-size""Letter""--margin-top""0.75in""--margin-bottom""0.75in""--encoding""UTF-8""--margin-left""0.75in""-
我在我的项目目录中完成了compasscreate.和compassinitrails。几个问题:我已将我的.sass文件放在public/stylesheets中。这是放置它们的正确位置吗?当我运行compasswatch时,它不会自动编译这些.sass文件。我必须手动指定文件:compasswatchpublic/stylesheets/myfile.sass等。如何让它自动运行?文件ie.css、print.css和screen.css已放在stylesheets/compiled。如何在编译后不让它们重新出现的情况下删除它们?我自己编译的.sass文件编译成compiled/t
对于具有离线功能的智能手机应用程序,我正在为Xml文件创建单向文本同步。我希望我的服务器将增量/差异(例如GNU差异补丁)发送到目标设备。这是计划:Time=0Server:hasversion_1ofXmlfile(~800kiB)Client:hasversion_1ofXmlfile(~800kiB)Time=1Server:hasversion_1andversion_2ofXmlfile(each~800kiB)computesdeltaoftheseversions(=patch)(~10kiB)sendspatchtoClient(~10kiBtransferred)Cl
我试图获取一个长度在1到10之间的字符串,并输出将字符串分解为大小为1、2或3的连续子字符串的所有可能方式。例如:输入:123456将整数分割成单个字符,然后继续查找组合。该代码将返回以下所有数组。[1,2,3,4,5,6][12,3,4,5,6][1,23,4,5,6][1,2,34,5,6][1,2,3,45,6][1,2,3,4,56][12,34,5,6][12,3,45,6][12,3,4,56][1,23,45,6][1,2,34,56][1,23,4,56][12,34,56][123,4,5,6][1,234,5,6][1,2,345,6][1,2,3,456][123
如何在buildr项目中使用Ruby?我在很多不同的项目中使用过Ruby、JRuby、Java和Clojure。我目前正在使用我的标准Ruby开发一个模拟应用程序,我想尝试使用Clojure后端(我确实喜欢功能代码)以及JRubygui和测试套件。我还可以看到在未来的不同项目中使用Scala作为后端。我想我要为我的项目尝试一下buildr(http://buildr.apache.org/),但我注意到buildr似乎没有设置为在项目中使用JRuby代码本身!这看起来有点傻,因为该工具旨在统一通用的JVM语言并且是在ruby中构建的。除了将输出的jar包含在一个独特的、仅限ruby
我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i
Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack
在rails源中:https://github.com/rails/rails/blob/master/activesupport/lib/active_support/lazy_load_hooks.rb可以看到以下内容@load_hooks=Hash.new{|h,k|h[k]=[]}在IRB中,它只是初始化一个空哈希。和做有什么区别@load_hooks=Hash.new 最佳答案 查看rubydocumentationforHashnew→new_hashclicktotogglesourcenew(obj)→new_has
我想用ruby编写一个小的命令行实用程序并将其作为gem分发。我知道安装后,Guard、Sass和Thor等某些gem可以从命令行自行运行。为了让gem像二进制文件一样可用,我需要在我的gemspec中指定什么。 最佳答案 Gem::Specification.newdo|s|...s.executable='name_of_executable'...endhttp://docs.rubygems.org/read/chapter/20 关于ruby-在Ruby中编写命令行实用程序