C# Socket網(wǎng)絡(luò)編程實例
本文實例講述了C# Socket網(wǎng)絡(luò)編程技巧。分享給大家供大家參考。具體分析如下:
客戶端要連接服務(wù)器:首先要知道服務(wù)器的IP地址。而服務(wù)器里有很多的應(yīng)用程序,每一個應(yīng)用程序?qū)?yīng)一個端口號
所以客戶端想要與服務(wù)器中的某個應(yīng)用程序進(jìn)行通信就必須要知道那個應(yīng)用程序的所在服務(wù)器的IP地址,及應(yīng)用程序所對應(yīng)的端口號
TCP協(xié)議:安全穩(wěn)定,一般不會發(fā)生數(shù)據(jù)丟失,但是效率低。利用TCP發(fā)生數(shù)據(jù)一般經(jīng)過3次握手(所有效率低,自己百度三次握手)
UDP協(xié)議:快速,效率高,但是不穩(wěn)定,容易發(fā)生數(shù)據(jù)丟失(沒有經(jīng)過三次握手,不管服務(wù)器有空沒空,信息全往服務(wù)器發(fā),所有效率搞,但服務(wù)器忙的時候就沒辦法處理你的數(shù)據(jù),容易造成數(shù)據(jù)丟失,不穩(wěn)定)
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;
namespace Socket通信
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
this.txtPort.Text = "5000";
this.txtIp.Text = "192.168.137.1";
}
private void btnStart_Click(object sender, EventArgs e)
{
//當(dāng)點擊開始監(jiān)聽的時候,在服務(wù)器端創(chuàng)建一個負(fù)責(zé)監(jiān)聽IP地址跟端口號的Socket
Socket socketWatch = new Socket(AddressFamily.InterNetwork,SocketType.Stream,ProtocolType.Tcp);
//Any:提供一個 IP 地址,指示服務(wù)器應(yīng)偵聽所有網(wǎng)絡(luò)接口上的客戶端活動。此字段為只讀。
IPAddress ip = IPAddress.Any;
//創(chuàng)建端口號對象;將txtPort.Text控件的值設(shè)為服務(wù)端的端口號
IPEndPoint point = new IPEndPoint(ip, Convert.ToInt32(txtPort.Text));
//監(jiān)聽
socketWatch.Bind(point);
ShowMsg("監(jiān)聽成功");
socketWatch.Listen(10);//連接隊列的最大長度 ;即:一個時間點內(nèi)最大能讓幾個客戶端連接進(jìn)來,超過長度就進(jìn)行排隊
//等待客戶端連接;Accept()這個方法能接收客戶端的連接,并為新連接創(chuàng)建一個負(fù)責(zé)通信的Socket
Thread th = new Thread(Listen); //被線程執(zhí)行的方法如果有參數(shù)的話,參數(shù)必須是object類型
Control.CheckForIllegalCrossThreadCalls = false; //因為.net不允許跨線程訪問的,所以這里取消跨線程的檢查。.net不檢查是否有跨線程訪問了,所有就不會報: “從不是創(chuàng)建控件“txtLog”的線程訪問它” 這個錯誤了,從而實現(xiàn)了跨線程訪問
th.IsBackground = true; //將th這個線程設(shè)為后臺線程。
//Start(object parameter); parameter:一個對象,包含線程執(zhí)行的方法要使用的數(shù)據(jù),即線程執(zhí)行Listen方法,Listen的參數(shù)
th.Start(socketWatch); //這個括號里的參數(shù)其實是Listen()方法的參數(shù)。因為Thread th = new Thread(Listen)這個括號里只能寫方法名(函數(shù)名) 但是Listen()方法是有參數(shù)的,所有就要用Start()方法將它的參數(shù)添加進(jìn)來
}
/// <summary>
/// 等待客戶端連接,如果監(jiān)控到有客戶端連接進(jìn)來就創(chuàng)建一個與之通信的Socket
/// </summary>
/// <param name="o"></param>
void Listen(object o) //這里為什么不直接傳遞Socket類型的參數(shù)呢? 原因是:被線程執(zhí)行的方法如果有參數(shù)的話,參數(shù)必須是object類型
{
Socket socketWatch = o as Socket;
while (true) //為什么這里要有個while循環(huán)?因為當(dāng)一個人連接進(jìn)來的時候創(chuàng)建了與之通信的Socket后就程序就會往下執(zhí)行了,就不會再回來為第二個人的連接創(chuàng)建負(fù)責(zé)通信的Socket了。(應(yīng)該是每個人(每個客戶端)創(chuàng)建一個與之通信的Socket)所以要寫在循環(huán)里。
{
//等待客戶端連接;Accept()這個方法能接收客戶端的連接,并為新連接創(chuàng)建一個負(fù)責(zé)通信的Socket
Socket socketSend = socketWatch.Accept();
dic.Add(socketSend.RemoteEndPoint.ToString(), socketSend); //(根據(jù)客戶端的IP地址和端口號找負(fù)責(zé)通信的Socket,每個客戶端對應(yīng)一個負(fù)責(zé)通信的Socket),ip地址及端口號作為鍵,將負(fù)責(zé)通信的Socket作為值填充到dic鍵值對中。
//我們通過負(fù)責(zé)通信的這個socketSend對象的一個RemoteEndPoint屬性,能夠拿到遠(yuǎn)程連過來的客戶端的Ip地址跟端口號
ShowMsg(socketSend.RemoteEndPoint.ToString() + ":" + "連接成功");//效果:192.168.1.32:連接成功
comboBox1.Items.Add(socketSend.RemoteEndPoint.ToString()); //將連接過來的每個客戶端都添加到combBox控件中。
//客戶端連接成功后,服務(wù)器應(yīng)該接收客戶端發(fā)來的消息。
Thread getdata = new Thread(GetData);
getdata.IsBackground = true;
getdata.Start(socketSend);
}
}
Dictionary<string, Socket> dic = new Dictionary<string, Socket>();
/// <summary>
/// 不停的接收客戶端發(fā)送過來的消息
/// </summary>
/// <param name="o"></param>
void GetData(object o)
{
while (true)
{
Socket socketSend = o as Socket;
//將客戶端發(fā)過來的數(shù)據(jù)先放到一個字節(jié)數(shù)組里面去
byte[] buffer = new byte[1024 * 1024 * 2]; //創(chuàng)建一個字節(jié)數(shù)組,字節(jié)數(shù)組的長度為2M
//實際接收到的有效字節(jié)數(shù); (利用Receive方法接收客戶端傳過來的數(shù)據(jù),然后把數(shù)據(jù)保存到buffer字節(jié)數(shù)組中,返回一個接收到的數(shù)據(jù)的長度)
int r = socketSend.Receive(buffer);
if (r == 0) //如果接收到的有效字節(jié)數(shù)為0 說明客戶端已經(jīng)關(guān)閉了。這時候就跳出循環(huán)了。
{
//只有客戶端給用戶發(fā)消息,不可能是發(fā)0個長度的字節(jié)。即便發(fā)空消息,空消息也是有過個長度的。所有接收到的有效字節(jié)長度為0就代表客戶端已經(jīng)關(guān)閉了
break;
}
//將buffer這個字節(jié)數(shù)組里面的數(shù)據(jù)按照UTF8的編碼,解碼成我們能夠讀懂的的string類型,因為buffer這個數(shù)組的實際存儲數(shù)據(jù)的長度是r個 ,所以從索引為0的字節(jié)開始解碼,總共解碼r個字節(jié)長度。
string str = Encoding.UTF8.GetString(buffer, 0, r);
ShowMsg(socketSend.RemoteEndPoint.ToString() + ":" + str);
}
}
private void ShowMsg(string str)
{
txtLog.AppendText(str + "\r\n"); //將str這個字符串添加到txtLog這個文本框中。
}
/// <summary>
/// 服務(wù)端給客戶端發(fā)信息
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void btnSend_Click_1(object sender, EventArgs e)
{
if (comboBox1.SelectedItem == null) //如果comboBox控件沒有選中值。就提示用戶選擇客戶端
{
MessageBox.Show("請選擇客戶端");
return;
}
string str = txtMes.Text; //獲取用戶輸入的內(nèi)容 (服務(wù)端要給客戶端發(fā)的信息)
byte[] strByte = Encoding.Default.GetBytes(str); //將信息轉(zhuǎn)換成二進(jìn)制字節(jié)數(shù)組
string getIp = comboBox1.SelectedItem as string; //comboBox存儲的是客戶端的(ip+端口號)
Socket socketSend = dic[getIp] as Socket; //根據(jù)這個(ip及端口號)去dic鍵值對中找對應(yīng) 賦值與客戶端通信的Socket【每個客戶端都有一個負(fù)責(zé)與之通信的Socket】
socketSend.Send(strByte); //將信息發(fā)送到客戶端
}
}
}
開打開始命令 cmd telnet 10.18.16.46 5000 即telnet 服務(wù)器ip地址 綁定的端口號
希望本文所述對大家的C#程序設(shè)計有所幫助。
相關(guān)文章
C#實現(xiàn)給DevExpress中GridView表格指定列添加進(jìn)度條
這篇文章主要為大家詳細(xì)介紹了如何利用C#實現(xiàn)給DevExpress中GridView表格指定列添加進(jìn)度條顯示效果,感興趣的小伙伴可以嘗試一下2022-06-06C#利用FluentFTP實現(xiàn)FTP上傳下載功能詳解
FTP作為日常工作學(xué)習(xí)中,非常重要的一個文件傳輸存儲空間,想必大家都非常的熟悉了,那么如何快速的實現(xiàn)文件的上傳下載功能呢,本文以一個簡單的小例子,簡述如何通過FluentFTP實現(xiàn)文件的上傳和下載功能2023-02-02C#如何將List<string>轉(zhuǎn)換為List<double>
這篇文章主要介紹了C#如何將List<string>轉(zhuǎn)換為List<double>問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-07-07解析美國東部時間與北京時間相互轉(zhuǎn)換的實現(xiàn)代碼
本篇文章是對美國東部時間與北京時間相互轉(zhuǎn)換的實現(xiàn)代碼進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05