浅谈C# Socket编程及C#如何使用多线程
去年暑假学习了几个月asp.net 最后几个星期弄了下C# socket .也算知道了个一知半解了, 好久没动C#了, 虽然这语言高级的让我对他没兴趣, 不过还是回忆回忆, 忘了一干二净就不好了.
C# Socket:
建议初学C# socket的菜鸟朋友不要使用TcpListenner, TcpClient这些MS封装好的类库, 这些封装好的类用起来的确方便, 但你用完了你又学到了什么了? 那该用什么了, 只用Socket这一个类. 不错,这样会麻烦点的,
但是, 在C#里面, 就连Socket, MS都进行了一翻封装,使得Socket使用起来也是十分的简单, 我刚学的时候写过一个很菜的TCP聊天程序, 两人对聊的. 大家可以去尝试下一S多C的聊天程序,TCP会了可以去做个UDP的.
UDP会了,学学SMTP, 哎, SMTP也是封装的太厉害, 都成傻瓜式的了, 然后大家可以看下MultiThread,也就是多线程, 这些都差不多了, 可以取尝试写个Proxy. 我当时就是这样学的, 呵呵, 不过我只是个菜鸟, 现在搞asm/c/C++去了,就把这些忘了差不多.
首先必须包含的两个命名空间:
Using System.Net;
Using System.Net.Sockets;
几个常用的类:(这些东西,查下MSDN就很详细了)
IPHostEntry, Dns,IPAddress,IPEndPoint,还有最重要的Socket
IPEndPoint: 这个是网络终结点,很好理解,就是网络上一个固定的地址:一个IP与一个端口的组合.
下面我还是以我以前写的一个很简单的聊天程序做示例吧, (很短代码的)
Form1.cs
//说明下, 这个是集Server与Client与一体的.
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Text;
using System.Windows.Forms;
using System.Net;
using System.Net.Sockets; //这个和上面的是使用Socket必须的.
using System.IO;
using System.Threading; //这个是使用多线程必需的.
namespace OnlySocket
{
public partial class Form1 : Form //partial表示这块代码只是Form1类的部分, Form1类继承自Form类
{
public Form1()
{
InitializeComponent(); //构造函数, 初始化容器.
}
Socket sock; //定义一个Socket类的对象 (默认为protected)
Thread th; //定义一个Thread类的对象
//
public static IPAddress GetServerIP() //静态函数, 无需实例化即可调用.
{
IPHostEntry ieh = Dns.GetHostByName(Dns.GetHostName()); //不多说了, Dns类的两个静态函数
//或用DNS.Resolve()代替GetHostName()
return ieh.AddressList[0]; //返回Address类的一个实例. 这里AddressList是数组并不奇怪,一个Server有N个IP都有可能
}
private void BeginListen() //Socket监听函数, 等下作为创建新线程的参数
{
IPAddress serverIp = GetServerIP(); //调用本类静态函数GetServerIP得到本机IPAddress.
IPEndPoint iep = new IPEndPoint(serverIp, Convert.ToInt32(tbPort.Text)); //本地终结点
sock = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp); //实例化内成员sock
Byte[] byteMessage = new Byte[100]; //存放消息的字节数组缓冲区, 注意数组表示方法,和C不同的.
this.lbIep.Text = iep.ToString();
sock.Bind(iep); //Socket类的一个重要函数, 绑定一个IP,
while (true) //这里弄了个死循环来监听端口, 有人会问死循环了,那程序不卡住了, 注意这只是个类, 这里还没有main函数呢.
{
try
{
sock.Listen(5); //好了,sock绑定了本地终结点就可以开始监听了,5表示最大连接数为5
Socket newSock = sock.Accept(); //这里又有Socket类的一个重要的方法:Accept, 该方法接受来自外面的Socket连接请求, 并返回一个Socket套接字, 这个套接字就开始处理这一个client与Server之间的对话
newSock.Receive(byteMessage); //接受client发送过来的数据保存到缓冲区.
string msg = "From [" + newSock.RemoteEndPoint.ToString() + "]:" +System.Text.Encoding.UTF8.GetString(byteMessage)+"\n"; //GetString()函数将byte数组转换为string类型.
rtbTalk.AppendText(msg+"\n"); //显示在文本控件里
}
catch (SocketException se) //捕捉异常,
{
lbState.Text = se.ToString(); //将其显示出来, 在此亦可以自定义错误.
}
}
}
private void btConnect_Click(object sender, EventArgs e) //连接按钮触发的事件: 连接Server
{
btConnect.Enabled = false;
btStopConnect.Enabled = true;
try
{
th = new Thread(new ThreadStart(BeginListen)); //创建一个新的线程专门用于处理监听,这句话可以分开写的,比如: ThreadStart ts=new ThreadStart(BeginListen); th=new Thread (ts); 不过要注意, ThreadStart的构造函数的参数一定要是无参数的函数. 在此函数名其实就是其指针, 这里是委托吗?
th.Start(); //启动线程
lbState.Text = "Listenning...";
}
catch (SocketException se) //处理异常
{
MessageBox.Show(se.Message, "出现问题", MessageBoxButtons.OK, MessageBoxIcon.Information);
}
catch (ArgumentNullException ae) //参数为空异常
{
lbState.Text = "参数错误";
MessageBox.Show(ae.Message, "错误", MessageBoxButtons.OK, MessageBoxIcon.Warning);
}
}
private void btStopConnect_Click(object sender, EventArgs e) //停止监听
{
btStopConnect.Enabled = false;
btConnect.Enabled = true;
sock.Close(); //关闭套接字
th.Abort(); //终止监听线程
lbState.Text = "Listenning stopped";
}
private void btExit_Click(object sender, EventArgs e)
{
sock.Close();
th.Abort();
Dispose(); //清理资源,就是释放内存
this.Close(); //关闭对话框, 退出程序
}
private void btSend_Click(object sender, EventArgs e)
{
try
{
IPAddress clientIp = IPAddress.Parse(tbTargetIp.Text); //类IPAddress的静态函数Parse() :将Text转化为IPAddress的一个实例.
int clientPort = Convert.ToInt32(tbPort.Text); //C#的这些转化函数很方便的,不像C++那样麻烦
IPEndPoint clientIep = new IPEndPoint(clientIp, clientPort); //这里用client表示不是很好....,
Byte[] byte_Message;
Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp); 实例化的时候还有很多参数的, 这个是Tcp的. Tcp的SocketType是Stream:数据流, 如果协议类型是UDP, 则是数据包传送, QQ就是用的UDP.
socket.Connect(clientIep); //Socket的又一个函数Connect(IPEndPoint) .连接远程套接字
byte_Message = System.Text.Encoding.UTF8.GetBytes(rtbWords.Text); //发现UTF8可支持中文,就用之
socket.Send(byte_Message);
rtbTalk.AppendText("\n"+"My words:" + rtbWords.Text + "\n");
socket.Shutdown(SocketShutdown.Both);
socket.Close();
}
catch (ArgumentNullException ae)
{
MessageBox.Show(ae.Message,"参数为空",MessageBoxButtons.OKCancel,MessageBoxIcon.Information);
}
catch (SocketException se)
{
MessageBox.Show(se.Message, "出现问题", MessageBoxButtons.OK, MessageBoxIcon.Information);
}
}
}
}
Program.cs
using System;
using System.Collections.Generic;
using System.Windows.Forms;
namespace OnlySocket
{
static class Program
{
/// <summary>
/// 应用程序的主入口点。
/// </summary>
[STAThread]
static void Main() //这儿才是main函数
{
Application.EnableVisualStyles();
Application.SetCompatibleTextRenderingDefault(false);
Application.Run(new Form1());
}
}
}
写了半天了, 够累的了, 虽然都是很基础的东西, 我自己写的时候也复习了一边 , 呵呵.
其实多线程我自己也不是很熟练, 记得去年暑假写过一个多线程扫描器, 也不知为啥, 线程开到50以上就异常, 很郁闷的. 其实当时我就是用的new Thread=Thread(new ThreadStart(Fun))实现的, 方法感觉很笨拙,呵呵.
大致代码好像是这样的吧:
先写个Scan类:
public class Scan
{
try{ public Scan(){ ...Init... }
public void Scan{ ..task循环扫描... } //task结构体里面有IP, 端口, 是否已扫描标记fLag}
catch{}
}
然后主函数里面可以这样搞:
Scan[] scanner = new Scan[XX]
Thread[] thread = new Thread[XX];
for (int i = 0; i < XX;i++)
{
scanner[i] = new Scan(this, i);
thread[i] = new Thread(new ThreadStart(scanner[i].StartScan));
thread[i].Start();
}
其实这样就可以简单的实现多线程了.