[英]Non-Static and Static method and class memory allocation
静态问题:
在基于C#的应用程序中,我必须使用静态类中的某些静态方法。 这是我的代码的一部分:
//Timer Event (interval: 100ms)
private void OnTimedEvent2(object source, ElapsedEventArgs e)
{
try
{
//...
DataUpdaterClass.Update("SomeString");
//...
}
catch (Exception ex)
{
//...
}
我的问题是,如果我在每个TimerTick,
中都调用Update
方法TimerTick,
它将是每次分配内存还是一次分配内存?
非静态问题:
相同的Timer
但非静态方法。
static IPEndPoint MyIPEndPoint = new IPEndPoint(IPAddress.Parse("127.0.0.1"), 9999);
static Socket MySocket = null;
////Timer Event (interval: 100ms)
private void OnTimedEvent2(object source, ElapsedEventArgs e)
{
try
{
if (MySocket == null)
{
MySocket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
MySocket.Connect(MyIPEndPoint);
}
//...
}
catch (Exception ex)
{
MySocket.Dispose();
MySocket = null;
}
}
我的第二个问题是,如果我的程序抛出错误,那么我的Socket
将被释放并且将来不会使用任何内存吗?
非常感谢。
编辑:
感谢您的回答。 我觉得我需要为您添加一些其他信息。
我的应用程序是Windows服务,它可以收集来自我们工厂的工业设备中有关产品重量的数据。 因此它需要连续运行。 现在,我正在通过服务检查已使用的内存,并且看到它总是更大。 它的增长非常缓慢,我看到GC可以完成工作,但是使用的内存仍在增长。 我发布了完整的Timer
事件和Update
方法。 我希望你能帮助我。
类,变量和方法的名称是匈牙利语,希望不会引起问题。
计时器事件:
static string Adat1 =static string IP1 = Convert.ToString(_3DESDekodoloClass.Decrypt(ConfigurationManager.AppSettings["merleg_1_IP"]));
static int Port1 =Convert.ToInt32(_3DESDekodoloClass.Decrypt(ConfigurationManager.AppSettings["merleg_1_Port"]));
static int Olvasott1 = 0;
IPEndPoint Vegpont1 = new IPEndPoint(IPAddress.Parse(IP1), Port1);
Socket Merleg1 = null;
static byte[] Buffer1 = new byte[1024];
//Starts in the OnStart() event.
private void OnTimedEvent1(object source, ElapsedEventArgs e)
{
try
{
if (Merleg1 == null)
{
Merleg1 = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
Merleg1.Connect(Vegpont1);
}
Olvasott1 = Merleg1.Receive(Buffer1);
Adat1 = Encoding.ASCII.GetString(Buffer1, 0, Olvasott1);
AdafeltoltoClass.Merleg_1_adatfeltolto(Adat1);
}
finally
{
if (Merleg1 != null)
{
Merleg1.Dispose();
}
Merleg1 = null;
}
}
这是我的Update
方法:
static AnritsuDataCollector fo_osztaly = new AnritsuDataCollector();
static SqlDataAdapter sda_1 = new SqlDataAdapter();
static SqlDataAdapter sda_2 = new SqlDataAdapter();
static SqlDataAdapter sda_gauss_1 = new SqlDataAdapter();
static SqlDataAdapter sda_gauss_2 = new SqlDataAdapter();
//Adatok kinyerése a config fájlból dekódolás után.
static string data_source = _3DESDekodoloClass.Decrypt(ConfigurationManager.AppSettings["data_source"]);
static string initial_catalog = _3DESDekodoloClass.Decrypt(ConfigurationManager.AppSettings["initial_catalog"]);
static string user_id = _3DESDekodoloClass.Decrypt(ConfigurationManager.AppSettings["user_id"]);
static string password = _3DESDekodoloClass.Decrypt(ConfigurationManager.AppSettings["password"]);
static string ProductId1 = "";
static string RankCode1 = "";
static string Suly1 = "";
static string ProductId2 = "";
static string RankCode2 = "";
static string Suly2 = "";
internal static void Merleg_1_adatfeltolto(string beerkezo)
{
try
{
//Az "con" SQL kapcsolat Pool-ba tétele.
using (SqlConnection con = new SqlConnection(@"Data Source=" + data_source + ";Initial Catalog=" + initial_catalog + ";User ID=" + user_id + ";Password=" + password + ";"))
{
//A kapcsolat megnyitása.
con.Open();
ProductId1 = beerkezo.Substring(4, 2);
RankCode1 = beerkezo.Substring(6, 1);
Suly1 = beerkezo.Substring(7, 5);
//Ha a suly tartalmaz szóközt.
if (Suly1.Contains(' '))
{
Suly1 = "00.00";
}
//SQL parancs az adatok feltöltéséhez.
using (sda_1.InsertCommand = new SqlCommand("INSERT INTO meresek (product_id, rank_code, merleg, suly, ido) VALUES (@product_id, @rank_code, @merleg, @suly, @ido)", con))
{
sda_1.InsertCommand.Parameters.Add("@product_id", SqlDbType.Int).Value = Convert.ToInt32(ProductId1);
sda_1.InsertCommand.Parameters.Add("@rank_code", SqlDbType.VarChar).Value = Convert.ToString(RankCode1);
sda_1.InsertCommand.Parameters.Add("@merleg", SqlDbType.Int).Value = 1;
sda_1.InsertCommand.Parameters.Add("@suly", SqlDbType.Float).Value = Convert.ToDouble(Suly1.Replace('.', ','));
sda_1.InsertCommand.Parameters.Add("@ido", SqlDbType.DateTime).Value = DateTime.Now;
//Az SQL utasítás elküldése a szervernek
sda_1.InsertCommand.ExecuteNonQuery();
}
using (sda_gauss_1.UpdateCommand = new SqlCommand("IF EXISTS(SELECT ROUND(suly,0) FROM gauss_statisztika_merleg_1 WHERE ROUND(suly,0) = @suly AND product_id = @product_id) BEGIN UPDATE gauss_statisztika_merleg_1 SET darab = darab + 1, utolso_meres = @utolso_meres WHERE ROUND(suly,0) = @suly AND product_id = @product_id END ELSE BEGIN INSERT INTO gauss_statisztika_merleg_1 VALUES(@product_id, @suly, 1, 1, @utolso_meres) END", con))
{
sda_gauss_1.UpdateCommand.Parameters.Add("@suly", SqlDbType.Float).Value = Math.Round(Convert.ToDouble(Suly1.Replace('.', ',')));
sda_gauss_1.UpdateCommand.Parameters.Add("@product_id", SqlDbType.Int).Value = Convert.ToInt32(ProductId1);
sda_gauss_1.UpdateCommand.Parameters.Add("@utolso_meres", SqlDbType.DateTime).Value = DateTime.Now;
sda_gauss_1.UpdateCommand.ExecuteNonQuery();
}
}
}
catch (Exception ex)
{
fo_osztaly.Eventlog.WriteEntry(ex.Message);
}
}
静态问题:调用已订阅的处理程序不会触发内存分配。 代码已在内存中。 代码执行可能会触发内存分配。
非静态问题:您不能保证何时将对象从内存中删除-答案可能是。 您的示例有一个错误-您应将“ catch”更改为“ finaly”,并在处置之前检查MySocket是否为null。
其他:根据您的问题,我认为您不应该特别注意内存的有效使用。 它是托管语言,大多数内存管理工作都是由.NET框架本身完成的
问题#1:您在OnTimedEvent2
调用了静态方法:编译器不会为此分配任何内存,只需将代码放入即可调用该方法。 在该方法中,还有另外两套方法:如果它分配内存,它将用于每次调用。 但实际上,这取决于DataUpdaterClass.Update
实现。
问题2:将MySocket变量设置为null,从而删除对分配的套接字的所有引用,将使垃圾回收器释放内存。 GC将根据自己的逻辑释放它。 所以是的,您可以考虑分配的套接字使用的内存将被释放。
附带说明:不要过多地关注内存管理。 .NET实现了Garbage Collection,以使开发人员摆脱内存管理问题。 而且我从来没有发现GC无法正常工作的情况。
在某些情况下,您必须注意软件分配和不释放的内存量,这两种情况不在其中
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.