- Paper: "High speed and accurate OMR using any typical scanner", Monfared, H., Ghazisaedy, M., 6th INTERNATIONAL CONFERENCE ON ELECTRICAL ENGINEERING, 2008
- Technical report : "E-Government, design and architecture", (2005) available at http://irit.ir/uploads/EG_Master_plan.doc
- Feasibility study : "High performance event driven architecture for modern core banking systems", Tejarat Bank. (2014)
- Interview: "Detecting known and unknown web attacks ; interview with CEO of Imen Rayaneh Shargh, producer of first Web Application Firewall in Iran", Information society security Journal. ( 2013 )
- Technical report: "Machine learning techniques for anomaly detection". ( 2012 )
Coding ways
Tutorials around algorithms, distributed programming and patterns in C++
Friday, March 7, 2014
Selected publications and technical reports
Thursday, May 16, 2013
How to use named group feature in Google RE2 regular expression engine
After some googling and then submitting a question in re2-dev google group, I couldn't find any useful help about using "named groups" in RE2.
for example if you want to extract protocol, url path and query string parts from a given url string, it's good idea to use named group in RE2:
sample urls :
http://localhost:8080/hello?name=Hassan
a sample but not perfect RegEx may be :
^(?P<PROTO>https?)://(?P<URL>.+)\?(?P<QUERY>.+)?$
The best way for getting PROTO, URL and QUERY values is using named group capturing feature in RE2.
this is the Code I wrote for this :
for example if you want to extract protocol, url path and query string parts from a given url string, it's good idea to use named group in RE2:
sample urls :
http://localhost:8080/hello?name=Hassan
a sample but not perfect RegEx may be :
^(?P<PROTO>https?)://(?P<URL>.+)\?(?P<QUERY>.+)?$
The best way for getting PROTO, URL and QUERY values is using named group capturing feature in RE2.
this is the Code I wrote for this :
x bool FindNamedGroups(const std::string &p_regex,const std::string &p_text,std::map<std::string,std::string> *p_group_value) { p_group_value->clear(); RE2 rx(p_regex); if(!rx.ok()) { std::cerr << "Invalid Regular Expression :" << p_regex << std::endl; return false; } size_t named_grp_size =rx.NumberOfCapturingGroups(); if(named_grp_size>10) { std::cerr << "No support for more than 10 named groups :" << named_grp_size<< std::endl; return false; } const std::map<std::string,int> &grp_to_idx=rx.NamedCapturingGroups(); RE2::Arg args[10]; std::string vars[10]; const RE2::Arg * const p_args[10]={&args[0],&args[1],&args[2],&args[3],&args[4],&args[5],&args[6],&args[7],&args[8],&args[9]}; int var_count=0; for(var_count=0;var_count<10;var_count++) args[var_count]=&vars[var_count]; re2::StringPiece sp_input(p_text); //after running following function. matched groups value will be stored in p_args which point to args which point to vars! bool found= RE2::FindAndConsumeN(&sp_input,rx,p_args,named_grp_size); if(!found) { return false ; } std::map<std::string,int>::const_iterator iter_grps=grp_to_idx.cbegin(); for(;iter_grps!=grp_to_idx.cend();++iter_grps) { (*p_group_value)[iter_grps->first]=vars[iter_grps->second-1]; } return true; }
//////////// USAGE ////////////////
FindNamedGroups("^(?P<PROTO>https?)://(?P<URL>.+)\\?(?P<QUERY>.+)?$","http://localhost:8080/hello?name=Hassan",&g_v); iter=g_v.cbegin(); for(;iter!=g_v.cend();++iter) std::cout << iter->first << " = " << iter->second << std::endl; x
Sunday, November 25, 2012
Toggle collapsing and formatting code in Visual Studio
Few useful tips for Visual Studio developers :
1- for collapse all methods in VS Editor : Ctrl+M+O
2- for un-collapse all methods : Ctrl+M+P
3- for toggle collapse for current method : Ctrl+M+M
4- for formatting code : Ctrl+A / Ctrl+K+F
Let's go back to programming ProWeb !
1- for collapse all methods in VS Editor : Ctrl+M+O
2- for un-collapse all methods : Ctrl+M+P
3- for toggle collapse for current method : Ctrl+M+M
4- for formatting code : Ctrl+A / Ctrl+K+F
Let's go back to programming ProWeb !
Sunday, August 12, 2012
Hanoi tower in C++
Here I'm sending a recursive solution for solving Hanoi tower problem with 3 columns and n disks.
Columns are in following order :
|| || ||
|| || ||
|| || ||
|| || ||
==== ==== ====
c_from c_other c_to
The problem is moving n disks from column c_from to column c_to ( we can't put bigger disks over smaller disks in a column )
Solution is :
1- moving n-1 disks from c_from to c_other
2- moving last disk from c_from to c_to
3- moving n-1 disks from c_other to c_to
.
here is simple code in C++ to solve the problem:
Columns are in following order :
|| || ||
|| || ||
|| || ||
|| || ||
==== ==== ====
c_from c_other c_to
The problem is moving n disks from column c_from to column c_to ( we can't put bigger disks over smaller disks in a column )
Solution is :
1- moving n-1 disks from c_from to c_other
2- moving last disk from c_from to c_to
3- moving n-1 disks from c_other to c_to
.
here is simple code in C++ to solve the problem:
class HanoiTower { private : int N; stack<int> towers[3]; public: enum TOWERS { A=0, B=1, C=2 }; HanoiTower(int pN) { N=pN; for(int i=N-1;i>=0;i--) towers[A].Push(i); } void MoveRec(int pN,TOWERS tFrom,TOWERS tTo) { if(pN==0) { int disk=towers[tFrom].Top(); towers[tFrom].Pop(); towers[tTo].Push(disk); cout <<" move "<<disk<<" from "<<tFrom<< " to "<< tTo<<endl; //cout <<disk+1<<tFrom+1<<tTo+1<<endl; return; } TOWERS other=(TOWERS)(3-tFrom-tTo); MoveRec(pN-1,tFrom,other); MoveRec(0,tFrom,tTo); MoveRec(pN-1,other,tTo); } }; // usage : HanoiTower hn(15); hn.MoveRec(14,HanoiTower::TOWERS::A,HanoiTower::TOWERS::C);
Saturday, August 11, 2012
Send Email in C++ ( SMTP Client in C++ )
Boost! Yes! It's a good library for you to help you develop cross-platform C++ applications; headache free.
among all libraries in boost, boost::asio is for implementing asynchronous input output operations and working with network.
After having a look at boost asio samples and SMTP RFC : http://www.ietf.org/rfc/rfc2821.txt I implemented this simple class to be used in your C++ programs.
the only tricky part is base64 encoding which is required for sending user/password to SMTP Server.
the only tricky part is base64 encoding which is required for sending user/password to SMTP Server.
#include <iostream> #include <istream> #include <ostream> #include <string> #include <boost/asio.hpp> #include <boost/bind.hpp> #include <boost/lexical_cast.hpp> #include <boost/archive/iterators/base64_from_binary.hpp> #include <boost/archive/iterators/transform_width.hpp> #include <boost/archive/iterators/ostream_iterator.hpp> using boost::asio::ip::tcp; using namespace boost::archive::iterators; typedef base64_from_binary<transform_width<const char *,6,8> > base64_text; class SMTPClient { public: SMTPClient(std::string pServer,unsigned int pPort,std::string pUser,std::string pPassword): mServer(pServer),mPort(pPort),mUserName(pUser),mPassword(pPassword),mSocket(mIOService),mResolver(mIOService) { tcp::resolver::query qry(mServer,boost::lexical_cast<std::string>( mPort )); mResolver.async_resolve(qry,boost::bind(&SMTPClient::handleResolve,this,boost::asio::placeholders::error, boost::asio::placeholders::iterator)); } bool Send(std::string pFrom,std::string pTo,std::string pSubject,std::string pMessage) { mFrom=pFrom; mTo=pTo; mSubject=pSubject; mMessage=pMessage; mIOService.run(); return mHasError; } private: std::string encodeBase64(std::string pData) { std::stringstream os; size_t sz=pData.size(); std::copy(base64_text(pData.c_str()),base64_text(pData.c_str()+sz),ostream_iterator<char>(os)); return os.str(); } void handleResolve(const boost::system::error_code& err,tcp::resolver::iterator endpoint_iterator) { if(!err) { tcp::endpoint endpoint=*endpoint_iterator; mSocket.async_connect(endpoint, boost::bind(&SMTPClient::handleConnect,this,boost::asio::placeholders::error,++endpoint_iterator)); } else { mHasError=true; mErrorMsg= err.message(); } } void writeLine(std::string pData) { std::ostream req_strm(&mRequest); req_strm << pData << "\r\n"; boost::asio::write(mSocket,mRequest); req_strm.clear(); } void handleConnect(const boost::system::error_code& err,tcp::resolver::iterator endpoint_iterator) { if (!err) { // The connection was successful. Send the request. std::ostream req_strm(&mRequest); writeLine("EHLO "+mServer); writeLine("AUTH LOGIN"); writeLine(encodeBase64(mUserName)); writeLine(encodeBase64(mPassword)); writeLine( "MAIL FROM:<"+mFrom+">"); writeLine( "RCPT TO:<"+mTo+">"); writeLine( "DATA"); writeLine( "SUBJECT:"+mSubject); writeLine( "From:"+mFrom); writeLine( "To:"+mTo); writeLine( ""); writeLine( mMessage ); writeLine( ".\r\n"); } else { mHasError=true; mErrorMsg= err.message(); } } std::string mServer; std::string mUserName; std::string mPassword; std::string mFrom; std::string mTo; std::string mSubject; std::string mMessage; unsigned int mPort; boost::asio::io_service mIOService; tcp::resolver mResolver; tcp::socket mSocket; boost::asio::streambuf mRequest; boost::asio::streambuf mResponse; bool mHasError; std::string mErrorMsg; };Yest ! that's it. you are free to copy and paste this code into your application:
SMTPClient mailc("yoursmtpserver.com",25,"user@yourdomain.com","password"); mailc.Send("from@yourdomain.com","to@somewhere.com","subject","Hello from C++ SMTP Client!");
Saturday, August 4, 2012
How to find DOS attackers in Linux
#netstat -anp | grep 'tcp\|udp' | awk '{print $5}' | cut -d: -f1 | grep -v "^$" | uniq -c | sort -nr
output will be ;
Number_of_connections Client_IP
------------------------------------- ------------------
output will be ;
Number_of_connections Client_IP
------------------------------------- ------------------
Thursday, August 2, 2012
Implementing thread safe queue in C++11
Concurrency again!
I'm going to show you how to implement thread safe queue using lock/mutex and condition variable based synchronization in C++11.
As you know protecting shared data in multi-threaded environment is very important.
at this following sample I demonstrated how to use mutexes and condition variables to write a thread-safe queue.
I put all description as comments to help you understand better.
I'm going to show you how to implement thread safe queue using lock/mutex and condition variable based synchronization in C++11.
As you know protecting shared data in multi-threaded environment is very important.
at this following sample I demonstrated how to use mutexes and condition variables to write a thread-safe queue.
I put all description as comments to help you understand better.
#include <thread> #include <queue> #include <mutex> #include <condition_variable> #include <memory> template <class T> class tsqueue { private : std::queue<T> mData; // check empty() method for description of "mutable" mutable std::mutex mMut; std::condition_variable mEmptyCondition; void wait_for_debugging() { std::this_thread::sleep_for(std::chrono::milliseconds(200+rand()%200)); } public: void Push(T pValue) { // block execution here, if other thread already locked mMute! std::lock_guard<std::mutex> lock(mMut); // if weare here no other thread is owned/locked mMute. so we can modify the internal data mData.push(pValue); // OK! notify one of threads which was blocked because , queue empty and make them happy! mEmptyCondition.notify_one(); wait_for_debugging(); } // //lock.unlock(); !!!! does not have such method! // now mMute is unlocked in destcutor of lock_guard! std::shared_ptr<T> Pop() { // own the mMute like in Push method excep that lock can be unlocked manually without need to destructor of lock! std::unique_lock<std::mutex> lock(mMut); // if we are here. mMute is locked and no other thread can access/modify the data! // wait() method first checks if mData is not empty, allowes execution to go on. else : // unlocks the mMut and waits for signla. // because mMute is released other threads have a chance to Push new data into queue // ... in notify this condition variable! mEmptyCondition.wait(lock,[this]{ return !mData.empty();}); // if we are are here, mData is not empty and mMut is locked ! // be careful ! std::shared_ptr<T> ret(new T(mData.front()) performs 2 memory allocation! std::shared_ptr<T> ret=std::shared_ptr<T>(std::make_shared<T>(mData.front())); mData.pop(); wait_for_debugging(); return ret; } std::shared_ptr<T> TryPop() { std::lock_guard<std::mutex> lock(mMut); if(mData.empty()) return std::shared_ptr<T>(); // null std::shared_ptr<T> ret=std::shared_ptr<T>(std::make_shared<T>(mData.front())); mData.pop(); wait_for_debugging(); return ret; } bool Empty() const { // this function is "const", so why are we allowed to modify a member vaiable "mMute"?.... //we can, because we marked it "mutable"! see definition of mMute. std::lock_guard<std::mutex> lock(mMut); return mData.empty(); } }; void test_tsq() { tsqueue<int> tsq; int N=100; std::thread thrpush([&]{ for(int i=0;i<N;i++) { tsq.Push(i); std::cout <<i<<" pushed \n"; } }); std::thread thrpop([&]{ for(int i=0;i<N/2;i++) { std::cout <<"popping\n"; std::shared_ptr<int> data=tsq.Pop(); std::cout <<i<<": poped "<<*data<<"\n"; } }); std::thread thrtrypop([&]{ int i=0; for(;i<N/2;) { std::cout <<"trying for pop\n"; std::shared_ptr<int> data=tsq.TryPop(); if(data.get()!=NULL) { std::cout <<i++<<": try poped "<<*data<<"\n"; } else std::cout <<"failed trying pop \n"; } }); thrpush.join(); thrpop.join(); thrtrypop.join(); }The only reminded point to be notices about mEmptyCondition.wait(...) method is that if this condition variable was notified after any push() method from other thread, It checks the mData.empty() condition for avoiding race condition. because there is a chance for "pop"ing last inserted item by other thread. So even being notified is not enough reason to continuing execution and modifying data. ( in this case because of notifyone(). only one condition variable is notified among all waiting condition variables)
Labels:
C++11
,
Concurrent programming
,
Queue
,
Thread safety
Subscribe to:
Posts
(
Atom
)
Friday, March 7, 2014
Selected publications and technical reports
- Paper: "High speed and accurate OMR using any typical scanner", Monfared, H., Ghazisaedy, M., 6th INTERNATIONAL CONFERENCE ON ELECTRICAL ENGINEERING, 2008
- Technical report : "E-Government, design and architecture", (2005) available at http://irit.ir/uploads/EG_Master_plan.doc
- Feasibility study : "High performance event driven architecture for modern core banking systems", Tejarat Bank. (2014)
- Interview: "Detecting known and unknown web attacks ; interview with CEO of Imen Rayaneh Shargh, producer of first Web Application Firewall in Iran", Information society security Journal. ( 2013 )
- Technical report: "Machine learning techniques for anomaly detection". ( 2012 )
Thursday, May 16, 2013
How to use named group feature in Google RE2 regular expression engine
After some googling and then submitting a question in re2-dev google group, I couldn't find any useful help about using "named groups" in RE2.
for example if you want to extract protocol, url path and query string parts from a given url string, it's good idea to use named group in RE2:
sample urls :
http://localhost:8080/hello?name=Hassan
a sample but not perfect RegEx may be :
^(?P<PROTO>https?)://(?P<URL>.+)\?(?P<QUERY>.+)?$
The best way for getting PROTO, URL and QUERY values is using named group capturing feature in RE2.
this is the Code I wrote for this :
for example if you want to extract protocol, url path and query string parts from a given url string, it's good idea to use named group in RE2:
sample urls :
http://localhost:8080/hello?name=Hassan
a sample but not perfect RegEx may be :
^(?P<PROTO>https?)://(?P<URL>.+)\?(?P<QUERY>.+)?$
The best way for getting PROTO, URL and QUERY values is using named group capturing feature in RE2.
this is the Code I wrote for this :
x bool FindNamedGroups(const std::string &p_regex,const std::string &p_text,std::map<std::string,std::string> *p_group_value) { p_group_value->clear(); RE2 rx(p_regex); if(!rx.ok()) { std::cerr << "Invalid Regular Expression :" << p_regex << std::endl; return false; } size_t named_grp_size =rx.NumberOfCapturingGroups(); if(named_grp_size>10) { std::cerr << "No support for more than 10 named groups :" << named_grp_size<< std::endl; return false; } const std::map<std::string,int> &grp_to_idx=rx.NamedCapturingGroups(); RE2::Arg args[10]; std::string vars[10]; const RE2::Arg * const p_args[10]={&args[0],&args[1],&args[2],&args[3],&args[4],&args[5],&args[6],&args[7],&args[8],&args[9]}; int var_count=0; for(var_count=0;var_count<10;var_count++) args[var_count]=&vars[var_count]; re2::StringPiece sp_input(p_text); //after running following function. matched groups value will be stored in p_args which point to args which point to vars! bool found= RE2::FindAndConsumeN(&sp_input,rx,p_args,named_grp_size); if(!found) { return false ; } std::map<std::string,int>::const_iterator iter_grps=grp_to_idx.cbegin(); for(;iter_grps!=grp_to_idx.cend();++iter_grps) { (*p_group_value)[iter_grps->first]=vars[iter_grps->second-1]; } return true; }
//////////// USAGE ////////////////
FindNamedGroups("^(?P<PROTO>https?)://(?P<URL>.+)\\?(?P<QUERY>.+)?$","http://localhost:8080/hello?name=Hassan",&g_v); iter=g_v.cbegin(); for(;iter!=g_v.cend();++iter) std::cout << iter->first << " = " << iter->second << std::endl; x
Sunday, November 25, 2012
Toggle collapsing and formatting code in Visual Studio
Few useful tips for Visual Studio developers :
1- for collapse all methods in VS Editor : Ctrl+M+O
2- for un-collapse all methods : Ctrl+M+P
3- for toggle collapse for current method : Ctrl+M+M
4- for formatting code : Ctrl+A / Ctrl+K+F
Let's go back to programming ProWeb !
1- for collapse all methods in VS Editor : Ctrl+M+O
2- for un-collapse all methods : Ctrl+M+P
3- for toggle collapse for current method : Ctrl+M+M
4- for formatting code : Ctrl+A / Ctrl+K+F
Let's go back to programming ProWeb !
Sunday, August 12, 2012
Hanoi tower in C++
Here I'm sending a recursive solution for solving Hanoi tower problem with 3 columns and n disks.
Columns are in following order :
|| || ||
|| || ||
|| || ||
|| || ||
==== ==== ====
c_from c_other c_to
The problem is moving n disks from column c_from to column c_to ( we can't put bigger disks over smaller disks in a column )
Solution is :
1- moving n-1 disks from c_from to c_other
2- moving last disk from c_from to c_to
3- moving n-1 disks from c_other to c_to
.
here is simple code in C++ to solve the problem:
Columns are in following order :
|| || ||
|| || ||
|| || ||
|| || ||
==== ==== ====
c_from c_other c_to
The problem is moving n disks from column c_from to column c_to ( we can't put bigger disks over smaller disks in a column )
Solution is :
1- moving n-1 disks from c_from to c_other
2- moving last disk from c_from to c_to
3- moving n-1 disks from c_other to c_to
.
here is simple code in C++ to solve the problem:
class HanoiTower { private : int N; stack<int> towers[3]; public: enum TOWERS { A=0, B=1, C=2 }; HanoiTower(int pN) { N=pN; for(int i=N-1;i>=0;i--) towers[A].Push(i); } void MoveRec(int pN,TOWERS tFrom,TOWERS tTo) { if(pN==0) { int disk=towers[tFrom].Top(); towers[tFrom].Pop(); towers[tTo].Push(disk); cout <<" move "<<disk<<" from "<<tFrom<< " to "<< tTo<<endl; //cout <<disk+1<<tFrom+1<<tTo+1<<endl; return; } TOWERS other=(TOWERS)(3-tFrom-tTo); MoveRec(pN-1,tFrom,other); MoveRec(0,tFrom,tTo); MoveRec(pN-1,other,tTo); } }; // usage : HanoiTower hn(15); hn.MoveRec(14,HanoiTower::TOWERS::A,HanoiTower::TOWERS::C);
Saturday, August 11, 2012
Send Email in C++ ( SMTP Client in C++ )
Boost! Yes! It's a good library for you to help you develop cross-platform C++ applications; headache free.
among all libraries in boost, boost::asio is for implementing asynchronous input output operations and working with network.
After having a look at boost asio samples and SMTP RFC : http://www.ietf.org/rfc/rfc2821.txt I implemented this simple class to be used in your C++ programs.
the only tricky part is base64 encoding which is required for sending user/password to SMTP Server.
the only tricky part is base64 encoding which is required for sending user/password to SMTP Server.
#include <iostream> #include <istream> #include <ostream> #include <string> #include <boost/asio.hpp> #include <boost/bind.hpp> #include <boost/lexical_cast.hpp> #include <boost/archive/iterators/base64_from_binary.hpp> #include <boost/archive/iterators/transform_width.hpp> #include <boost/archive/iterators/ostream_iterator.hpp> using boost::asio::ip::tcp; using namespace boost::archive::iterators; typedef base64_from_binary<transform_width<const char *,6,8> > base64_text; class SMTPClient { public: SMTPClient(std::string pServer,unsigned int pPort,std::string pUser,std::string pPassword): mServer(pServer),mPort(pPort),mUserName(pUser),mPassword(pPassword),mSocket(mIOService),mResolver(mIOService) { tcp::resolver::query qry(mServer,boost::lexical_cast<std::string>( mPort )); mResolver.async_resolve(qry,boost::bind(&SMTPClient::handleResolve,this,boost::asio::placeholders::error, boost::asio::placeholders::iterator)); } bool Send(std::string pFrom,std::string pTo,std::string pSubject,std::string pMessage) { mFrom=pFrom; mTo=pTo; mSubject=pSubject; mMessage=pMessage; mIOService.run(); return mHasError; } private: std::string encodeBase64(std::string pData) { std::stringstream os; size_t sz=pData.size(); std::copy(base64_text(pData.c_str()),base64_text(pData.c_str()+sz),ostream_iterator<char>(os)); return os.str(); } void handleResolve(const boost::system::error_code& err,tcp::resolver::iterator endpoint_iterator) { if(!err) { tcp::endpoint endpoint=*endpoint_iterator; mSocket.async_connect(endpoint, boost::bind(&SMTPClient::handleConnect,this,boost::asio::placeholders::error,++endpoint_iterator)); } else { mHasError=true; mErrorMsg= err.message(); } } void writeLine(std::string pData) { std::ostream req_strm(&mRequest); req_strm << pData << "\r\n"; boost::asio::write(mSocket,mRequest); req_strm.clear(); } void handleConnect(const boost::system::error_code& err,tcp::resolver::iterator endpoint_iterator) { if (!err) { // The connection was successful. Send the request. std::ostream req_strm(&mRequest); writeLine("EHLO "+mServer); writeLine("AUTH LOGIN"); writeLine(encodeBase64(mUserName)); writeLine(encodeBase64(mPassword)); writeLine( "MAIL FROM:<"+mFrom+">"); writeLine( "RCPT TO:<"+mTo+">"); writeLine( "DATA"); writeLine( "SUBJECT:"+mSubject); writeLine( "From:"+mFrom); writeLine( "To:"+mTo); writeLine( ""); writeLine( mMessage ); writeLine( ".\r\n"); } else { mHasError=true; mErrorMsg= err.message(); } } std::string mServer; std::string mUserName; std::string mPassword; std::string mFrom; std::string mTo; std::string mSubject; std::string mMessage; unsigned int mPort; boost::asio::io_service mIOService; tcp::resolver mResolver; tcp::socket mSocket; boost::asio::streambuf mRequest; boost::asio::streambuf mResponse; bool mHasError; std::string mErrorMsg; };Yest ! that's it. you are free to copy and paste this code into your application:
SMTPClient mailc("yoursmtpserver.com",25,"user@yourdomain.com","password"); mailc.Send("from@yourdomain.com","to@somewhere.com","subject","Hello from C++ SMTP Client!");
Saturday, August 4, 2012
How to find DOS attackers in Linux
#netstat -anp | grep 'tcp\|udp' | awk '{print $5}' | cut -d: -f1 | grep -v "^$" | uniq -c | sort -nr
output will be ;
Number_of_connections Client_IP
------------------------------------- ------------------
output will be ;
Number_of_connections Client_IP
------------------------------------- ------------------
Thursday, August 2, 2012
Implementing thread safe queue in C++11
Concurrency again!
I'm going to show you how to implement thread safe queue using lock/mutex and condition variable based synchronization in C++11.
As you know protecting shared data in multi-threaded environment is very important.
at this following sample I demonstrated how to use mutexes and condition variables to write a thread-safe queue.
I put all description as comments to help you understand better.
I'm going to show you how to implement thread safe queue using lock/mutex and condition variable based synchronization in C++11.
As you know protecting shared data in multi-threaded environment is very important.
at this following sample I demonstrated how to use mutexes and condition variables to write a thread-safe queue.
I put all description as comments to help you understand better.
#include <thread> #include <queue> #include <mutex> #include <condition_variable> #include <memory> template <class T> class tsqueue { private : std::queue<T> mData; // check empty() method for description of "mutable" mutable std::mutex mMut; std::condition_variable mEmptyCondition; void wait_for_debugging() { std::this_thread::sleep_for(std::chrono::milliseconds(200+rand()%200)); } public: void Push(T pValue) { // block execution here, if other thread already locked mMute! std::lock_guard<std::mutex> lock(mMut); // if weare here no other thread is owned/locked mMute. so we can modify the internal data mData.push(pValue); // OK! notify one of threads which was blocked because , queue empty and make them happy! mEmptyCondition.notify_one(); wait_for_debugging(); } // //lock.unlock(); !!!! does not have such method! // now mMute is unlocked in destcutor of lock_guard! std::shared_ptr<T> Pop() { // own the mMute like in Push method excep that lock can be unlocked manually without need to destructor of lock! std::unique_lock<std::mutex> lock(mMut); // if we are here. mMute is locked and no other thread can access/modify the data! // wait() method first checks if mData is not empty, allowes execution to go on. else : // unlocks the mMut and waits for signla. // because mMute is released other threads have a chance to Push new data into queue // ... in notify this condition variable! mEmptyCondition.wait(lock,[this]{ return !mData.empty();}); // if we are are here, mData is not empty and mMut is locked ! // be careful ! std::shared_ptr<T> ret(new T(mData.front()) performs 2 memory allocation! std::shared_ptr<T> ret=std::shared_ptr<T>(std::make_shared<T>(mData.front())); mData.pop(); wait_for_debugging(); return ret; } std::shared_ptr<T> TryPop() { std::lock_guard<std::mutex> lock(mMut); if(mData.empty()) return std::shared_ptr<T>(); // null std::shared_ptr<T> ret=std::shared_ptr<T>(std::make_shared<T>(mData.front())); mData.pop(); wait_for_debugging(); return ret; } bool Empty() const { // this function is "const", so why are we allowed to modify a member vaiable "mMute"?.... //we can, because we marked it "mutable"! see definition of mMute. std::lock_guard<std::mutex> lock(mMut); return mData.empty(); } }; void test_tsq() { tsqueue<int> tsq; int N=100; std::thread thrpush([&]{ for(int i=0;i<N;i++) { tsq.Push(i); std::cout <<i<<" pushed \n"; } }); std::thread thrpop([&]{ for(int i=0;i<N/2;i++) { std::cout <<"popping\n"; std::shared_ptr<int> data=tsq.Pop(); std::cout <<i<<": poped "<<*data<<"\n"; } }); std::thread thrtrypop([&]{ int i=0; for(;i<N/2;) { std::cout <<"trying for pop\n"; std::shared_ptr<int> data=tsq.TryPop(); if(data.get()!=NULL) { std::cout <<i++<<": try poped "<<*data<<"\n"; } else std::cout <<"failed trying pop \n"; } }); thrpush.join(); thrpop.join(); thrtrypop.join(); }The only reminded point to be notices about mEmptyCondition.wait(...) method is that if this condition variable was notified after any push() method from other thread, It checks the mData.empty() condition for avoiding race condition. because there is a chance for "pop"ing last inserted item by other thread. So even being notified is not enough reason to continuing execution and modifying data. ( in this case because of notifyone(). only one condition variable is notified among all waiting condition variables)
Labels:
C++11
,
Concurrent programming
,
Queue
,
Thread safety
Subscribe to:
Posts
(
Atom
)