委托与线程在C#编程中的应用及选择

简介: 委托是一种表示对具有特定参数列表和返回类型的方法的引用的类型。可以使用委托将方法作为参数传递给其他方法,或者异步地调用方法。线程是一个执行单元,它可以与进程中的其他线程并发运行。可以使用线程来同时执行多个任务,或者并行化计算密集型的工作。委托和线程之间的区别在于,委托是一种引用方法的方式,而线程是一种执行方法的方式。可以使用委托在不同的线程上调用方法,要么使用委托的 BeginInvoke 和 EndInvoke 方法,要么使用 ThreadPool 或 Task 类。

显示器11.jpg

委托是一种表示对具有特定参数列表和返回类型的方法的引用的类型。可以使用委托将方法作为参数传递给其他方法,或者异步地调用方法

线程是一个执行单元,它可以与进程中的其他线程并发运行。可以使用线程来同时执行多个任务,或者并行化计算密集型的工作

委托和线程之间的区别在于,委托是一种引用方法的方式,而线程是一种执行方法的方式。可以使用委托在不同的线程上调用方法,要么使用委托的 BeginInvoke 和 EndInvoke 方法,要么使用 ThreadPool 或 Task 类

在爬虫程序中,哪一种更合适取决于具体的设计和需求。一般来说,使用委托与 ThreadPool 或 Task 比创建和管理自己的线程更高效和方便。但是,如果想要更多地控制线程的优先级、身份或生命周期,可能需要使用线程。也可能想要考虑使用 C# 5 或更高版本中的 async/await 关键字,它们使异步编程变得更容易和清晰。例如采集每天采集TOP5的新闻网站,归纳整理当天的热点新闻为案例,代码如下:

usingSystem;
usingSystem.Collections.Generic;
usingSystem.Linq;
usingSystem.Net;
usingSystem.Net.Http;
usingSystem.Threading.Tasks;
usingHtmlAgilityPack;
namespaceAsyncWebCrawler{
classProgram    {
staticasyncTaskMain(string[] args)
        {
varnewsSites=newList<string>()
            {
"https://www.bbc.com/news",
"https://www.cnn.com/",
"https://www.nytimes.com/",
"https://www.theguardian.com/international",
"https://www.aljazeera.com/",
            };
varhttpClientHandler=newHttpClientHandler            {
// 亿牛云 动态转发代理IP// 爬虫加强版 IP和端口                 stringproxyHost="www.16yun.cn";
stringproxyPort="31111";
// 爬虫加强版 代理验证信息stringproxyUser="16YUN";
stringproxyPass="16IP";
// 设置代理服务器               Proxy=newWebProxy(string.Format("{0}:{1}", proxyHost, proxyPort), true),
UseProxy=true,
Credentials=newNetworkCredential(proxyUser, proxyPass)
            };
varhttpClient=newHttpClient(httpClientHandler);
vartasks=newList<Task<NewsSite>>();
// 开始异步采集foreach (varnewsSiteinnewsSites)
            {
tasks.Add(CollectNewsSiteAsync(httpClient, newsSite));
            }
// 等待所有异步采集任务完成varresults=awaitTask.WhenAll(tasks);
// 处理采集结果,整理热点标题varhotTopics=newList<string>();
foreach (varresultinresults)
            {
hotTopics.AddRange(result.GetHotTopics());
            }
// 输出结果Console.WriteLine("Hot Topics:");
foreach (varhotTopicinhotTopics)
            {
Console.WriteLine(hotTopic);
            }
        }
// 异步采集新闻网站首页staticasyncTask<NewsSite>CollectNewsSiteAsync(HttpClienthttpClient, stringnewsSite)
        {
varhtml=awaithttpClient.GetStringAsync(newsSite);
vardocument=newHtmlDocument();
document.LoadHtml(html);
returnnewNewsSite(newsSite, document);
        }
// 新闻网站类classNewsSite        {
privatestring_url;
privateHtmlDocument_document;
publicNewsSite(stringurl, HtmlDocumentdocument)
            {
_url=url;
_document=document;
            }
// 获取热点标题publicList<string>GetHotTopics()
            {
varhotTopics=newList<string>();
varheadlineNodes=_document.DocumentNode.SelectNodes("//h2[contains(@class,'headline')]");
if (headlineNodes!=null)
                {
foreach (varheadlineNodeinheadlineNodes)
                    {
hotTopics.Add($"{_url}: {headlineNode.InnerText}");
                    }
                }
returnhotTopics;
            }
        }
    }
}

该示例代码会采集五个新闻网站的首页,并整理当天的热点标题。采集过程是异步的,使用 async/await 关键字实现,同时使用动态转发代理IP提高采集效率。在处理采集结果时,代码会等待所有异步采集任务完成后再进行处理,以保证异步任务全部完成。

相关文章
|
17天前
|
存储 安全 编译器
学懂C#编程:属性(Property)的概念定义及使用详解
通过深入理解和使用C#的属性,可以编写更清晰、简洁和高效的代码,为开发高质量的应用程序奠定基础。
67 12
|
15天前
|
存储 安全 Java
Java多线程编程秘籍:各种方案一网打尽,不要错过!
Java 中实现多线程的方式主要有四种:继承 Thread 类、实现 Runnable 接口、实现 Callable 接口和使用线程池。每种方式各有优缺点,适用于不同的场景。继承 Thread 类最简单,实现 Runnable 接口更灵活,Callable 接口支持返回结果,线程池则便于管理和复用线程。实际应用中可根据需求选择合适的方式。此外,还介绍了多线程相关的常见面试问题及答案,涵盖线程概念、线程安全、线程池等知识点。
98 2
|
1月前
|
缓存 Java 调度
多线程编程核心:上下文切换深度解析
在现代计算机系统中,多线程编程已成为提高程序性能和响应速度的关键技术。然而,多线程编程中一个不可避免的概念就是上下文切换(Context Switching)。本文将深入探讨上下文切换的概念、原因、影响以及优化策略,帮助你在工作和学习中深入理解这一技术干货。
48 10
|
2月前
|
缓存 Java 开发者
Java多线程编程的陷阱与最佳实践####
本文深入探讨了Java多线程编程中常见的陷阱,如竞态条件、死锁和内存一致性错误,并提供了实用的避免策略。通过分析典型错误案例,本文旨在帮助开发者更好地理解和掌握多线程环境下的编程技巧,从而提升并发程序的稳定性和性能。 ####
|
1月前
|
安全 算法 Java
Java多线程编程中的陷阱与最佳实践####
本文探讨了Java多线程编程中常见的陷阱,并介绍了如何通过最佳实践来避免这些问题。我们将从基础概念入手,逐步深入到具体的代码示例,帮助开发者更好地理解和应用多线程技术。无论是初学者还是有经验的开发者,都能从中获得有价值的见解和建议。 ####
|
1月前
|
Java 调度
Java中的多线程编程与并发控制
本文深入探讨了Java编程语言中多线程编程的基础知识和并发控制机制。文章首先介绍了多线程的基本概念,包括线程的定义、生命周期以及在Java中创建和管理线程的方法。接着,详细讲解了Java提供的同步机制,如synchronized关键字、wait()和notify()方法等,以及如何通过这些机制实现线程间的协调与通信。最后,本文还讨论了一些常见的并发问题,例如死锁、竞态条件等,并提供了相应的解决策略。
51 3
|
1月前
|
算法 调度 开发者
多线程编程核心:上下文切换深度解析
在多线程编程中,上下文切换是一个至关重要的概念,它直接影响到程序的性能和响应速度。本文将深入探讨上下文切换的含义、原因、影响以及如何优化,帮助你在工作和学习中更好地理解和应用多线程技术。
43 4
|
2月前
|
缓存 Java 开发者
Java多线程并发编程:同步机制与实践应用
本文深入探讨Java多线程中的同步机制,分析了多线程并发带来的数据不一致等问题,详细介绍了`synchronized`关键字、`ReentrantLock`显式锁及`ReentrantReadWriteLock`读写锁的应用,结合代码示例展示了如何有效解决竞态条件,提升程序性能与稳定性。
177 6
|
13天前
|
安全 Java API
【JavaEE】多线程编程引入——认识Thread类
Thread类,Thread中的run方法,在编程中怎么调度多线程
|
1月前
|
监控 Java 数据库连接
Java线程管理:守护线程与用户线程的区分与应用
在Java多线程编程中,线程可以分为守护线程(Daemon Thread)和用户线程(User Thread)。这两种线程在行为和用途上有着明显的区别,了解它们的差异对于编写高效、稳定的并发程序至关重要。
40 2