【转】C#多线程下的调优

本文转载自博客园-包子wxl:https://www.cnblogs.com/wei325/p/16065342.html 一、原子操作 先看一段问题代码 /// <summary> /// 获取自增 /// </summary> public static void GetIncrement() { long result = 0; Console.WriteLine("开始计算"); //10个并发执行 Parallel.For(0, 10, (i) => { for (int j = 0; j < 10000; j++) { result++; } }); Console.WriteLine("结束计算"); Console.WriteLine($"result正确值应为:{10000 * 10}"); Console.WriteLine($"result 现值为:{result}"); Console.ReadLine(); } 这是多线程下,result的值不同步的原因。 1.基于Lock实现 平时大家用的最多的应该就是加锁了,同一时间,只有一个线程进入代码块。 实现代码: private static Object _obj = new object(); /// <summary> /// 原子操作基于Lock实现 /// </summary> public static void AtomicityForLock() { long result = 0; Console....

April 14, 2022 · 8 分钟 · Remo

C# 重写(override)Equals 和 GetHashCode

Equals 和 GetHashCode Equals 每个实现都必须遵循以下约定: 自反性(Reflexive): x.Equals(x) 必须返回 true 对称性(Symmetric): x.Equals(y) 为 true 时,y.Equals(x) 也为 true 传递性(Transitive): 对于任何非 null 的应用值 x, y 和 z,如果 x.Equals(y) 返回 true,并且 y.Equals(z) 也返回true,那么 x.Equals(z) 必须返回 true 一致性(Consistence): 如果多次将对象与另一个对象比较,结果始终相同。只要未修改 x 和 y 的应用对象,x.Equals(y) 连续调用 x.Equals(y) 返回相同的值 非null(Non-null): 如果 x 不是 null,y 为 null,则 x.Equals(y) 必须为 false GetHashCode: 两个相等对象根据 equals 方法比较时相等,那么这两个对象中任意一个对象的 GetHashCode 方法都必须产生同样的整数 在我们未对对象进行修改时,多次调用 GetHashCode 使用返回同一个整数。在同一个应用程序中多次执行,每次执行返回的整数可以不一致 如果两个对象根据 Equals 方法比较不相等时,那么调用这两个对象中任意一个对象的 GetHashCode 方法,不一同的整数。但不同的对象,产生不同整数,有可能提高散列表的性能 以上内容摘自@冯辉 的博客: https://www....

April 2, 2022 · 2 分钟 · Remo

ASP.NET 6 修改 WebRoot 路径

问题 最近的项目中需要修改 Web Root 路径,按照老方法发现报异常,于是 Google 得知相关方法在 ASP.NET 6 中有所修改。 代码 ASP.NET 5 public static IHostBuilder CreateHostBuilder(string[] args) => Host.CreateDefaultBuilder(args) .ConfigureWebHostDefaults(webBuilder => { //这里指定新位置 webBuilder.UseWebRoot("webroot") .UseStartup<Startup>(); }); ASP.NET 6 var builder = WebApplication.CreateBuilder(new WebApplicationOptions { Args = args, //这里指定新位置,也可以使用绝对路径 WebRootPath = "webroot" }); var app = builder.Build(); 若要修改 Content Root 路径也可以在这里一并修改。 总结 根据微软官方文档解释: Content 根目录是指向以下内容的基路径: 托管应用的可执行文件 (.exe)。 构成应用程序的已编译程序集 (.dll)。 应用使用的内容文件,例如:...

December 23, 2021 · 1 分钟 · Remo

Asp.Net Core使用Quartz.NET实现简单定时任务

所需 Nuget 包 Quartz.AspNetCore Quartz Quartz.Extensions.DependencyInjection Quartz.Extensions.Hosting 本示例仅需从nuget安装第一个包即可,其他三个包会通过依赖关系自动安装。 示例代码 本文基于.NET 6最新的模板。 新建一个cs文件,如ExampleJob.cs: [DisallowConcurrentExecution] public class ExampleJob : IJob { public async Task Execute(IJobExecutionContext context) { try { //这里是需要定时执行的相关代码 } catch (Exception e) { //异常处理 } } } 在Program.cs中添加: builder.Services.AddQuartz(config => { config.UseDefaultThreadPool(options => { options.MaxConcurrency = 2; }); config.UseMicrosoftDependencyInjectionJobFactory(); config.ScheduleJob<ExampleJob>(trigger => trigger .WithIdentity("ExampleTrigger") //立即开始第一次执行 .StartNow() //此后每次执行的间隔,这里是1小时,并且一直重复下去 .WithSimpleSchedule(x=>x.WithIntervalInHours(1).RepeatForever()) .WithDescription("A simple example")); }); // Quartz.Extensions.Hosting hosting builder....

December 7, 2021 · 1 分钟 · Remo

AspectCore 配合 Log4Net 全局记录Asp.Net Core Web Api异常日志

配置AspectCore-Framework 首先通过Nuget安装AspectCore.Extensions.DependencyInjection。 接下来我们可以配置拦截器: //继承自AbstractInterceptorAttribute public class CustomInterceptorAttribute : AbstractInterceptorAttribute { public override async Task Invoke(AspectContext context, AspectDelegate next) { try { Console.WriteLine("调用前"); await next(context);//执行调用的方法等 } catch (Exception ex) { Console.WriteLine("捕获到异常"); throw; } finally { Console.WriteLine("调用结束"); } } } 然后在Startup.cs中的ConfigureServices方法中配置代理(.NET6中对应的是builder.Services) services.ConfigureDynamicProxy(config => { services.AddTransient<IExmapleService, ExmapleService>(); services.AddControllers(); config.Interceptors.AddTyped<CustomInterceptorAttribute>(Predicates.ForService("*Service")); config.Interceptors.AddTyped<CustomInterceptorAttribute>(Predicates.ForService("*Controller")); }); 以上配置是对名称以Service或Controller结尾的进行代理,还有其他一些规则如: //全部代理 config.Interceptors.AddTyped<CustomInterceptorAttribute>(); //名称以Execute开头的方法会被代理 config.Interceptors.AddTyped<CustomInterceptorAttribute>(Predicates.ForMethod("Execute*")); //以Service结尾则不会被代理 config.NonAspectPredicates.AddService("Service"); //更多详细规则可以查阅官方文档 //https://github.com/dotnetcore/AspectCore-Framework/blob/master/docs/1.%E4%BD%BF%E7%94%A8%E6%8C%87%E5%8D%97.md ...... 也可以通过NonAspectAttribute对Service或Method进行单独设置: //无论在Startup.cs中如何配置,该接口都不会通过代理 [NonAspect] public interface IExampleService { void Method(); } 最后在Program....

February 27, 2021 · 2 分钟 · Remo