.Net网络通讯编程[利用Socket实现字串、文件、序列化对象传输]--使用封装的网络服务3[聊天室][使用IE浏览本页]

原创 2012年03月28日 10:14:26

本案例使用前面封装的网络服务类编写简易聊天室,采用Tcp和多线程相关技术:

image

App.config文件:

示范代码<?xml version="1.0" encoding="utf-8" ?>
<configuration>
    <appSettings>
        <!--本地服务器名-->
        <add key="ServerName" value="127.0.0.1"/>
        <!--本地端口-->
        <add key="ServerPort" value="6000"/>

        <!--远程服务器名-->
        <add key="RemoteServerName" value="127.0.0.1"/>
        <!--远程服务器端口-->
        <add key="RemoteServerPort" value="6000"/>
    </appSettings>
</configuration>

 

 

窗体内所有代码:

示范代码using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
using TcpLabCommon;
using System.Net.Sockets;
using System.Net;
using System.Configuration;
using System.Threading;
using System.IO;

namespace TcpLabChat1
{
    public partial class FrmChat : Form
    {
        delegate void UpdateViewHandler(NetPacket packet);
        delegate void UpdateViewWithMsgHandler(string msg);
        /// <summary>
        /// Tcp服务对象[提供发包和收包功能],接收消息和文件使用
        /// </summary>
        NetPacketService tcpPacketServiceSvr = null;

        /// <summary>
        /// Tcp服务对象[提供发包和收包功能],发送消息和文件使用
        /// </summary>
        NetPacketService tcpPacketServiceClient = null;
               
        /// <summary>
        /// 客户端Socket[发送消息和文件]
        /// </summary>
        Socket clientSocket = null;

        public string ServerName
        {
            get
            {
                return ConfigurationManager.AppSettings["ServerName"];
            }
        }

        public Int32 ServerPort
        {
            get
            {
                return Convert.ToInt32(ConfigurationManager.AppSettings["ServerPort"]);
            }
        }

        public String RemoteServerName
        {
            get
            {
                return ConfigurationManager.AppSettings["RemoteServerName"];
            }
        }

        public Int32 RemoteServerPort
        {
            get
            {
                return Convert.ToInt32(ConfigurationManager.AppSettings["RemoteServerPort"]);
            }
        }

        public FrmChat()
        {
            InitializeComponent();
        }



        private void FrmChat_Load(object sender, EventArgs e)
        {
            Thread thread = new Thread(new ThreadStart(ReceiveMessage));
            thread.IsBackground = true;//设置当前线程为后台线程,以使当主窗体被关闭后自动结束线程
            thread.Start();
        }

        private void ReceiveMessage()
        {
            Socket serverSocket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
            EndPoint endPoint = new IPEndPoint(IPAddress.Parse(ServerName), ServerPort);//创建一个终结点
            serverSocket.Bind(endPoint);//绑定终结点
            serverSocket.Listen(0);//开始监听

            Socket ctSocket = serverSocket.Accept();//接收一个请求
            NetworkStream netStream = new NetworkStream(ctSocket);
            tcpPacketServiceSvr = new NetPacketTcpService(netStream);
            
            while(true)
            {
                //在窗口界面所在的线程运行委托指向的方法,避免跨线程访问UI问题
                Invoke(new UpdateViewHandler(UpdateView), tcpPacketServiceSvr.ReceivePacket());
            }
        }

        /// <summary>
        /// 更新界面
        /// </summary>
        /// <param name="packet"></param>
        private void UpdateView(NetPacket packet)
        {
            String msglist = txtMsgList.Text;
            switch (packet.PacketHead.PType)
            {
                case PacketType.STRING:
                    msglist += "\r\n对方说:"+packet.Data.ToString();
                    break;
                case PacketType.BINARY:
                    NetFile f = tcpPacketServiceSvr.ParseFile((Byte[])packet.Data);
                    msglist += "\r\n接收了对方发送的文件:"+f.FileName;
                    FileStream fs = null;
                    try
                    {
                        string fname = Application.StartupPath+"\\"+f.FileName;
                        fs = new FileStream(fname, FileMode.Create);                        
                        fs.Write(f.Content, 0, f.Content.Length);                        
                        fs.Flush();                       
                    }
                    catch (Exception e)
                    {
                        MessageBox.Show(e.Message);
                    }
                    finally
                    {
                        fs.Close();
                    }
                    break;
                case PacketType.COMPLEX:
                    break;
                default:
                    break;
            }

            txtMsgList.Text = msglist;
        }

        private void UpdateView(string msg)
        {
            String msglist = txtMsgList.Text;
            msglist += "\r\n"+msg;
            txtMsgList.Text = msglist;       
        }

        /// <summary>
        /// 发送消息
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void btnSendMsg_Click(object sender, EventArgs e)
        {
            if (txtMsg.Text.Trim().Length == 0)
                return;

            if (clientSocket == null)
            {
                clientSocket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
                clientSocket.Connect(IPAddress.Parse(RemoteServerName), RemoteServerPort);
                NetworkStream netStream = new NetworkStream(clientSocket);
                tcpPacketServiceClient = new NetPacketTcpService(netStream);
            }

            tcpPacketServiceClient.SendText(txtMsg.Text.Trim());
            String msglist = txtMsgList.Text;
            msglist += "\r\n我说:" + txtMsg.Text.Trim();            
            txtMsgList.Text = msglist;
        }

        /// <summary>
        /// 发送文件
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void btnSendFile_Click(object sender, EventArgs e)
        {
            OpenFileDialog fd = new OpenFileDialog();
            if (fd.ShowDialog() == System.Windows.Forms.DialogResult.Cancel)
            {
                return;
            }

            if (clientSocket == null)
            {
                clientSocket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
                clientSocket.Connect(IPAddress.Parse(RemoteServerName), RemoteServerPort);
                NetworkStream netStream = new NetworkStream(clientSocket);
                tcpPacketServiceClient = new NetPacketTcpService(netStream);
            }

            //在新的线程中完成文件发送
            Thread thread = new Thread(delegate() //采用C#2.0匿名方法语法
            {                
                tcpPacketServiceClient.SendFile(fd.FileName);

                //在窗口界面所在的线程运行委托指向的方法,避免跨线程访问UI问题                
                Invoke(new UpdateViewWithMsgHandler(UpdateView), "我发送了文件:" + new FileInfo(fd.FileName).Name);
            });
            thread.IsBackground = true;
            thread.Start();
            
        }
    }
}

 

使用方式:请将编译后的debug目录拷贝两份,分别配置不同的端口即可,比如a目录的配置端口为本地6001,远程6002,b目录的配置端口为本地6002,远程6001,这样即可在同一台机器实现通信。

.Net网络通讯编程[利用Socket实现字串、文件、序列化对象传输]--使用封装的网络服务4[聊天室][使用IE浏览本页]

本案例使用前面封装的网络服务类编写简易聊天室,采用Tcp和异步套接字相关技术: App.config文件: 示范代码xml version="1.0" encoding="utf-8" ?...
  • jiangtongcn
  • jiangtongcn
  • 2012-03-28 10:15:38
  • 693

.Net网络通讯编程[利用Socket实现字串、文件、序列化对象传输]--使用封装的网络服务1[使用IE浏览本页]

直接使用Socket做服务器端和客户端,采用Tcp协议: 客户端代码: 示范代码using System; using System.Collections.Generic; using Sy...
  • jiangtongcn
  • jiangtongcn
  • 2012-03-28 10:11:49
  • 635

.Net网络通讯编程[利用Socket实现字串、文件、序列化对象传输]--使用封装的网络服务2[使用IE浏览本页]

直接使用Socket做客户端,采用Udp 客户端代码: 示范代码using System; using System.Collections.Generic; using System.Lin...
  • jiangtongcn
  • jiangtongcn
  • 2012-03-28 10:13:20
  • 529

对象的序列化和Socket简单使用

对象序列化就是把对象转化为字节序列的过程,相对的反序列化就是把字节序列转化为对象的过程,是把对象转化成容易传输的流的方法,一般是生成传输的字节流或者是IO文件(长期存储)。 相信大家在学习一个新知识...
  • m0_38024791
  • m0_38024791
  • 2017-05-17 20:13:43
  • 581

Java Socket传输对象(序列化)

1.首先需要一个普通的对象类,由于需要序列化这个对象以便在网络上传输,所以实现java.io.Serializable接口就是必不可少的了...
  • SHEN525758203
  • SHEN525758203
  • 2014-08-08 13:58:48
  • 2703

.Net网络通讯编程[利用Socket实现字串、文件、序列化对象传输]--类设计1[使用IE浏览本页]

本案例使用.Net Socket的Tcp、Udp实现字串、文件、各种序列化对象的网络传输,同时封装了Tcp的粘包、半包处理细节,定义了网络封包格式,在发送端和接收端无需考虑内部传输细节。以下是类设计:...
  • jiangtongcn
  • jiangtongcn
  • 2012-03-28 10:02:55
  • 932

.Net网络通讯编程[利用Socket实现字串、文件、序列化对象传输]--类设计2[使用IE浏览本页]

本案例使用.Net Socket的Tcp、Udp实现字串、文件、各种序列化对象的网络传输,同时封装了Tcp的粘包、半包处理细节,定义了网络封包格式,在发送端和接收端无需考虑内部传输细节。以下是类设计:...
  • jiangtongcn
  • jiangtongcn
  • 2012-03-28 10:04:18
  • 1204

关于Socket传输流的序列化与反序列化

我想做的是 将一个自定义对象序列化后传输到网络的另一边,然后在接收端再反序列化 //客户端代码如下 public class ClientControlManager     {   ...
  • nuozhiyou
  • nuozhiyou
  • 2016-06-02 20:39:52
  • 2611

Socket通信之序列化对象通信

  • 2014年11月11日 11:53
  • 1KB
  • 下载

Android使用Serializable实现序列化传输对象

最近在做一个基于Socket通信项目,在实现客户端与服务器进行对象传输时使用到了Serializable接口对传输对象进行序列化,下面将和大家分享一下我的开发实例,欢迎各位交流探讨,不足之处希望各位多...
  • u010198148
  • u010198148
  • 2015-04-26 08:57:24
  • 1073
收藏助手
不良信息举报
您举报文章:.Net网络通讯编程[利用Socket实现字串、文件、序列化对象传输]--使用封装的网络服务3[聊天室][使用IE浏览本页]
举报原因:
原因补充:

(最多只允许输入30个字)