zoukankan      html  css  js  c++  java
  • 浅谈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();

                }

    其实这样就可以简单的实现多线程了.

  • 相关阅读:
    阿里云服务器绑定 微信公众号 服务器配置 问题记录
    Junit 报错: Failed to load ApplicationContext
    ERROR 1819 (HY000): Your password does not satisfy the current policy requirements
    idea 多项目部署JRebel
    sql优化1
    sql优化
    IntelliJ IDEA 2017版 spring-boot使用Spring Data JPA搭建基础版的三层架构
    IntelliJ IDEA 2017版 spring-boot 报错Consider defining a bean of type 'xxx' in your configuration问题解决方案
    IntelliJ IDEA 2017版 spring-boot 实现jpa基本部署,通过实体类自动建立数据库
    Nginx的两种负载均衡搭建(Tomcat版)
  • 原文地址:https://www.cnblogs.com/top5/p/1938634.html
Copyright © 2011-2022 走看看