using System.Threading
using System.IO
namespace CopyTest
{
//FileBuffer用来存放和取出缓冲区变量
public class FileBuffer
{
private int m_readsize = 1024
//定义了m_capacity个字节的缓冲区
private byte[] m_buffer = new byte[4096]
//确认缓冲区内已放字节的个数
private int bufferCount = 0
//确定读写的位置
private int readLocation = 0, writeLocation = 0
public FileBuffer()
{
}
//从缓冲区中取数据
public byte[] getBuffer()
{
//加上了共享锁
lock (this)
{
//判断如果缓冲区内无内容,则读取者进入wait状态,并且释放对象锁
if (bufferCount == 0)
{
Console.WriteLine("缓冲区无数据,无法读取")
Monitor.Wait(this)
}
byte[] newBuf = new byte[m_readsize]
Buffer.BlockCopy(m_buffer, readLocation, newBuf, 0, m_readsize)
//已经从缓冲区读取了内容,所以bufferCount要进行自减.
bufferCount -= m_readsize
//求余的目的是为了循环使用缓冲区
readLocation = (readLocation + m_readsize) % m_buffer.Length
//通知对象的第一个等待线程可以从WaitSleepJoin转换到Started状态.
Monitor.Pulse(this)
//返回给读取者取出的数值
return newBuf
}
}
//将数据放入缓冲区
public void setBuffer(byte[] writeValue)
{
//锁住共享数据区
lock (this)
{
//如果缓冲区已满,那么进入waitsleepjoin状态
if (bufferCount == m_buffer.Length)
{
Console.WriteLine("缓冲区溢出!")
Monitor.Wait(this)
}
//向缓冲区写入数据
Buffer.BlockCopy(writeValue, 0, m_buffer, writeLocation, m_readsize)
//自加,代表缓冲区现在到底有几个数据
bufferCount += m_readsize
//用%实现缓冲区的循环利用
writeLocation = (writeLocation + m_readsize) % m_buffer.Length
//唤醒waitSleepJoin状态的进程,到started状态
Monitor.Pulse(this)
}//使用lock隐式的释放了共享锁
}
}
//写入者类,向缓冲区中放入数据
public class Writer
{
//定义了同步变量
FileBuffer shared
FileStream file
//此处构造函数的作用是在启动类中调用写入者的时候,把启动类中定义的sharedLocation传过来
public Writer(FileBuffer sharedLocation)
{
file = new FileStream("C:\\Test.txt", FileMode.Open)
shared = sharedLocation
}
//定义写入过程
public void Write()
{
//将数据放入缓冲区
Byte[] datas = new byte[1024]
for (int byteread = 0byteread <= file.Lengthbyteread += datas.Length)
{
file.Read(datas, byteread, datas.Length)
shared.setBuffer(datas)
}
file.Close()
//得到当前线程的名字
string name = Thread.CurrentThread.Name
//此线程执行完毕
Console.WriteLine(name + "done writeing")
}
}
public class Reader
{//定义读取者
byte[] value
FileStream file
//定义同步变量
FileBuffer shared
//定义构造函数,负责传递启动类中的shared
public Reader(FileBuffer sharedLocation)
{
file = new FileStream("C:\\Data.txt", FileMode.Create)
shared = sharedLocation
}
public void Read()
{
//从缓冲区中循环读取
for (int bytewrite = 0bytewrite <= 65535)
{
value = shared.getBuffer()
file.Write(value, bytewrite, value.Length)
bytewrite += value.Length
}
file.Close()
//取得当前线程的名字
string name = Thread.CurrentThread.Name
Console.WriteLine(name + "done reading")
}
}
public class ThreadTest
{ //设置为启动类
public static void Main()
{
FileBuffer shared = new FileBuffer()
//初始化了写入者和读取者,并且把shared参数传递了过去
Writer Writer1 = new Writer(shared)
Reader Reader1 = new Reader(shared)
Thread WriterThread = new Thread(new ThreadStart(Writer1.Write))
WriterThread.Name = "写入者"
Thread ReaderThread = new Thread(new ThreadStart(Reader1.Read))
ReaderThread.Name = "读取者"
//启动这两个线程
WriterThread.Start()
ReaderThread.Start()
WriterThread.Join()
ReaderThread.Join()
Console.ReadLine()
}
}
}
多线程就是多个指令序列同时执行,能同时占有多个CPU核心,提高整体运算速度。不过不建议用多线程复制,多个线程同时访问同一个物理设备,同时只有一个线程能获得这个设备的控制权,其他线程都在等待。因此文件复制的瓶颈在disk i/o,不在CPU,多线程不能提高disk i/o的性能,反而有可能增加磁盘的寻道时间。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)