c# - 在 C# 应用程序中使用多线程时出现 OutOfMemory 异常

标签 c# multithreading out-of-memory

我正在用 C# 编写一个应用程序,它循环浏览 Wikipedia 本地数据库副本的文章。我使用一堆正则表达式在这些文章中找到正确的信息,启动一个线程来为每篇文章获取图像,保存信息并转到下一篇文章。

我需要使用代理列表来下载这些图像,以免被谷歌禁止。 由于代理可能很慢,我使用线程进行并行下载。

如果我不使用线程,应用程序可以正常工作,但需要一段时间才能获取所有信息。

如果我使用线程,应用程序会一直运行,直到它使用大约 500 个线程,然后我得到一个 OutOfMemory 异常。

问题是它只使用 ~300Mo 的 RAM,所以它不会使用可用总内存 (8Go) 和分配给单个 32 位应用程序的内存中的所有内存。

每个应用程序的线程数是否有限制?

编辑:

这是下载海报的代码(以 getPosterAsc() 开头)。

    string ddlValue = "";
    private void tryDownload(object obj)
    {
        WebClient webClientProxy = new WebClient();
        Tuple<WebProxy, int> proxy = (Tuple<WebProxy, int>)((object[])obj)[0];
        if (proxy != null)
            webClientProxy.Proxy = proxy.Item1;
        try
        {
            ddlValue = webClientProxy.DownloadString((string)((object[])obj)[1]);
        }
        catch (Exception ex) { 
            ddlValue = "";

            Console.WriteLine("trydownload:" + ex.Message);
        }

        webClientProxy.Dispose();
    }

    public void getPoster(object options = null)
    {
        if (options == null)
            options = new object[2] { toSave, false };
        if (!AppVar.debugMode && AppVar.getImages && this.getImage)
        {
            if (this.original_name != "" && !this.ambName && this.suitable)
            {
                Log.CountImgInc();

                MatchCollection MatchList;
                string basic_options = "";
                string value = "";
                WebClient webClient = new WebClient();
                Regex reg;
                bool found = false;

                if (original_name.Split(' ').Length > 1) image_options = "";

                if (!found)
                {
                    bool succes = false;
                    int countTry = 0;
                    while (!succes)
                    {
                        Tuple<WebProxy, int> proxy = null;
                        if (countTry != 5)
                            proxy = Proxy.getProxy();

                        try
                        {
                            Thread t = new Thread(tryDownload);
                            if (!(bool)((object[])options)[1])
                                t.Start(new object[] { proxy, @"http://www.google.com/search?as_st=y&tbm=isch&as_q=" + image_options + "+" + basic_options + "+" + image_options_before + "%22" + simplify(original_name) + "%22+" + " OR %22" + original_name + "%22+" + image_options_after + this.image_format });
                            else
                                t.Start(new object[] { proxy, @"http://www.google.com/search?as_st=y&tbm=isch&as_q=" + image_options + "+" + basic_options + "+" + image_options_before + "%22" + simplify(original_name) + "%22+" + " OR %22" + original_name + "%22+" + image_options_after + "&biw=1218&bih=927&tbs=isz:ex,iszw:758,iszh:140,ift:jpg&tbm=isch&source=lnt&sa=X&ei=kuG7T6qaOYKr-gafsOHNCg&ved=0CIwBEKcFKAE" });
                            if (!t.Join(40000))
                            {
                                Proxy.badProxy(proxy.Item1.Address.Host, proxy.Item1.Address.Port);
                                continue;
                            }
                            else
                            {
                                value = ddlValue;
                                if (value != "")
                                    succes = true;
                                else
                                    Proxy.badProxy(proxy.Item1.Address.Host, proxy.Item1.Address.Port);
                            }
                        }
                        catch (Exception ex)
                        {
                            if (proxy != null)
                                Proxy.badProxy(proxy.Item1.Address.Host, proxy.Item1.Address.Port);
                        }
                        countTry++;
                    }

                    reg = new Regex(@"imgurl\=(.*?)&amp;imgrefurl", RegexOptions.IgnoreCase);
                    MatchList = reg.Matches(value);
                    if (MatchList.Count > 0)
                    {
                        bool foundgg = false;
                        int j = 0;
                        while (!foundgg && MatchList.Count > j)
                        {
                            if (MatchList[j].Groups[1].Value.Substring(MatchList[j].Groups[1].Value.Length - 3, 3) == "jpg")
                            {
                                try
                                {
                                    string guid = Guid.NewGuid().ToString();
                                    webClient.DownloadFile(MatchList[j].Groups[1].Value, @"c:\temp\" + guid + ".jpg");

                                    FileInfo fi = new FileInfo(@"c:\temp\" + guid + ".jpg");
                                    this.image_size = fi.Length;

                                    using (Image img = Image.FromFile(@"c:\temp\" + guid + ".jpg"))
                                    {
                                        int minHeight = this.cov_min_height;
                                        if ((bool)((object[])options)[1])
                                            minHeight = 100;

                                        if (img.RawFormat.Equals(System.Drawing.Imaging.ImageFormat.Jpeg) && img.HorizontalResolution > 70 && img.Size.Height > minHeight && img.Size.Width > this.cov_min_width && this.image_size < 250000)
                                        {
                                            foundgg = true;
                                            image_name = guid;
                                            image_height = img.Height;
                                            image_width = img.Width;
                                            img.Dispose();
                                            if ((bool)((object[])options)[0])
                                            {
                                                Mediatly.savePoster(this, (bool)((object[])options)[1]);
                                            }
                                        }
                                        else
                                        {
                                            img.Dispose();
                                            File.Delete(@"c:\temp\" + guid.ToString() + ".jpg"); 
                                        }
                                    }
                                }
                                catch (Exception ex)
                                {
                                }
                            }

                            j++;
                        }
                    }
                }

                webClient.Dispose();
                Log.CountImgDec();
            }
        }
    }

    public void getPosterAsc(bool save = false, bool banner = false)
    {
        ThreadPool.QueueUserWorkItem(new WaitCallback(getPoster), new object[2] { save, banner });
    }

最佳答案

我会确保您正在使用线程池来“管理”您的线程。正如有人所说,每个线程消耗大约 1MB 的内存,并且根据系统硬件,这可能会导致您的问题。

解决此问题的一种可能方法是使用线程池。这减少了通过在可能的情况下共享和回收线程来生成所有线程所产生的开销。这允许低级线程设施(有许多线程处于事件状态)但限制了这样做的性能损失。

线程池还会限制同时运行的工作线程(注意,这些都是后台线程)的数量。太多的操作线程是一个很大的管理开销,可以“render the CPU cache ineffective”。一旦达到您将施加的线程池限制,额外的作业将排队并在另一个工作线程空闲时执行。我觉得这是一种更有效、更安全且资源效率更高的方式来执行您的要求。

根据您当前的代码,有多种方法可以进入线程池:

  1. BackgroundWorker
  2. ThreadPool.QueueUserWorkItem
  3. 异步委托(delegate)。
  4. PLINQ。

我个人会使用 TPL,因为它很棒! 我希望这会有所帮助。

关于c# - 在 C# 应用程序中使用多线程时出现 OutOfMemory 异常,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10716254/

相关文章:

c# - 将 System.IO.Abstractions 与 DirectoryInfo 一起使用

c# - Powershell Mysql 错误

c++ - qt信号槽多线程死锁

java - 防止 ConcurrentLinkedQueue<T> 出现内存不足异常

ruby - 服务器定期无响应,OOM Killer 不活动?

c# - 测试 ReactiveCommand 和 ReactiveObject ViewModel

c# - 如何从同一应用程序的两个实例安全地写入日志文件?

java - 如何使用 JavaFX 使 Label 的内容在一秒后发生变化?

java - 如何检查线程内主线程的变量?

java - 创建镜像时避免内存不足