多线程

线程、进程和应用程序域

进程:进程是一个操作系统上的概念,用来实现多任务并发执行,是资源分配的最小单元,各个进程是相互独立的,可以理解为执行当中的程序,在操作系统中一般用一个称为PCB的结构体表示,里面存放了一些线程共用的、进程独立的数据;

应用程序域:是一个程序运行的逻辑区域,一个进程可以有多个应用程序域,一个应用程序域可以有多个线程,任一时刻一个线程只能运行在一个应用程序域中;

线程:进程因为包含了太多的数据,在做任务切换的时候非常消耗系统资源,所以就产生了线程,线程是操作系统进行任务调度的最少单元,是进程的子内容,一个进程可以有多个线程,各个线程之间共享进程里面的数据,线程主要是由CPU寄存器、调用栈和线程本地存储器(Thread Local Storage,TLS)组成的;

线程基础知识

  1. 线程的状态

线程在不同的时刻有不同的线程状态,ThreadState枚举用来标识一个线程的状态

  1. 线程的优先级

前面说线程是操作系统用来进行任务调度的最小单元,所以线程就必须有一些信息提供给操作系统,这些信息是操作系统进行任务调度的依据,一般会为线程设置优先级别,用来保证程序当中比较紧急和重要的任务能优先得到执行,ThreadPriority枚举用来设置线程的优先级别;

  1. 前后台线程

Thread有一个属性IsBackground,通过把此属性设置为true,就可以把线程设置为后台线程!这时应用程序域将在主线程完成时就被卸载,而不会等待异步线程的运行。一般将那些当主程序关闭后随之关闭的线程设置为后台线程,否则设置为前台线程;

.NET中实现多线程的几种方案

Thread类实现多线程

  1. 提供一个方法,该方法将作为线程的执行体

Public void ThreadTestMethod(object obj)

{

//方法体

}

  1. 创建Thread类实例

Thread th=new Thread(ThreadTestMethod);    //该构造方法有四个形式的重载

  1. 两个委托

public delegate void ParameterizedThreadStart(object obj);

public delegate void ThreadStart();

ParameterizedThreadStart委托接受一个object类型的参数,这样我们就可以为线程执行的方法传递参数了,而ThreadStart委托不接受参数;

  1. 启动线程

Th.Start(null);   如果在实例化Thread类的实例的时候传递的是一个ParameterizedThreadStart类型的委托,则可以在启动一个线程的时候为线程的执行传递相关的参数,而如果是ThreadStart类型的委托则不能传递参数;

线程池ThreadPool实现多线程

向线程池中注册一个线程:

public static bool QueueUserWorkItem(WaitCallback callBack, object state);

WaitCallback委托:

public delegate void WaitCallback(object state);

state参数可以为线程传递参数;

简单的例子:

static void TestThreadMethod(object state)

{

while (true)

{

Thread.Sleep(1000);

Console.WriteLine(Thread.CurrentThread.ManagedThreadId); Console.WriteLine(state.ToString());

}

}

bool isSuccess= ThreadPool.QueueUserWorkItem(TestThreadMethod ,"A test paramter");

 

异步委托

方式一:等待直至完成

声明委托类型:

delegate void TestDelegate(string para);

创建线程执行代码(函数):

static void TestDelegateMethod(string para)

{

Thread.Sleep(10000);

Console.WriteLine(Thread.CurrentThread.ManagedThreadId);

Console.WriteLine(para);

}

定义委托变量:

TestDelegate del = new TestDelegate(TestDelegateMethod);

调用:

IAsyncResult asynResult = del.BeginInvoke("Test Parameter", null, null);

//do something...

del.EndInvoke(asynResult);

说明:这是一种使用委托创建多线程应用程序的简单方式,但往往并不是最合适的方式,因为EndInvoke方法会使得主线程等待异步线程的执行完毕,也可以在BeginInvoke和EndInvoke之间写一些代码,但这样显得非常不灵活,因为我们不知道开启的异步线程什么时候执行完毕,所以,我们希望有这样的一种方式去执行异步委托:当异步线程执行完毕后,能主动调用我们提供的方法,这就是回调方式;

方式二:回调方式

与等待直至完成方式相比,前面定义委托类型、定义委托变量和指定线程执行代码的过程都是一致的,唯一不同的是在掉用的时候需要给BeginInvoke传递后两个参数(第三个参数也可以不传)。

BeginInvoke的参数说明:

第一部分参数由委托类型的参数决定,这里的委托TestDelegate只包含一个string类型的参数,所以BeginInvoke函数的第一个参数就是一个string类型的;

第二个参数是一个AsyncCallback类型的委托,原型为:

public delegate void AsyncCallback(IAsyncResult ar);通过传递一个符合该签名的函数名,则可以在异步线程执行完毕之后主动调用该函数;

在回调函数中获得委托变量的两种方式:

方式一:

del.BeginInvoke("Test Parameter", new AsyncCallback(TestCallBackFunction), null);

static void TestCallBackFunction(IAsyncResult result)

{

Console.WriteLine("回到函数被执行了!");

TestDelegate del = (TestDelegate)result.AsyncState;

del.EndInvoke(result);

}

方式二:

del.BeginInvoke("Test Parameter", new AsyncCallback(TestCallBackFunction), del);    //将del传递给了第三个参数

static void TestCallBackFunction(IAsyncResult result)

{

Console.WriteLine("回到函数被执行了!");

TestDelegate del = (TestDelegate)(result as AsyncResult). AsyncDelegate;

del.EndInvoke(result);

}

这种方式将委托变量del作为附加参数传递到BeginInvoke函数中,最终它会被封装在一个AsyncResult类型的变量中,并且该类型实现了IAsyncResult接口,随后当异步线程执行完毕调用回调函数的时候会将该变量传递给回调函数中(即result),这样就可以通过AsyncDelegate拿到委托了,当然,也可以将委托变量的作用域设置为更广的范围,这样在回调函数中就可以直接拿到委托对象了;

为什么要拿到委托变量?

因为拿到了委托变量(del)才能调用EndInvoke方法,那么为什么要调用EndInvoke方法?因为EndInvoke方法可以拿到异步线程执行函数的返回值,所以这就是为什么EndInvoke的返回类型与委托的返回类型一致了。

System.Threading.Timer类实现多线程

Timer类提供了一种简单的执行异步操作的方式,一般的用法如下:

static void TestTimerMethod(object para)

{

Console.Write(para.ToString());

}

//传递的方法名参数实际会被动态创建一个委托变量

Timer t = new Timer(TestTimerMethod, "*", 1000, 1000);

委托原型:public delegate void TimerCallback(object state);

使用的时候只需要调用Timer类的一个适当的构造函数重载,然后线程池就会通过委托变量指向的方法创建一个线程到等待队列,而该线程什么时候开始执行由第三个参数决定,每次执行的间隔由第四个参数决定,上面的示例中TestTimerMethod方法会在1秒后开始执行,然后每隔1秒再执行一次,每次执行都会将一个“*”字符串传递进去打印出来,所以,第二个参数是用来为异步线程的执行传递参数用的;

多线程的安全问题

跨线程访问控件的两种处理方式

当开启一个线程执行异步操作的时候如果在该线程中访问了不是该线程创建的控件时,会出现一个错误:线程间操作无效: 从不是创建控件“textBox1”的线程访问它。 解决这个问题有两种处理方式。

1.在窗体的构造函数中:Control.CheckForIllegalCrossThreadCalls = false;

这种简单的处理方式使得跨线程访问检查被忽略,这就不能保证程序安全执行;

2.让创建控件的线程去访问自己的控件

if (textBox1.InvokeRequired)

{

this.Invoke(new Action<TextBox, string>((t, s) => t.Text = s), textBox1, text.ToString());

}

else

{

textBox1.Text = text.ToString();

}

InvokeRequired属性获取控件被访问的时候是否需要Invoke调用,并且这里是调用的Form的Invoke方法,同样也可以使用TextBox控件自身的Invoke方法实现,原理是一样的,Invoke方法有多个形式的重载,这里是常见的一种,第一个参数传递的是一个泛型的Action,在构造这个Action的时候传递进去了一个Lambda表达式,Lambda表达式的代码修改了textBox1控件的Text属性;Invoke方法后面的参数是一个可变参数,这里为前面的委托(Action)提供所需要的参数;

锁lock

多线程带来了编程中的许多好处,使得程序可以在一个时间段内处理多个事情(宏观上),但是也提高了编程的复杂程度,多线程会带来许多问题,而造成问题的根本原因在于线程是“同时”执行的,所以解决问题的方法就是让它们在执行一些特定操作的时候不要同时执行了(这些操作会带来错误才这样处理),.NET中提供了LOCK的机制。

通过一个简单的示例演示一下LOCK的概念:

  1. 新建一个窗体应用程序,在默认打开的窗体上面拖出两个按钮;
  2. 添加一个私有的成员变量count:

private int count = 0;

  1. 第一个按钮的点击事件

private void button1_Click(object sender, EventArgs e)

{

for (int i = 0; i < 100; i++)

{

Thread th = new Thread(MultiThreadTest);

th.Start();

}

}

在这个按钮的点击事件处理程序当中,启动了100个线程去执行MultiThreadTest指向的代码。

  1. MultiThreadTest函数

private void MultiThreadTest()

{

for (int i = 0; i < 100; i++)

{

Thread.Sleep(10);

count++;

}

}

这个函数中使成员变量count自增100,所以最终的count值应该是100*100=10000(其实并不是)

  1. 使用另一个按钮的事件处理程序显示当前count的值

private void button2_Click(object sender, EventArgs e)

{

MessageBox.Show(count.ToString());

}

上面说过100个线程都对count变量进行自增,所以理想的结果应该是count==10000,但是并不是这样,下面是对话框显示的最终线程执行完毕之后的结果:

"9996"

  1. 使用锁

.NET中实现线程锁很简单,只需要改变一行代码就可以得到正确的结果:

lock (this)

{

count++;

}

在访问count的时候这里其实做了两部操作:先读取出count的值,然后加1再赋值给count,问题就出现在这样的一个过程中,如果在读得count值的时候其它的线程已经修改了count的值,那么当前线程获得的就不是一个实际的、最新的值,也就是说本线程的赋值会覆盖掉这个过程中其它线程的++操作,于是我们加上了一个Lock语句;

显示的结果如下:

" 1000;"

  1. 锁的原理(自己的理解)

每一个引用类型的对象都有一个同步索引块,指示当前使用该对象的线程数,每个线程执行到Lock语句块的时候就会判断当前锁定项(这里是this,当前窗体对象)的同步索引块是否等于0(即没有线程在访问该变量),如果等于0则进入执行块,首先将同步索引块的索引加1,表示当前多了一个线程使用this,等lock块执行完成再将同步索引块中的索引值减1,使得其它线程能够继续访问,这样就相当于实现了一个排队机制,使得在适当的时候该串行执行的代码串行执行;

SOCKET网络编程

协议

1、 TCP/IP(Transmission Control Protocol/Internet Protocol)即传输控制协议/网间协议,是一个工业标准的协议集,它是为广域网(WANs)设计的。

2、UDP(User Data Protocol,用户数据报协议)是与TCP相对应的协议。它是属于TCP/IP协议族中的一种。

Socket概念

socket称作“套接字”,用于描述IP地址和端口,是一个通信链的句柄。

三类端口

  1. 公认端口(Well Known Ports):从0到1023,它们紧密绑定(binding)于一些服务。通常这些端口的通讯 明确表明了某种服务的协议。例如:80端口实际上总是HTTP通讯。
  2. 注册端口(Registered Ports):从1024到49151。它们松散地绑定于一些服务。也就是说有许多服务绑定于这些端口,这些端口同样用于许多其它目的。例如:许多系统处理动态端口从1024左右开始。
  3. 动态和/或私有端口(Dynamic and/or Private Ports):从49152到65535。理论上,不应为服务分配这些端 口。实际上,机器通常从1024起分配动态端口。但也有例外:SUN的RPC端口从32768开始。

Socket种类

1. 流式Socket(STREAM):是一种面向连接的Socket,针对于面向连接的TCP服务应用,安全,但是效率低;

2. 数据报式Socket(DATAGRAM):是一种无连接的Socket,对应于无连接的UDP服务应用.不安全(丢失,顺序混乱,在接收端要分析重排及要求重发),但效率高.

Socket一般应用模式(服务器端和客户端)

l  服务器端的Socket(至少需要两个),一个负责接收客户端连接请求(但不负责与客户端通信)每成功接收到一个客户端的连接便在服务端产生一个对应的负责通信的Socket,在接收到客户端连接时创建,为每个连接成功的客户端请求在服务端都创建一个对应的Socket(负责和客户端通信).

l  客户端的Socket,必须指定要连接的服务端地址和端口,通过创建一个Socket对象来初始化一个到服务器端的TCP连接。

Socket的通讯过程

服务器端:

1、 申请一个socket

2、 绑定到一个IP地址和一个端口上

3、 开启侦听,等待接授连接

客户端:

1、 申请一个socket

2、 连接服务器(指明IP地址和端口号)

3、 服务器端接到连接请求后,产生一个新的socket(端口大于1024)与客户端建立连接并进行通讯,原监听socket继续监听。

Socket的构造函数

连接通过构造函数完成。

public Socket(AddressFamily addressFamily, SocketType socketType, ProtocolType protocolType)

如:mySocket = new Socket(AddressFamily.InterNetwork,SocketType.Stream, ProtocolType.Tcp);

Socket方法

  • IPAddress类:包含了一个IP地址
  • IPEndPoint类:包含了一对IP地址和端口号
  • Socket (): 创建一个Socket
  • Bind(): 绑定一个本地的IP和端口号(IPEndPoint)
  • Listen(): 让Socket侦听传入的连接尝试,并指定侦听队列容量
  • Connect(): 初始化与另一个Socket的连接
  • Accept(): 接收连接并返回一个新的socket
  • Send(): 输出数据到Socket
  • Receive(): 从Socket中读取数据
  • Close(): 关闭Socket (销毁连接)

多线程、Socket的更多相关文章

  1. 多线程socket UDP收发数据

    多线程socket收发数据 from threading import Thread from socket import * def sendData(): while True: sendInfo ...

  2. Java多线程 Socket使用

    点我跳过黑哥的卑鄙广告行为,进入正文. Java多线程系列更新中~ 正式篇: Java多线程(一) 什么是线程 Java多线程(二)关于多线程的CPU密集型和IO密集型这件事 Java多线程(三)如何 ...

  3. Java多线程Socket在控制台输出的多人聊天室编程

    服务器端代码 import java.io.IOException; import java.net.ServerSocket; import java.net.Socket; import java ...

  4. Java 多线程Socket编程通讯--实现聊天室代码

    1.创建服务器类 import java.io.IOException; import java.net.ServerSocket; import java.net.Socket; import ja ...

  5. Java的多线程+Socket 后台 Ver 2.0

    package com.wulala; import java.io.IOException;import java.net.ServerSocket;import java.net.Socket; ...

  6. 使用libevent进行多线程socket编程demo

    最近要对一个用libevent写的C/C++项目进行修改,要改成多线程的,故做了一些学习和研究. libevent是一个用C语言写的开源的一个库.它对socket编程里的epoll/select等功能 ...

  7. linux多线程socket编程一些心得

    http://hi.baidu.com/netpet/blog/item/2cc79216d9012b54f2de32b9.html 前段时间将新的web模型办到linux上来,用epoll代替了IO ...

  8. 多线程+socket实现多人聊天室

    最近在学习多线程的时候打算做一个简单的多线程socke聊天的程序,结果发现网上的代码都没有完整的实现功能,所以自己实现了一个demo: demo功能大致就是,有一个服务端负责信息转发,多个客户端发送消 ...

  9. Linux下C编写基本的多线程socket服务器

    不想多说什么,会搜这些东西的都是想看代码的吧. 一开始不熟悉多线程的时候还在想怎么来控制一个线程的结束,后来发现原来有pthread_exit()函数可以直接在线程函数内部调用结束这个线程. 开始还想 ...

  10. day08 多线程socket 编程,tcp粘包处理

    复习下socket 编程的步骤: 服务端:   1 声明socket 实例 server = socket.socket()  #括号里不写  默认地址簇使用AF_INET  即 IPv4       ...

随机推荐

  1. perl静态编译DBD

    编译DBD 项目中经常使用perl,但perl在连接数据库时,需要依赖DBI,DBD驱动,但默认安装DBD驱动时,需要依赖数据库的lib库. 比如perl连接MySQL,需要安装MySQL clien ...

  2. ♫【网站优化】Reflow / Repaint

    web移动开发最佳实践之js篇 浏览器的回流与重绘 by 张盛志 DOM性能瓶颈与Javascript性能优化 浏览器的渲染原理简介 其中一个跟浏览器有关的原因,那就是浏览器需要花时间.花精力去渲染. ...

  3. Linux学习笔记32——select()函数分析【转】

    Select在Socket编程中还是比较重要的,可是对于初学Socket的人来说都不太爱用Select写程序,他们只是习惯写诸如 connect.accept.recv或recvfrom这样的阻塞程序 ...

  4. HDU 1712 ACboy needs your help 典型的分组背包

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1712 ACboy needs your help Time Limit: 1000/1000 MS ( ...

  5. [转]在MacOS和iOS系统中使用OpenCV

    OpenCV 是一个开源的跨平台计算机视觉库,实现了图像处理和计算机视觉方面的很多通用算法. 最近试着在MacOS和iOS上使用OpenCV,发现网上关于在MacOS和iOS上搭建OpenCV的资料很 ...

  6. 一步一步制作yaffs/yaffs2根文件系统(七)---真挚地道歉以及纠正前边出现的错误!

    接上一节http://blog.csdn.net/mybelief321/article/details/10040939 说实话,我当时写这个系列的博文的时候,感觉对BusyBox算是有点了解,直到 ...

  7. 关键字instanceof和final

    Instanceof关键字(类似oc的isKindOfClass 和 isMemberOfClass) instanceof(实例类型) 关键字作用: 1.判断某一个对象是否属于某一个类 2.inst ...

  8. 手机APP上下滚动翻页效果

    //页面初期加载时        $(document).ready(function () { //加载第一页            LoadList(); //滚动换页            $( ...

  9. PHP文件操作常用函数总结

    一 .解析路径: 1 获得文件名: basename(); 给出一个包含有指向一个文件的全路径的字符串,本函数返回基本的文件名.如果文件名是以 suffix 结束的,那这一部分也会被去掉. eg: $ ...

  10. postgresql9.5 run 文件linux安装后配置成开机服务

    网上出现的比较多安装方法要么是源码安装,要么是yum安装,我发觉都要配置很多属性,比较麻烦,所以现在我在centos7长用 run文件来安装 http://get.enterprisedb.com/p ...