一:有图有真相,很简单

a,

b,

Thread 构造函数
(ParameterizedThreadStart)
初始化 Thread 类的新实例,指定允许对象在线程启动时传递给线程的委托。

参数
start
类型:System.Threading.ParameterizedThreadStart
ParameterizedThreadStart 委托,它表示此线程开始执行时要调用的方法。

c,服务器的线程和套接字

线程:

threadmain                  listen()

threadaccept            accept()

threadconmmunication         send() receive()

套接字:

socketlisten      listen()

socketsend      send()

socketreceive  receive()

二:客户端代码:

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 System.Net.Sockets;
using System.Threading;
using System.Net;
using System.IO; namespace client
{
public partial class Client : Form
{
public Client()
{
InitializeComponent(); }
//创建 1个客户端套接字 和1个负责监听服务端请求的线程
Socket socketClient = null;
Thread threadClient = null;
private delegate void SetText(string text);
/// <summary>
/// 链接服务端
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void btn_login_Click(object sender, EventArgs e)
{
socketClient = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
//这里的ip地址,端口号都是服务端绑定的相关数据。
IPAddress ip = IPAddress.Parse(tb_ip.Text.Trim());
IPEndPoint endpoint = new IPEndPoint(ip, Convert.ToInt32(tb_socket.Text.Trim()));
socketClient.Connect(endpoint);//链接有端口号与IP地址确定服务端. ThreadStart thS = new ThreadStart(this.ReceMsg); //客户端在接受服务端发送过来的数据是通过Socket 中的Receive方法,
//该方法会阻断线程,所以我们自己为该方法创建了一个线程
threadClient = new Thread(thS);
threadClient.IsBackground = true;//设置后台线程
threadClient.Start();
}
//接收服务端数据
public void ReceMsg()
{
while (true)//持续监听服务端发来的消息
{
//定义一个2M的内存缓冲区 用于临时性存储接收到的信息
byte[] buffer = new byte[ * * ];
//将客户端套接字接收到的数据存入内存缓冲区, 并获取其长度
int length = socketClient.Receive(buffer);
//clientSocket.Receive(buffer);//接收服务端发送过来的数据
string ReceiveMsg = System.Text.Encoding.UTF8.GetString(buffer,,length);//把接收到的字节数组转成字符串显示在文本框中,如果没有长度,就会以零填充满,后面的日期就会不显示了。
ShowMsg("接收到数据: " + ReceiveMsg);
}
} //消息框里面数据
void ShowMsg(string str)
{
if (this.tb_infor.InvokeRequired)
{
SetText st = new SetText(ShowMsg);
this.Invoke(st, new object[] { str });
}
else
{
//string Ystr = "";
//if (tb_infor.Text != "")
//{
// Ystr = tb_infor.Text + "\r\n";
//}
//tb_infor.Text = Ystr + str + GetCurrentTime();
//代替上面的方法
tb_infor.AppendText(str + GetCurrentTime() + "\r\n");
}
}/// <summary>
/// 发送消息
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void btn_sendmessage_Click(object sender, EventArgs e)
{ string txtMsg = tb_message.Text;
//将输入的内容字符串转换为机器可以识别的字节数组
byte[] buffer = Encoding.UTF8.GetBytes(txtMsg);
socketClient.Send(buffer);
ShowMsg("发送的数据: " + txtMsg); }
#endregion
/// <summary>
/// 获取当前系统时间的方法
/// </summary>
/// <returns>当前时间</returns>
private DateTime GetCurrentTime()
{
DateTime currentTime = new DateTime();
currentTime = DateTime.Now;
return currentTime;
}
}
}

三:服务器代码:

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 System.Net.Sockets;
using System.Net;
using System.Threading;
using System.IO; namespace socket
{
public partial class Server : Form
{
public Server()
{
InitializeComponent();
//关闭对文本框的非法线程操作检查,不建议用
//TextBox.CheckForIllegalCrossThreadCalls = false;
}
Socket socketWatch = null;//负责监听客户端的套接字
Thread threadWatch = null;//负责监听客户端的线程 //声明一个带参数委托
private delegate void SetText(string text);
/// 开始服务端监听
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void btn_StartServer_Click(object sender, EventArgs e)
{
#region
//创建一个Socket实例
//第一个参数表示使用ipv4
//第二个参数表示发送的是数据流
//第三个参数表示使用的协议是Tcp协议
socketWatch = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
//获取ip地址
IPAddress ip = IPAddress.Parse(tb_ip.Text.Trim());
//创建一个网络通信节点,这个通信节点包含了ip地址,跟端口号。
//这里的端口我们设置为1029,这里设置大于1024,为什么自己查一下端口号范围使用说明。
IPEndPoint endPoint = new IPEndPoint(ip, int.Parse(tb_socket.Text.Trim()));
//设置SOCKET允许多个SOCKET访问同一个本地IP地址和端口号
socketWatch.SetSocketOption(SocketOptionLevel.Socket, SocketOptionName.ReuseAddress, true);
//Socket绑定网路通信节点
socketWatch.Bind(endPoint);
//将套接字的监听队列长度限制为10
socketWatch.Listen(); ShowMsg("开启监听!"); //实力化一个线程上的委托
ThreadStart threadDelegate = new ThreadStart(this.accpet);
//实力化一个处理线程委托的的新线程
threadWatch = new Thread(threadDelegate);
//等同于上面两句newthread = new Thread(new ThreadStart(this.accpet));
threadWatch.IsBackground = true;
threadWatch.Start();
#endregion
} //消息框里面数据
public void ShowMsg(string str)
{
if (this.tb_infor.InvokeRequired)
{
//实例化一个委托
SetText d = new SetText(ShowMsg);
this.Invoke(d, new object[] { str });
}
else
{ //string Ystr = "";
//if (tb_infor.Text != "")
//{
// Ystr = tb_infor.Text + "\r\n";
//}
//this.tb_infor.Text = Ystr + str + GetCurrentTime();
tb_infor.AppendText(str + GetCurrentTime() + "\r\n");
}
}
Socket SocketConnection = null;//创建一个负责和客户端通信的套接字
public void accpet()
{
while (true)//注意该循环,服务端要持续监听,要不然一个客户端链接过后就无法链接第二个客户端了。
{
//创建一个接收客户端通信的Socket
SocketConnection = socketWatch.Accept();
//如果监听到客户端有链接,则运行到下一部,提示,链接成功!
ShowMsg("链接成功!");
//创建一个通信线程
ParameterizedThreadStart pts = new ParameterizedThreadStart(ServerRecMsg);
Thread thr = new Thread(pts);
thr.IsBackground = true;
//启动线程
thr.Start(SocketConnection);
} }
/// <summary>
/// 向客户端发送信息
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void btn_sendany_Click(object sender, EventArgs e)
{
string SendMsg = tb_message.Text;
if (SendMsg != "")
{
byte[] buffer = System.Text.Encoding.UTF8.GetBytes(SendMsg); //将要发送的数据,生成字节数组。
SocketConnection.Send(buffer);
ShowMsg("向客户端发送了:" + SendMsg);
} } /// <summary>
/// 获取当前系统时间的方法
/// </summary>
/// <returns>当前时间</returns>
private DateTime GetCurrentTime()
{
DateTime currentTime = new DateTime();
currentTime = DateTime.Now;
return currentTime;
} /// <summary>
/// 接收客户端发来的信息
/// </summary>
/// <param name="socketClientPara">客户端套接字对象</param>
private void ServerRecMsg(object socketClientPara)
{
Socket socketServer = socketClientPara as Socket;
while (true)
{
//创建一个内存缓冲区 其大小为1024*1024字节 即1M
byte[] arrServerRecMsg = new byte[ * * ];
//将接收到的信息存入到内存缓冲区,并返回其字节数组的长度
int length = socketServer.Receive(arrServerRecMsg);
//将机器接受到的字节数组转换为人可以读懂的字符串
string strSRecMsg = Encoding.UTF8.GetString(arrServerRecMsg, , length);
//将发送的字符串信息附加到文本框txtMsg上
ShowMsg("接受客户端数据:" + strSRecMsg); } }
}
}

socket(tcp)互发信息的更多相关文章

  1. 7-51单片机ESP8266学习-AT指令(8266TCP服务器,编写自己的C#TCP客户端发信息给单片机控制小灯的亮灭)

    http://www.cnblogs.com/yangfengwu/p/8780182.html 自己都是现做现写,如果想知道最终实现的功能,请看最后 先把源码和资料链接放到这里 链接: https: ...

  2. 初识python:scoket 单用户互发消息

    实现功能: 启动"服务器".通过"客户端1"连接"服务器",然后互发消息.在此过程中,有"客户端2"连接到"服 ...

  3. day29 python 套接字socket TCP udp 形式发送信息的区别

    我们经常把socket翻译为套接字,socket是在应用层和传输层之间的一个抽象层,它把TCP/IP层复杂的操作抽象为几个简单的接口供应用层调用已实现进程在网络中通信. socket起源于UNIX,在 ...

  4. 分布式消息总线,基于.NET Socket Tcp的发布-订阅框架之离线支持,附代码下载

    一.分布式消息总线以及基于Socket的实现 在前面的分享一个分布式消息总线,基于.NET Socket Tcp的发布-订阅框架,附代码下载一文之中给大家分享和介绍了一个极其简单也非常容易上的基于.N ...

  5. ActionScript简单实现Socket Tcp应用协议分析器

    转自..smark http://www.cnblogs.com/smark/archive/2012/05/15/2501507.html ActionScript简单实现Socket Tcp应用协 ...

  6. 与众不同 windows phone (30) - Communication(通信)之基于 Socket TCP 开发一个多人聊天室

    原文:与众不同 windows phone (30) - Communication(通信)之基于 Socket TCP 开发一个多人聊天室 [索引页][源码下载] 与众不同 windows phon ...

  7. 27.Socket,TCP,UDP,HTTP基本通信原理

    Socket,TCP,UDP,HTTP基本通信原理(摘自百度): TCP.UDP,HTTP 底层通信都是通过 socket 套接字实现 网络上不同的计算机,也可以通信,那么就得使用网络套接字(sock ...

  8. 程序猿修仙之路--数据结构之你是否真的懂数组? c#socket TCP同步网络通信 用lambda表达式树替代反射 ASP.NET MVC如何做一个简单的非法登录拦截

    程序猿修仙之路--数据结构之你是否真的懂数组?   数据结构 但凡IT江湖侠士,算法与数据结构为必修之课.早有前辈已经明确指出:程序=算法+数据结构  .要想在之后的江湖历练中通关,数据结构必不可少. ...

  9. 基于UDP的socket tcp和udp的区别(小白进击篇)

    目录 16.基于udp协议的socket通信 为什么udp不会有粘包现象 DGRAM datagram#数据报文 发送sento (发送的信息,发送给的地址) 接收revefrom 客户端 服务端 t ...

随机推荐

  1. CGLIB学习笔记

    0 概述 CGLIB基于ASM实现.提供比反射更为强大的动态特性.使用CGLIB可以非常方便的实现的动态代理. 0.1 CGLIB包结构 net.sf.cglib.core    底层字节码处理类. ...

  2. HTML5 FileReader读取Blob对象API详解

    使用FileReader对象,web应用程序可以异步的读取存储在用户计算机上的文件(或者原始数据缓冲)内容,可以使用File对象或者Blob对象来指定所要读取的文件或数据.其中File对象可以是来自用 ...

  3. Android Activity各启动模式的差异

    Activity共有四种启动模式:standard,singleTop,singleTask,singleInstance 为了方便描述和理解,布局文件.Manifest文件和各个java文件如下: ...

  4. java中XMLGregorianCalendar类型和Date类型之间的相互转换

    import java.text.SimpleDateFormat;import java.util.Date;import java.util.GregorianCalendar;import ja ...

  5. P2P

    https://www.ppmoney.com/Withdraw http://www.daibang.com/

  6. 国际化 native2ascii用法

    cmd下输入: native2ascii -encoding GBK(需要编译成哪种语言) (中文文件路劲) (英文文件路劲) 其他固定 例如 native2ascii -encoding GBK C ...

  7. 抓取锁的sql语句-第七次修改

    最近闲来没事,把之前写的那个抓取锁的存储过程重新修改.优化了一下,呵呵 create or replace procedure solve_lock_061203_wanjie(v_msg out v ...

  8. C#读取Visual FoxPro(*.dbf)数据并使用SqlBulkCopy插入到SqlServer 2008 R2数据表中

    公司数据库从32位的SqlServer 2005升级到64位的SqlServer 2008 R2后,无法再像原来通过Link Server连接VFP同步数据,因此考虑用代码程序从VFP数据库中读取所需 ...

  9. 获取C++类成员变量的地址偏移

    今天有在校学生问怎么获取类中的成员变量的地址偏移量,这个应该是很多初学C++的人很好奇的问题.以前我在学校的时候,也有过这种需求.忘了当时是要写什么“奇怪的程序”了,反正需要获取一个类的成员变量的地址 ...

  10. QWidget QMainWindow QDialog 三者区别

    Qt类是一个提供所需的像全局变量一样的大量不同的标识符的命名空间.通常情况下,你可以忽略这个类.QObject和一些其它类继承了它,所以在这个Qt命名空间中定义的所有标识符通常情况下都可以无限制的使用 ...