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