用.NET Core实现装饰模式和.NET Core的Stream简介

该文章综合了几本书的内容.

某咖啡店项目的解决方案

某咖啡店供应咖啡, 客户买咖啡的时候可以添加若干调味料, 最后要求算出总价钱.

Beverage是所有咖啡饮料的抽象类, 里面的cost方法是抽象的. description变量在每个子类里面都需要设置(表示对咖啡的描述).

每个子类实现cost方法, 表示咖啡的价格.

除了这些类之外, 还有调味品:

问题是调味品太多了, 如果使用继承来做的话, 各种组合简直是类的爆炸.

而且还有其他的问题, 如果牛奶的价格上涨了怎么办? 如果再加一种焦糖调料呢?

另一种解决方案

父类里面有调味料的变量(bool), 并且在父类里面直接实现cost方法(通过是否有某种调味料来计算价格).

子类override父类的cost方法, 但是也调用父类的cost方法, 这样就可以把子类这个咖啡的价格和父类里计算出来的调味料的价格加到一起算出最终的价格了.

下面就是:

看起来不错, 那么, 问题来了:

  • 调味料价格变化的话就不得不修改现有代码了

  • 如果有新的调味料那么久必须在父类里面添加新的方法, 并且修改cost方法了.

  • 可能某些调味料根本不适用于某些咖啡

  • 如果某个客户想要两个摩卡(调味料)怎么办?

设计原则

类应该对扩展开放 而对修改关闭.

装饰模式

使用装饰模式, 我们可以购买一个咖啡, 并且在运行时使用调味料对它进行装饰.

大约步骤如下:

  1. 买某种咖啡

  2. 使用mocha调味料装饰它

  3. 使用whip调味料装饰它

  4. 调用cost方法, 并且使用委托来计算调味料的总价格

 

 

到目前我知道了这些:

  • 装饰器的父类和它所要装饰的对象的父类是一样的

  • 可以使用多个装饰器来装饰某对象

  • 既然装饰器和被装饰对象的父类是一样的, 那传递的时候就传递被装饰过的对象就好了.

  • 装饰器会在委托给它要装饰的对象之前和/或之后添加自己的行为以便来完成余下的工作.

  • 对象可以在任意时刻被装饰, 所以可以在运行时使用任意多个装饰器对对象进行装饰.

装饰模式定义

 动态的对某个对象进行扩展(附加额外的职责), 装饰器是除了继承之外的另外一种为对象扩展功能的方法.

下面看看该模式的类图:

重新设计

这个就很好理解了, 父类都是Beverage(饮料), 左边是四种具体实现的咖啡, 右边上面是装饰器的父类, 下面是具体的装饰器(调味料).

这里需要注意的是, 装饰器和咖啡都继承于同一个父类只是因为需要它们的类型匹配而已, 并不是要继承行为.

.NET Core 代码实现

Beverage:

namespace DecoratorPattern.Core

{

    public abstract class Beverage 

    {

        public virtual string Description { get; protected set; } = "Unknown Beverage";

        

        public abstract double Cost();

    }

}

CondimentDecorator:

namespace DecoratorPattern.Core

{

    public abstract class CondimentDecorator : Beverage

    {

        public abstract override string Description { get; }

    }

}

Espresso 浓咖啡:

using DecoratorPattern.Core;


namespace DecoratorPattern.Coffee

{

    public class Espresso : Beverage

    {

        public Espresso()

        {

            Description = "Espresso";

        }

        public override double Cost()

        {

            return 1.99;

        }

    }

}

HouseBlend:

using DecoratorPattern.Core;


namespace DecoratorPattern.Coffee

{

    public class HouseBlend : Beverage

    {

        public HouseBlend()

        {

            Description = "HouseBlend";

        }


        public override double Cost()

        {

            return .89;

        }

    }

}

Mocha:


using DecoratorPattern.Core;


namespace DecoratorPattern.Condiments

{

    public class Mocha : CondimentDecorator

    {

        private readonly Beverage beverage;


        public Mocha(Beverage beverage) => this.beverage = beverage;


        public override string Description => $"{beverage.Description}, Mocha";


        public override double Cost()

        {

            return .20 + beverage.Cost();

        }

    }

}

Whip:

using DecoratorPattern.Core;


namespace DecoratorPattern.Condiments

{

    public class Whip : CondimentDecorator

    {

        private readonly Beverage beverage;


        public Whip(Beverage beverage) => this.beverage = beverage;


        public override string Description => $"{beverage.Description}, Whip";


        public override double Cost()

        {

            return .15 + beverage.Cost();

        }

    }

}

Program:

using System;

using DecoratorPattern.Coffee;

using DecoratorPattern.Condiments;

using DecoratorPattern.Core;


namespace DecoratorPattern

{

    class Program

    {

        static void Main(string[] args)

        {

            var beverage = new Espresso();

            Console.WriteLine($"{beverage.Description} $ {beverage.Cost()}");


            Beverage beverage2 = new HouseBlend();

            beverage2 = new Mocha(beverage2);

            beverage2 = new Mocha(beverage2);

            beverage2 = new Whip(beverage2);

            Console.WriteLine($"{beverage2.Description} $ {beverage2.Cost()}");

        

        }

    }

}


运行结果:

 

.NET Core里面的装饰模式例子: Streams 和 I/O

首先需要知道, System.IO命名空间是低级I/O功能的大本营.

Stream的结构

.NET Core里面的Stream主要是三个概念: 存储(backing stores 我不知道怎么翻译比较好), 装饰器, 适配器.

backing stores是让输入和输出发挥作用的端点, 例如文件或者网络连接. 就是下面任意一点或两点:

  • 一个源, 从它这里字节可以被顺序的读取

  • 一个目的地, 字节可以被连续的写入.

程序员可以通过Stream类来发挥backing store的作用. Stream类有一套方法, 可以进行读取, 写入, 定位等操作. 个数组不同的是, 数组是把所有的数据都一同放在了内存里, 而stream则是顺序的/连续的处理数据, 要么是一次处理一个字节, 要么是一次处理特定大小(不能太大, 可管理的范围内)的数据.

于是, stream可以用比较小的固定大小的内存来处理无论多大的backing store.

中间的那部分就是装饰器Stream. 它符合装饰模式.

从图中可以看到, Stream又分为两部分:

  • Backing Store Streams: 硬连接到特定类型的backing store, 例如FileStream和NetworkStream

  • Decorator Streams 装饰器Stream: 使用某种方式把数据进行了转化, 例如DeflateStream和CryptoStream.

装饰器Stream有如下结构性的优点(参考装饰模式):

  • 无需让backing store stream去实现例如压缩, 加密等功能.

  • 装饰的时候接口(interface)并没有变化

  • 可以在运行时进行装饰

  • 可以串联装饰(先后进行多个装饰)

backing store和装饰器stream都是按字节进行处理的. 尽管这很灵活和高效, 但是程序一般还是采用更高级别的处理方式例如文字或者xml.

适配器通过使用特殊化的方法把类里面的stream进行包装成特殊的格式. 这就弥合了上述的间隔.

例如 text reader有一个ReadLine方法, XML writer又WriteAttributes方法.

注意: 适配器包装了stream, 这点和装饰器一样, 但是不一样的是, 适配器本身并不是stream, 它一般会把所有针对字节的方法都隐藏起来. 所以本文就不介绍适配器了.

总结一下:

backing store stream 提供原始数据, 装饰器stream提供透明的转换(例如加密); 适配器提供方法来处理高级别的类型例如字符串和xml.

想要连成串的话, 秩序把对象传递到另一个对象的构造函数里.

使用Stream

Stream抽象类是所有Stream的基类.

它的方法和属性主要分三类基本操作: 读, 写, 寻址(Seek); 和管理操作: 关闭(close), 冲(flush)和设定超时:

这些方法都有异步的版本, 加async, 返回Task即可.

一个例子:

using System;

using System.IO;


namespace Test

{

    class Program

    {

        static void Main(string[] args)

        {

            // 在当前目录创建按一个 test.txt 文件

            using (Stream s = new FileStream("test.txt", FileMode.Create))

            {

                Console.WriteLine(s.CanRead); // True

                Console.WriteLine(s.CanWrite); // True

                Console.WriteLine(s.CanSeek); // True

                s.WriteByte(101);

                s.WriteByte(102);

                byte[] block = { 1, 2, 3, 4, 5 };

                s.Write(block, 0, block.Length); // 写 5 字节

                Console.WriteLine(s.Length); // 7

                Console.WriteLine(s.Position); // 7

                s.Position = 0; // 回到开头位置

                Console.WriteLine(s.ReadByte()); // 101

                Console.WriteLine(s.ReadByte()); // 102

                                                 // 从block数组开始的地方开始read:

                Console.WriteLine(s.Read(block, 0, block.Length)); // 5

                                                                   // 假设最后一次read返回 5, 那就是在文件结尾, 所以read会返回0:

                Console.WriteLine(s.Read(block, 0, block.Length)); // 0

            }

        }

    }

}

运行结果:

 

异步例子:

using System;

using System.IO;

using System.Threading.Tasks;


namespace Test

{

    class Program

    {

        static void Main(string[] args)

        {

            Task.Run(AsyncDemo).GetAwaiter().GetResult();

        }


        async static Task AsyncDemo()

        {

            using (Stream s = new FileStream("test.txt", FileMode.Create))

            {

                byte[] block = { 1, 2, 3, 4, 5 };

                await s.WriteAsync(block, 0, block.Length); 

                s.Position = 0;

                Console.WriteLine(await s.ReadAsync(block, 0, block.Length));

            }

        }

    }

}

异步版本比较适合慢的stream, 例如网络的stream.

读和写

CanRead和CanWrite属性可以判断Stream是否可以读写.

Read方法把stream的一块数据写入到数组, 返回接受到的字节数, 它总是小于等于count这个参数. 如果它小于count, 就说明要么是已经读取到stream的结尾了, 要么stream给的数据块太小了(网络stream经常这样).

一个读取1000字节stream的例子:

           // 假设s是某个stream

            byte[] data = new byte[1000];

            // bytesRead 的结束位置肯定是1000, 除非stream的长度不足1000

            int bytesRead = 0;

            int chunkSize = 1;

            while (bytesRead < data.Length && chunkSize > 0)

                bytesRead +=

                   chunkSize = s.Read(data, bytesRead, data.Length - bytesRead);

ReadByte方法更简单一些, 一次就读一个字节, 如果返回-1表示读取到stream的结尾了. 返回类型是int.

Write和WriteByte就是相应的写入方法了. 如果无法写入某个字节, 那就会抛出异常.

上面方法签名里的offset参数, 表示的是缓冲数组开始读取或写入的位置, 而不是指stream里面的位置.

寻址 Seek

CanSeek为true的话, Stream就可以被寻址. 可以查询和修改可寻址的stream(例如文件stream)的长度, 也可以随时修改读取和写入的位置.

Position属性就是所需要的, 它是相对于stream开始位置的.

Seek方法就允许你移动到当前位置或者stream的尾部.

注意改变FileStream的Position会花去几微秒. 如果是在大规模循环里面做这个操作的话, 建议使用MemoryMappedFile类.

对于不可寻址的Stream(例如加密Stream), 想知道它的长度只能是把它读完. 而且你要是想读取前一部分的话必须关闭stream, 然后再开始一个全新的stream才可以.

关闭和Flush

Stream用完之后必须被处理掉(dispose)来释放底层资源例如文件和socket处理. 通常使用using来实现.

  • Dispose和Close方法功能上是一样的.

  • 重复close和flush一个stream不会报错.

关闭装饰器stream的时候会同时关闭装饰器和它的backing store stream.

针对一连串的装饰器装饰的stream, 关闭最外层的装饰器就会关闭所有.

有些stream从backing store读取/写入的时候有一个缓存机制, 这就减少了实际到backing store的往返次数以达到提高性能的目的(例如FileStream).

这就意味着你写入数据到stream的时候可能不会立即写入到backing store; 它会有延迟, 直到缓冲被填满.

Flush方法会强制内部缓冲的数据被立即的写入. Flush会在stream关闭的时候自动被调用. 所以你不需要这样写: s.Flush(); s.Close();

超时

如果CanTimeout属性为true的话, 那么该stream就可以设定读或写的超时.

网络stream支持超时, 而文件和内存stream则不支持.

支持超时的stream, 通过ReadTimeout和WriteTimeout属性可以设定超时, 单位毫秒. 0表示无超时.

Read和Write方法通过抛出异常的方式来表示超时已经发生了.

线程安全

stream并不是线程安全的, 也就是说两个线程同时读或写一个stream的时候就会报错.

Stream通过Synchronized方法来解决这个问题. 该方法接受stream为参数, 返回一个线程安全的包装结果.

这个包装结果在每次读, 写, 寻址的时候会获得一个独立锁/排他锁, 所以同一时刻只有一个线程可以执行操作.

实际上, 这允许多个线程同时为同一个数据追加数据, 而其他类型的操作(例如同读)则需要额外的锁来保证每个线程可以访问到stream相应的部分.

Backing Store Stream

FileStream

文件流

构建一个FileStream

FileStream fs1 = File.OpenRead("readme.bin"); // Read-onlyFileStream fs2 = File.OpenWrite(@"c:\temp\writeme.tmp"); // Write-onlyFileStream fs3 = File.Create(@"c:\temp\writeme.tmp"); // Read/write

 

OpenWrite和Create对于已经存在的文件来说, 它的行为是不同的.

Create会把现有文件的内容清理掉, 写入的时候从头开写.

OpenWrite则是完整的保存着现有的内容, 而stream的位置定位在0. 如果写入的内容比原来的内容少, 那么OpenWrite打开并写完之后的内容是原内容和新写入内容的混合体.

直接构建FileStream:

var fs = new FileStream ("readwrite.tmp", FileMode.Open); // Read/write

 

其构造函数里面还可以传入其他参数, 具体请看文档.

File类的快捷方法:

下面这些静态方法会一次性把整个文件读进内存:

  • File.ReadAllText(返回string)

  • File.ReadAllLines(返回string数组) 

  • File.ReadAllBytes(返回byte数组)

下面的方法直接写入整个文件:

  • File.WriteAllText

  • File.WriteAllLines

  • File.WriteAllBytes

  • File.AppendAllText (很适合附加log文件) 

还有一个静态方法叫File.ReadLines: 它有点想ReadAllLines, 但是它返回的是一个懒加载的IEnumerable<string>. 这个实际上效率更高一些, 因为不必一次性把整个文件都加载到内存里. LINQ非常适合处理这个结果. 例如:

int longLines = File.ReadLines ("filePath").Count (l => l.Length > 80);

 

指定的文件名:

可以是绝对路径也可以是相对路径.

可已修改静态属性Environment.CurrentDirectory的值来改变当前的路径. (注意: 默认的当前路径不一定是exe所在的目录)

AppDomain.CurrentDomain.BaseDirectory会返回应用的基目录, 它通常是包含exe的目录. 

指定相对于这个目录的地址最好使用Path.Combine方法:

            string baseFolder = AppDomain.CurrentDomain.BaseDirectory;            string logoPath = Path.Combine(baseFolder, "logo.jpg");Console.WriteLine(File.Exists(logoPath));

 

通过网络对文件读写要使用UNC路径:

例如: \\JoesPC\PicShare \pic.jpg 或者 \\10.1.1.2\PicShare\pic.jpg.

FileMode:

所有的FileStream的构造器都会接收一个文件名和一个FileMode枚举作为参数. 如果选择FileMode请看下图:

其他特性还是需要看文档.

MemoryStream

MemoryStream在随机访问不可寻址的stream时就有用了.

如果你知道源stream的大小可以接受, 你就可以直接把它复制到MemoryStream里:

            var ms = new MemoryStream();sourceStream.CopyTo(ms);

 

可以通过ToArray方法把MemoryStream转化成数组.

GetBuffer方法也是同样的功能, 但是因为它是直接把底层的存储数组的引用直接返回了, 所以会更有效率. 不过不幸的是, 这个数组通常比stream的真实长度要长.

注意: Close和Flush 一个MemoryStream是可选的. 如果关闭了MemoryStream, 你就再也不能对它读写了, 但是仍然可以调用ToArray方法来获取其底层的数据.

Flush则对MemoryStream毫无用处.

PipeStream

PipeStream通过Windows Pipe 协议, 允许一个进程(process)和另一个进程通信.

分两种:

  • 匿名进程(快一点), 允许同一个电脑内的父子进程单向通信.

  • 命名进程(更灵活), 允许同一个电脑内或者同一个windows网络内的不同电脑间的任意两个进程间进行双向通信

pipe很适合一个电脑上的进程间交互(IPC), 它并不依赖于网络传输, 这也意味着没有网络开销, 也不在乎防火墙.

注意: pipe是基于Stream的, 一个进程等待接受一串字符的同时另一个进程发送它们.

PipeStream是抽象类.

具体的实现类有4个:

匿名pipe:

  • AnonymousePipeServerStream

  • AnonymousePipeClientStream

命名Pipe:

  • NamedPipeServerStream

  • NamePipeClientStream

命名Pipe

命名pipe的双方通过同名的pipe进行通信. 协议规定了两个角色: 服务器和客户端. 按照下述方式进行通信:

  • 服务器实例化一个NamedPipeServerStream然后调用WaitForConnection方法.

  • 客户端实例化一个NamedPipeClientStream然后调用Connect方法(可以设定超时).

然后双方就可以读写stream来进行通信了.

例子:

using System;

using System.IO;

using System.IO.Pipes;

using System.Threading.Tasks;


namespace Test

{

    class Program

    {

        static void Main(string[] args)

        {

            Console.WriteLine(DateTime.Now.ToString());

            using (var s = new NamedPipeServerStream("pipedream"))

            {

                s.WaitForConnection();

                s.WriteByte(100); // Send the value 100.

                Console.WriteLine(s.ReadByte());

            }

            Console.WriteLine(DateTime.Now.ToString());

        }

    }

}

using System;

using System.IO.Pipes;


namespace Test2

{

    class Program

    {

        static void Main(string[] args)

        {

            Console.WriteLine(DateTime.Now.ToString());

            using (var s = new NamedPipeClientStream("pipedream"))

            {

                s.Connect();

                Console.WriteLine(s.ReadByte());

                s.WriteByte(200); // Send the value 200 back.

            }

            Console.WriteLine(DateTime.Now.ToString());

        }

    }

}

命名的PipeStream默认情况下是双向的, 所以任意一方都可以进行读写操作, 这也意味着服务器和客户端必须达成某种协议来协调它们的操作, 避免同时进行发送和接收.

还需要协定好每次传输的长度.

在处理长度大于一字节的信息的时候, pipe提供了一个信息传输的模式, 如果这个启用了, 一方在调用read的时候可以通过检查IsMessageComplete属性来知道消息什么时候结束.

例子:

static byte[] ReadMessage(PipeStream s)

        {

            MemoryStream ms = new MemoryStream();

            byte[] buffer = new byte[0x1000]; // Read in 4 KB blocks

            do { ms.Write(buffer, 0, s.Read(buffer, 0, buffer.Length)); }

            while (!s.IsMessageComplete); return ms.ToArray();

        }

注意: 针对PipeStream不可以通过Read返回值是0的方式来它是否已经完成读取消息了. 这是因为它和其他的Stream不同, pipe stream和network stream没有确定的终点. 在两个信息传送动作之间, 它们就干等着.

这样启用信息传输模式, 服务器端 :

using (var s = new NamedPipeServerStream("pipedream", PipeDirection.InOut, 1, PipeTransmissionMode.Message))

            {

                s.WaitForConnection();

                byte[] msg = Encoding.UTF8.GetBytes("Hello");

                s.Write(msg, 0, msg.Length);

                Console.WriteLine(Encoding.UTF8.GetString(ReadMessage(s)));

            }

客户端: 

using (var s = new NamedPipeClientStream("pipedream"))

            {

                s.Connect();

                s.ReadMode = PipeTransmissionMode.Message;

                Console.WriteLine(Encoding.UTF8.GetString(ReadMessage(s)));

                byte[] msg = Encoding.UTF8.GetBytes("Hello right back!");

                s.Write(msg, 0, msg.Length);

            }

匿名pipe:

匿名pipe提供父子进程间的单向通信. 流程如下:

  • 服务器实例化一个AnonymousPipeServerStream, 并指定PipeDirection是In还是Out

  • 服务器调用GetClientHandleAsString方法来获取一个pipe的标识, 然后会把它传递给客户端(通常是启动子进程的参数 argument)

  • 子进程实例化一个AnonymousePipeClientStream, 指定相反的PipeDirection

  • 服务器通过调用DisposeLocalCopyOfClientHandle释放步骤2的本地处理, 

  • 父子进程间通过读写stream进行通信

因为匿名pipe是单向的, 所以服务器必须创建两份pipe来进行双向通信

例子:

server:

using System;

using System.Diagnostics;

using System.IO;

using System.IO.Pipes;

using System.Text;

using System.Threading.Tasks;


namespace Test

{

    class Program

    {

        static void Main(string[] args)

        {

            string clientExe = @"D:\Projects\Test2\bin\Debug\netcoreapp2.0\win10-x64\publish\Test2.exe";

            HandleInheritability inherit = HandleInheritability.Inheritable;

            using (var tx = new AnonymousPipeServerStream(PipeDirection.Out, inherit))

            using (var rx = new AnonymousPipeServerStream(PipeDirection.In, inherit))

            {

                string txID = tx.GetClientHandleAsString();

                string rxID = rx.GetClientHandleAsString();

                var startInfo = new ProcessStartInfo(clientExe, txID + " " + rxID);

                startInfo.UseShellExecute = false; // Required for child process

                Process p = Process.Start(startInfo);

                tx.DisposeLocalCopyOfClientHandle(); // Release unmanaged

                rx.DisposeLocalCopyOfClientHandle(); // handle resources.

                tx.WriteByte(100);

                Console.WriteLine("Server received: " + rx.ReadByte());

                p.WaitForExit();

            }

        }

    }

}

client:

using System;

using System.IO.Pipes;


namespace Test2

{

    class Program

    {

        static void Main(string[] args)

        {

            string rxID = args[0]; // Note we're reversing the

            string txID = args[1]; // receive and transmit roles.

            using (var rx = new AnonymousPipeClientStream(PipeDirection.In, rxID))

            using (var tx = new AnonymousPipeClientStream(PipeDirection.Out, txID))

            {

                Console.WriteLine("Client received: " + rx.ReadByte());

                tx.WriteByte(200);

            }

        }

    }

}

最好发布一下client成为独立运行的exe:

dotnet publish --self-contained --runtime win10-x64

 

运行结果:

 

匿名pipe不支持消息模式, 所以你必须自己来为传输的长度制定协议. 有一种做法是: 在每次传输的前4个字节里存放一个整数表示消息的长度, 可以使用BitConverter类来对整型和长度为4的字节数组进行转换.

BufferedStream

BufferedStream对另一个stream进行装饰或者说包装, 让它拥有缓冲的能力.它也是众多装饰stream类型中的一个.

缓冲肯定会通过减少往返backing store的次数来提升性能.

下面这个例子是把一个FileStream装饰成20k的缓冲stream:

           // Write 100K to a file:

            File.WriteAllBytes("myFile.bin", new byte[100000]);

            using (FileStream fs = File.OpenRead("myFile.bin"))

            using (BufferedStream bs = new BufferedStream(fs, 20000)) //20K buffer

            {

                bs.ReadByte();

                Console.WriteLine(fs.Position); // 20000

            }

        }

通过预读缓冲, 底层的stream会在读取1字节后, 直接预读了20000字节, 这样我们在另外调用ReadByte 19999次之后, 才会再次访问到FileStream.

这个例子是把BufferedStream和FileStream耦合到一起, 实际上这个例子里面的缓冲作用有限, 因为FileStream有一个内置的缓冲. 这个例子也只能扩大一下缓冲而已.

关闭BufferedStream就会关闭底层的backing store stream.. 

先写到这里, 略微有点跑题了, 但是.NET Core的Stream这部分没写完, 另开一篇文章再写吧.


原文地址 https://www.cnblogs.com/cgzl/p/8697949.html


.NET社区新闻,深度好文,欢迎访问公众号文章汇总 http://www.csharpkit.com

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/321769.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

这个拖后腿的“in”

问题之源C# 7.2推出了全新的参数修饰符in&#xff0c;据说是能提升一定的性能&#xff0c;官方MSDN文档描述是&#xff1a;Add the in modifier to pass an argument by reference and declare your design intent to pass arguments by reference to avoid unnecessary copyin…

Surging 微服务框架使用入门

前言本文非 Surging 官方教程&#xff0c;只是自己学习的总结。如有哪里不对&#xff0c;还望指正。 我对 surging 的看法我目前所在的公司采用架构就是类似与Surging的RPC框架&#xff0c;在.NET 4.0框架上搭建Socket RPC&#xff0c;通过分组轮询的方式调度RPC&#xff0c;经…

RabbitMQ教程C#版 - 工作队列

先决条件本教程假定RabbitMQ已经安装&#xff0c;并运行在localhost标准端口&#xff08;5672&#xff09;。如果你使用不同的主机、端口或证书&#xff0c;则需要调整连接设置。从哪里获得帮助如果您在阅读本教程时遇到困难&#xff0c;可以通过邮件列表联系我们。1.工作队列&…

IdentityServer4实战 - 基于角色的权限控制及Claim详解

一.前言大家好&#xff0c;许久没有更新博客了&#xff0c;最近从重庆来到了成都&#xff0c;换了个工作环境&#xff0c;前面都比较忙没有什么时间&#xff0c;这次趁着清明假期有时间&#xff0c;又可以分享一些知识给大家。在QQ群里有许多人都问过IdentityServer4怎么用Role…

11、java中的I/O流(1)

我对于流的理解是这样的&#xff0c;计算机的本质本来就是对输入的数据进行操作&#xff0c;然后将结果输出的一种工具&#xff0c;数据在各个数据源节点之间进行流动&#xff0c;感觉流就是对这种状态的一种抽象&#xff0c;一个数据流表示的就是一系列数据序列&#xff0c;ja…

ASP.NET Core 集成测试

集成测试集成测试&#xff0c;也叫组装测试或联合测试。在单元测试的基础上&#xff0c;将所有模块按照设计要求&#xff08;如根据结构图&#xff09;组装成为子系统或系统&#xff0c;进行集成测试。实践表明&#xff0c;一些模块虽然能够单独地工作&#xff0c;但并不能保证…

使用C#开发Android应用之WebApp

近段时间了解了一下VS2017开发安卓应用的一些技术&#xff0c;特地把C#开发WebApp的一些过程记录下来&#xff0c;欢迎大家一起指教、讨论&#xff0c;废话少说&#xff0c;是时候开始表演真正的技术了。。1、新建空白Android应用2、拖一个WebView控件进来3、打开模拟器Genymot…

ASP.NET Core依赖注入深入讨论

这篇文章我们来深入探讨ASP.NET Core、MVC Core中的依赖注入&#xff0c;我们将示范几乎所有可能的操作把依赖项注入到组件中。依赖注入是ASP.NET Core的核心&#xff0c;它能让您应用程序中的组件增强可测试性&#xff0c;还使您的组件只依赖于能够提供所需服务的某些组件。举…

使用静态基类方案让 ASP.NET Core 实现遵循 HATEOAS Restful Web API

Hypermedia As The Engine Of Application State (HATEOAS)HATEOAS&#xff08;Hypermedia as the engine of application state&#xff09;是 REST 架构风格中最复杂的约束&#xff0c;也是构建成熟 REST 服务的核心。它的重要性在于打破了客户端和服务器之间严格的契约&…

【招聘(北京)】.NETCORE开发工程师(微服务方向)

组织&#xff1a;华汽集团北京研发中心位置&#xff1a;北京市朝阳区焦奥中心官网&#xff1a;www.sinoauto.com邮箱&#xff1a;taoxu.weisinoauto.com 项目&#xff1a;打造面向国内汽车后市场用户的一站式云服务平台&#xff08;华汽云&#xff09;&#xff0c;形态包括B2B、…

确保线程安全下使用Queue的Enqueue和Dequeue

场景是这样&#xff0c;假设有一台设备会触发类型为Alarm的告警信号&#xff0c;并把信号添加到一个Queue结构中&#xff0c;每隔一段时间这个Queue会被遍历检查&#xff0c;其中的每个Alarm都会调用一个相应的处理方法。问题在于&#xff0c;检查机制是基于多线程的&#xff0…

编写一个Java程序,其中包含三个线程: 厨师(Chef)、服务员(Waiter)和顾客(Customer)

编写一个Java程序&#xff0c;其中包含三个线程: 厨师(Chef)、服务员(Waiter)和顾客(Customer)。他们的行动如下: 厨师准备菜肴&#xff0c;每次准备一个。服务员等待菜肴准备好&#xff0c;然后将其送到顾客那里。顾客等待服务员送来菜看后才开始吃。所有三个角色应该循环进行…

Hangfire使用ApplicationInsigts监控

起因我司目前使用清真的ApplicationInsights来做程序级监控。&#xff08;ApplicationInsights相关文档: https://azure.microsoft.com/zh-cn/services/application-insights/ &#xff09;其实一切都蛮好的&#xff0c;但是我们基于Hangfire的Job系统却无法被Ai所监控到&#…

NET主流ORM框架分析

接上文我们测试了各个ORM框架的性能&#xff0c;大家可以很直观的看到各个ORM框架与原生的ADO.NET在境删改查的性能差异。这里和大家分享下我对ORM框架的理解及一些使用经验。ORM框架工作原理所有的ORM框架的工作原理都离不开下面这张图&#xff0c;只是每个框架的实现程度不同…

20、java中的类加载机制

1、类加载机制是什么&#xff1f; 类加载机制指的就是jvm将类的信息动态添加到内存并使用的一种机制。 2、那么类加载的具体流程是什么呢&#xff1f; 一般说类加载只有三步&#xff1a;加载、连接和初始化&#xff0c;其中连接包括验证、准备和解析&#xff0c;用于将运行时加…

【北京】BXUG第12期活动基于 .NET Core构建微服务和Xamarin

分享主题&#xff1a;基于 .NET Core构建微服务实战分享分享者&#xff1a;薛锋 北京切尔思科技架构师 兼任东北大学信息安全工程师和技术主播&#xff0c;行业内专注于研究 .NET Core和Web应用&#xff0c;具有比较扎实的技术基础和数年的从业经历。在GitHub上主持数个开…

谈谈ASP.NET Core中的ResponseCaching

前言前面的博客谈的大多数都是针对数据的缓存&#xff0c;今天我们来换换口味。来谈谈在ASP.NET Core中的ResponseCaching&#xff0c;与ResponseCaching关联密切的也就是常说的HTTP缓存。在阅读本文内容之前&#xff0c;默认各位有HTTP缓存相关的基础&#xff0c;主要是Cache-…

使用 dynamic 类型让 ASP.NET Core 实现 HATEOAS 结构的 RESTful API

上一篇写的是使用静态基类方法的实现步骤: 使用dynamic (ExpandoObject)的好处就是可以动态组建返回类型, 之前使用的是ViewModel, 如果想返回结果的话, 肯定需要把ViewModel所有的属性都返回, 如果属性比较多, 就有可能造成性能和灵活性等问题. 而使用ExpandoObject(dynamic)就…

使用 BenchmarkDotnet 测试代码性能

先来点题外话&#xff0c;清明节前把工作辞了&#xff08;去 tm 的垃圾团队&#xff0c;各种拉帮结派、勾心斗角&#xff09;。这次找工作就得慢慢找了&#xff0c;不能急了&#xff0c;希望能找到个好团队&#xff0c;好岗位吧。顺便这段时间也算是比较闲&#xff0c;也能学习…

2017西安交大ACM小学期数论 [阅兵式]

阅兵式 发布时间: 2017年6月25日 12:53 最后更新: 2017年7月3日 09:27 时间限制: 1000ms 内存限制: 128M 描述 阅兵式上&#xff0c;将士们排成一个整齐的方阵&#xff0c;每个将士面朝前方。问正中心的将士能向前看到几个将士&#xff1f;注意&#xff0c;一条直线上的将…