|
|||||||
|
|
|
![]() |
|
|
Strumenti |
|
|
#1 |
|
Senior Member
Iscritto dal: Aug 2004
Città: Palermo
Messaggi: 1079
|
C#: come trasformare classi in byte da inviare su udp?
Salve a tutti sto scrivendo un client ed un server udp che si scambiano dei dati.
Girando su internet ho trovato esempi di come fare in modo che due host si scambino stringhe: 1)la stringa viene convertita in array di byte 2)la stringa viene inviata sul socket. Ora la conversione in array di byte la faccio in questo modo: Codice:
string daInviare = "Messaggio da inviare";
Byte[] sendBytes = Encoding.ASCII.GetBytes(daInviare);
Codice:
int recv;
byte[] data = new byte[1024];
recv = newsock.ReceiveFrom(data, ref Remote);
Console.WriteLine(Encoding.ASCII.GetString(data));
Grazie a tutti, ciao ciao. |
|
|
|
|
|
#2 | |
|
Senior Member
Iscritto dal: Feb 2002
Messaggi: 906
|
Quote:
fai un &cast: (const char*) &Dati //dati della struttura N.B. sul client e il server devi avere le stesse strutture di sizeoff() //uguali! se no cosa comunicano... esempio: //invio send (socket, (const char*) &Dati, sizeof(MiaStruct), 0); //ricezione char rec(1024); bytes = recv(socket, rec, sizeof(MiaStruct), 0); Ultima modifica di okay : 26-10-2006 alle 23:01. |
|
|
|
|
|
|
#3 |
|
Senior Member
Iscritto dal: Aug 2004
Città: Palermo
Messaggi: 1079
|
Ciao, grazie mille, le stesse strutture di sizeof in che senso?
Invece mi chiedo un'altra cosa se comunicano due computer diversi non ci possono essere problemi differenti per quanto riguarda la dimensione della struct, l'ordinamento di byte (little o big endian)? Questi problemi se li gestisce in automatico il CLR oppure devo fare tutto a mano come capitava in C (ho programmato qualche socket in C tempo fa e ricordo cose simili)? Grazie, ciao ciao. |
|
|
|
|
|
#4 | ||
|
Senior Member
Iscritto dal: Feb 2002
Messaggi: 906
|
Quote:
struct miastruct int a char b ; //server struct miastruct int a char b//se commenti char b la dimensione della struttura cambia // e avrai un crash al prg se tenti di prendere b che non c'è ; Quote:
winsock big endian bit alto 7 + significativo 0 bit basso meno significativo 7 | 0 | 0 | 0 target.sin_port = htons (wPort); La funzione htons (Host to Network) converte un numero nel formato del computer locale in quello della rete (è stato scelto il Big-Endian per internet vedi sopra). Anche se si sa che il computer locale è Big-Endian è ottima cosa usare lo stesso questa funzione, che in tal caso sarà solo una macro vuota. ciao |
||
|
|
|
|
|
#5 |
|
Senior Member
Iscritto dal: Aug 2004
Città: Palermo
Messaggi: 1079
|
Ecco io mi riferivo proprio a queste operazioni. In C ricordo che dovevo farle per forza dato che lavoravo a basso livello, in C# cercando di lavorare ad un livello più alto, utilizzando le sue librerie questi controlli li posso evitare?
Ricordo, anche se in modo non preciso, che controlli simili in java non erano necessari, ci pensava la JVM. In C# non ci pensa il CLR? Grazie, ciao ciao. |
|
|
|
|
|
#6 |
|
Member
Iscritto dal: Dec 2001
Città: Cernobbio -Co-
Messaggi: 47
|
Il modo più semplice per inviare oggetti sulla rete è quello di ricorrere alla serializzazione.
Ti riporto un esempio molto semplice, dove l'oggetto che si invia contiene una stringa Crea la classe in questo modo: Codice:
[Serializable]
public class ClassToSend
{
private string message;
public ClassToSend(string message)
{
this.message = message;
}
public override string ToString()
{
return "Message: " + this.message;
}
}
Codice:
public class Client
{
public void SendMessage(string message, int port)
{
ClassToSend messageContainer = new ClassToSend(message);
TcpClient tcpClient = new TcpClient("localhost", port);
NetworkStream stream = tcpClient.GetStream();
BinaryFormatter binaryFormatter = new BinaryFormatter();
binaryFormatter.Serialize(stream, messageContainer);
stream.Close();
}
}
Codice:
public class Server
{
TcpListener listener;
public Server(int port)
{
listener = new TcpListener(port);
}
public void Start()
{
Thread listenThread = new Thread(Listen);
listenThread.Start();
}
private void Listen()
{
Console.WriteLine("In attesa...");
listener.Start();
TcpClient client = listener.AcceptTcpClient();
Stream stream = client.GetStream();
BinaryFormatter binaryFormatter = new BinaryFormatter();
ClassToSend message = (ClassToSend)binaryFormatter.Deserialize(stream);
Console.WriteLine(message.ToString());
}
}
Codice:
static void Main(string[] args)
{
Server server = new Server(1234);
server.Start();
Client client = new Client();
client.SendMessage("ciao", 1234);
Console.ReadLine();
}
__________________
micheledellatorre.net |
|
|
|
|
|
#7 |
|
Senior Member
Iscritto dal: Sep 2001
Città: Piazzola sul Brenta (PD)
Messaggi: 262
|
mi riaggancio a questo thread per una questione che mi sta assillando da qualche giorno:
ho una libreria scritta in C++ con la quale scambio da un applicativo all'altro dei pacchetti di dimensione variabile. precisamente invio su un socket il numero di byte che trasmetterò, e successivamente il buffer di char che codifica la mia struttura. questo tramite una funzione write(void *pcMem, int iSize)... se io volessi replicare lo stesso concetto sotto c#(per comunicare con gli applicativi scritti in c++!), dovrei fare il marshalling della struttura che voglio inviare prima di serializzarla...in modo da assicurarmi che l'allineamento dei dati replichi quello implementato dal compilatore c. però una struttura a dimensione variabile tipo questa: [StructLayout(LayoutKind.Sequential, Pack = 1)] struct T_MY_DATA { public uint uiSize; [MarshalAs(UnmanagedType.ByValArray, SizeConst = 32)]; public byte[] pcMem; // }; mi permette di inviare SOLAMENTE strutture con array di 32 byte(o in generale il numero di byte specificati in SizeConst...)...e questo non è buono... in attesa di illuminazione... thk
__________________
linux user #344456 |
|
|
|
|
| Strumenti | |
|
|
Tutti gli orari sono GMT +1. Ora sono le: 23:40.



















