Click here to Skip to main content
6,822,613 members and growing! (19,046 online)
Email Password   helpLost your password?
General Programming » Internet / Network » Internet and networks     Intermediate License: The Code Project Open License (CPOL)

MC++ TCP server talking to MFC & MC++ TCP clients

By Nishant Sivakumar

Basic demonstration of the TcpListener/TcpClient classes.
C++/CLI, VC7, Windows, .NET1.0, MFC, Dev
Posted:20 Oct 2001
Updated:1 Jul 2002
Views:119,201
Bookmarked:24 times
printPrint   add Share
      Discuss Discuss   Broken Article?Report  
53 votes for this article.
Popularity: 6.95 Rating: 4.03 out of 5

1

2

3
3 votes, 10.0%
4
27 votes, 90.0%
5

Introduction

The original article proposed to demonstrate elementary network programming by designing a very simple SMTP server class. It had no support for sending attachments and was quite uncomplicated in nature. Since there are much better classes available within the BCL for sending emails, the exercise seemed to be quite futile. Therefore I have updated the article so that it provides a decent introduction to network programming. There are already several introductions to network programming on the web and I wanted this to be slightly different from all of them. Whether I was successful in that endeavor is up to the end-readers to judge.

Basically with the advent of .NET, remoting and web services are the preferred means of network communication. Both mechanisms are object oriented and allows the programmer to work from a high level of abstraction. But there are often situations, where you might want to communicate with native programs that were written prior to .NET, or for some reason you cannot expect all clients to be running .NET enabled programs. That's where we can use the slightly lower level network classes for our network-based programs. There is a Socket class available but I prefer to use the TcpListener and TcpClient classes which offer a higher level of abstraction.

In this article we'll design a multi-threaded TCP server using Managed C++. This server is very simple in functionality. It accepts a line of text and returns the reversed version of the same. We'll write an MFC dialog based application that will use the MFC CSocket class to chat with our Managed C++ TCP server. Since we have used basic TCP protocol, it does not matter what language the client program is written in. Just to complete the wheel, I also include a Managed C++ version of the TCP client, which also serves to demonstrate the TcpClient class.

TCP server - MC++

We write two classes here - SpookServer which basically starts our TCP server and listens for connections, and SpookClient which is a class for handling individual client connections. The SpookServer class itself spawns off a new thread in which it waits for connections. And for each connection, it instantiates a SpookClient object which immediately spawns off a new thread to handle the particular client connection. This system of one thread per client is quite acceptable in our case because each TCP chat involves so little process flow. It's just a matter of accepting a line of text, reversing it and sending it back.

The SpookServer class

__gc class SpookServer
{
private:
    TcpListener* lis;
    Thread* t;
    ManualResetEvent* mre;
public:
    SpookServer(int port)
    {   
        mre = new ManualResetEvent(true);
        lis = new TcpListener(port);
        t = new Thread(new ThreadStart(this,&SpookServer::ServerThread));
        t->Start();     
    }
    void CloseServer()
    {
        mre->Reset();
        lis->Stop();
    }
private:
    void ServerThread()
    {
        lis->Start();
        while(mre->WaitOne(100,false))
        {
            try
            {
                TcpClient* c = lis->AcceptTcpClient();
                SpookClient* sc = new SpookClient(c);
            }
            catch(Exception* e)
            {               
            }
                    
        }
    }
};

Well, the class is rather simple, but I'll still give a quick run-through of how it works. We use a ManualResetEvent object for synchronization. This is basically so that the calling class or method has a means of stopping the server anytime, in our case this is _tmain. As you can see, we spawn off a new thread where we listen for connections. This is so that we can return control to the calling class. We use the AcceptTcpClient method of the TcpListener class which returns a TcpClient object that is a reference to the connected client. Immediately we instantiate a new SpookClient object passing the returned TcpClient object to it's constructor. And we go back to AcceptTcpClient, thus allowing a seemingly innumerable number of clients to be able to connect at any one time.

The SpookClient class

__gc class SpookClient
{
private:
    TcpClient* m_c;
    Thread* t;
public:
    SpookClient(TcpClient* c)
    {
        m_c = c;
        t = new Thread(new ThreadStart(this,&SpookClient::StartChat));
        t->IsBackground = true;
        t->Start();
    }
private:
    void StartChat()
    {
        NetworkStream* ns = m_c->GetStream();
        unsigned char buffer __gc[] = new unsigned char __gc[128];
        String* str = "";
        while(true)
        {
            int d = ns->Read(buffer,0,127);
            str = String::Concat(str,Encoding::ASCII->GetString(buffer,0,d));
            if(str->LastIndexOf('\n') != -1)
                break;
        }

        str = str->Trim();
        str = String::Concat("\n",str);     
        buffer = Encoding::ASCII->GetBytes(str);
        Array::Reverse(buffer);
        ns->Write(buffer,0,buffer->Length);     
        ns->Flush();
        ns->Close();
        m_c->Close();
    }
};

The SpookClient class has a TcpClient member and we assign the TcpClient object passed to the constructor to this member object. And we start off a new thread. Inside the thread, we first call TcpClient.GetStream to get the underlying NetworkStream object. Once we have the NetworkStream object we can use Write and Read on it to send and receive data, respectively. We simply accept a line of text, by looping till a \n is encountered and then we reverse it and send it back. Remember to Flush and Close the NetworkStream object and then call Close on the TcpClient object as well.

TCP client - MFC

void CTcpClientMFCDlg::OnBnClickedButton1()
{
    UpdateData(TRUE);
    m_text += "\n";
    CSocket s;
    s.Create();
    if(s.Connect("127.0.0.1",20248))
    {
        s.Send(m_text,m_text.GetLength());
                
        m_text = "";
        char buff[17];
        
        int i;
        
        while(m_text.Find('\n') == -1)
        {
            i=s.Receive(buff,16);
            buff[i]=0;
            m_text += buff;         
        }
        
        m_text.Trim();      
        
        UpdateData(false);      
    }
    s.Close();
}

Well, we don't do anything spectacular here. We simply send the text using CSocket::Send() and receive back the reversed line of text using CSocket::Receive. You'd think that with using network streams and all, the data would have been slightly personalized for .NET purposes, but apparently the raw data is preserved. This indicates that you can use the TcpListener and TcpClient classes for any kind of Tcp programming including popular protocols like HTTP, SMTP, POP3 etc.

TCP client - MC++

__gc class SpookClient
{
public:
    String* Reverse(String* s)
    {
        TcpClient* tl = new TcpClient("127.0.0.1",20248);
        NetworkStream* nw = tl->GetStream();
        unsigned char buffer __gc[] = new unsigned char __gc[128];
        buffer = Encoding::ASCII->GetBytes(s);
        nw->Write(buffer,0,buffer->Length);
        unsigned char retbuffer __gc[] = new unsigned char __gc[128];
        String* str = "";
        while(true)
        {
            int d = nw->Read(retbuffer,0,127);
            str = String::Concat(str,Encoding::ASCII->GetString(retbuffer,0,d));
            if(str->LastIndexOf('\n') != -1)
                break;
        }
        nw->Close();
        tl->Close();
        return str;
    }
};

I have written another class here called SpookClient, not to be confused with the other class in the server program that has the same name. This uses a TcpClient object to connect to the server, gets the underlying NetworkStream using GetStream and uses Write and Read for sending and receiving data, to and from the server. This code snippet would look very similar to our client handler class in the server program because they both demonstrate the use of the NetworkStream class.

License

This article, along with any associated source code and files, is licensed under The Code Project Open License (CPOL)

About the Author

Nishant Sivakumar


Member
Nish is a real nice guy living in Atlanta, who has been coding since 1990, when he was 13 years old. Originally from sunny Trivandrum in India, he recently moved to Atlanta from Toronto and is a little sad that he won't be able to play in snow anymore.

Nish has been a Microsoft Visual C++ MVP since October, 2002 - awfully nice of Microsoft, he thinks. He maintains an MVP tips and tricks web site - www.voidnish.com where you can find a consolidated list of his articles, writings and ideas on VC++, MFC, .NET and C++/CLI. Oh, and you might want to check out his blog on C++/CLI, MFC, .NET and a lot of other stuff - blog.voidnish.com

Nish loves reading Science Fiction, P G Wodehouse and Agatha Christie, and also fancies himself to be a decent writer of sorts. He has authored a romantic comedy Summer Love and Some more Cricket as well as a programming book – Extending MFC applications with the .NET Framework.

Nish's latest book C++/CLI in Action published by Manning Publications is now available for purchase. You can read more about the book on his blog.

Despite his wife's attempts to get him into cooking, his best effort so far has been a badly done omelette. Some day, he hopes to be a good cook, and to cook a tasty dinner for his wife.
Location: United States United States

Other popular Internet / Network articles:

Article Top
You must Sign In to use this message board.
FAQ FAQ 
 
Noise Tolerance  Layout  Per page   
 Msgs 1 to 17 of 17 (Total in Forum: 17) (Refresh)FirstPrevNext
GeneralCode Converted From /clr:oldSyntax to /Clr PinmemberDragynMaster13:44 10 Feb '07  
GeneralCSocket Receive Pinmemberfahad_iub1:09 17 Nov '04  
GeneralRe: CSocket Receive PinsussAnonymous3:08 20 Sep '05  
GeneralPOP3 Server Pinmemberdo van sung4:27 7 Nov '04  
GeneralHow about creating a Java Client PinmemberSeanV9:04 22 Jul '03  
GeneralRe: How about creating a Java Client PineditorNishant S9:20 22 Jul '03  
GeneralC++/MFC PinmemberSid Kraft10:32 27 Jul '02  
GeneralThanks! PinmemberMatt Newman6:57 2 Jul '02  
GeneralRe: Thanks! PinsubeditorNishant S7:41 2 Jul '02  
GeneralRe: Thanks! PinmemberMatt Newman12:17 2 Jul '02  
GeneralNo error checking.... PinmemberAnders Molin1:31 1 Nov '01  
GeneralRe: No error checking.... PinmemberNish [BusterBoy]1:43 1 Nov '01  
GeneralRe: No error checking.... PinmemberAnders Molin1:50 1 Nov '01  
GeneralRe: No error checking.... PinmemberNish [BusterBoy]1:56 1 Nov '01  
GeneralRe: No error checking.... PinmemberRickard Andersson5:48 12 Jun '02  
GeneralRe: No error checking.... PinmemberAnders Molin9:44 12 Jun '02  
GeneralI used something similiar PinmemberIan S13:12 28 Nov '01  

General General    News News    Question Question    Answer Answer    Joke Joke    Rant Rant    Admin Admin   

Use Ctrl+Left/Right to switch messages, Ctrl+Up/Down to switch threads.

PermaLink | Privacy | Terms of Use
Last Updated: 1 Jul 2002
Editor: Nishant Sivakumar
Copyright 2001 by Nishant Sivakumar
Everything else Copyright © CodeProject, 1999-2010
Web17 | Advertise on the Code Project