c++ udp的socket通訊使用jsoncpp傳輸
阿新 • • 發佈:2019-02-10
window平臺
使用vs2013編碼udp的socket,其中傳送和接收資料時用json協議。
步驟
- 解壓上面下載的Jsoncpp檔案,在jsoncpp-src-0.5.0/makefiles/msvc2010目錄裡找到jsoncpp.sln,使用你的編譯器開啟並編譯
- 編譯後進入資料夾當前目錄下的debug找到lib_json.lib;
- 開啟專案屬性》配置屬性》vc++目錄,找到包含目錄進行編輯,新增標頭檔案與原始檔的存放地址(如:\jsoncpp\jsoncpp-master\include\json)
- 返回vc++,找到庫目錄,新增之前找到的lib_json.lib的目錄(如:H:\jsoncpp\jsoncpp-master\makefiles\msvc2010\Debug)
- 進入c\c++》程式碼生成》執行庫,調整為MTD\MT模式
執行程式時遇到的問題
- error C4996: ‘Json::Reader::Nodes’: Use CharReader and CharReaderBuilder instead
解決:原始檔開頭新增:#pragma warning (disable: 4996) - error LNK2038: 檢測到“RuntimeLibrary”的不匹配項: 值“MTd_StaticDebug”不匹配值“MDd_DynamicDebug”
其實就是所選工程的執行庫不匹配導致的
在工程上右鍵-》屬性-》c/c++-》程式碼生成-》執行庫
有四個選項及含義分別如下:
多執行緒除錯Dll (/MDd) 對應的是MD_DynamicDebug
多執行緒Dll (/MD) 對應的是MD_DynamicRelease
多執行緒(/MT) 對應的是MD_StaticRelease
多執行緒(/MTd)對應的是MD_StaticDebug
error LNK2038: 檢測到“RuntimeLibrary”的不匹配項: 值“MTd_StaticDebug”不匹配值“MDd_DynamicDebug
從上面的錯誤提示可知,我們原來選擇的是 多執行緒除錯Dll (/MDd) 選項 ,把它改為 多執行緒(/MTd)
- warning LNK4098: 預設庫“LIBCMT”與其他庫的使用衝突
連結器->輸入->忽略特定庫: libcmt.lib
程式碼懶得改,有些變數除錯用後也沒刪除
客戶端
//#define _CRT_SECURE_NO_WARNINGS //_s版本是安全版本,像gets,gets_s,printf,printf_s之類的,但是隻能在win下執行
#pragma warning(disable:4996)
#include <memory>
#include <stdio.h>
#include <iostream>
#include <string>
#include <winsock2.h>
#pragma comment(lib,"ws2_32.lib") //編譯器設定--連結庫
using namespace std;
const int PORT = 3804;
#include <json.h>
//#pragma comment(lib, "lib_json")
int main()
{
Json::Value root;
root["key_string"] = Json::Value("value_string"); // 新建一個 Key(名為:key_string),賦予字串值:"value_string"。
root["key_number"] = Json::Value(12345); // 新建一個 Key(名為:key_number),賦予數值:12345。
root["key_boolean"] = Json::Value(true); // 新建一個 Key(名為:key_boolean),賦予bool值:false。
root["key_double"] = Json::Value(12.345); // 新建一個 Key(名為:key_double),賦予 double 值:12.345。
Json::FastWriter styled_writer;
string s=styled_writer.write(root);
//cout << root << endl;
int n;
WSADATA wd;
n = WSAStartup(MAKEWORD(2, 2), &wd);
if (n)
{
cout << "WSAStartup函式錯誤!" << endl;
return -1;
}
SOCKET sock = socket(AF_INET, SOCK_DGRAM, 0); //SOCK_DGRAM代表UDP
if (INVALID_SOCKET == sock)
{
cout << "socket建立失敗!" << endl;
return -1;
}
SOCKADDR_IN sa1;
sa1.sin_family = AF_INET;//AF_INET 就是區域網,
sa1.sin_port = htons(PORT);
sa1.sin_addr.S_un.S_addr = inet_addr("172.05.187.1");
//可以不用bind();系統預設隨機分配
/*sockaddr_in sa = { AF_INET }; //隨機分配一個埠 (僅初始化一個值,讓客戶端自己分配埠)
n = bind(sock, (sockaddr*)&sa, sizeof(sa));//要sa1分開,不然,不能連線。
if (n == SOCKET_ERROR)
{
cout << "bind函式失敗!" << endl;
cout << "錯誤碼是:" << WSAGetLastError() << endl;
return -1;
}*/
//sockaddr_in sa1 = { AF_INET, htons(8009) }; //必須使用htons,因為網路位元組序的緣故
//sa1.sin_addr.S_un.S_addr = inet_addr("192.168.191.1");//指定要發往的IP和埠號
char c[256];
strcpy(c, s.c_str());
cout << c<< endl;
int n1 = 1;
while (n1)
{
//sendto(sock, s.c_str(), sizeof(s), 0, (sockaddr*)&sa1, sizeof(sa1)); //傳送資料
cout << strlen(c) << endl;
sendto(sock, c, sizeof(c), 0, (sockaddr*)&sa1, sizeof(sa1));
fflush(stdin);//清空快取區,移植性差,vs2015不能用
//gets(c);//#define _CRT_SECURE_NO_WARNINGS ,否則會報錯。 不能使用gets,可用fgets(buf, BUFSIZE, stdin);
cin >> n1;
}
return 0;
}
伺服器端
#pragma warning (disable: 4996)
#include <cstdio>
#include <iostream>
#include <string>
#include <winsock2.h>
#pragma comment(lib,"ws2_32.lib")
using namespace std;
const int PORT = 3804;
#include <json.h>
int main()
{
int n;
WSADATA wd;
n = WSAStartup(MAKEWORD(2, 2), &wd);
if (n)
{
cout << "WSAStartup函式錯誤!" << endl;
return -1;
}
SOCKET sock = socket(AF_INET, SOCK_DGRAM, 0);
if (INVALID_SOCKET == sock)
{
cout << "socket建立失敗!" << endl;
cout << "錯誤碼是:" << WSAGetLastError() << endl;
return -1;
}
sockaddr_in sa = { AF_INET, htons(PORT) };
/*
SOCKADDR_IN sa;
sa.sin_family = AF_INET;
sa.sin_port = htons(PORT);
sa.sin_addr.S_un.S_addr = htonl(INADDR_ANY);
*/
n = bind(sock, (sockaddr*)&sa, sizeof(sa));
if (n == SOCKET_ERROR)
{
cout << "bind繫結埠失敗!" << endl;
cout << "錯誤碼是:" << WSAGetLastError() << endl;
return -1;
}
else
{
cout << "埠釋出成功:" << PORT << endl;
}
char s[256];
while (true)
{
n = recv(sock, s, sizeof(s), 0); //recv返回的是實際copy的位元組數
//s[n] = '\0';
cout << strlen(s) << endl;
Json::Value va;
Json::Reader re;
n = re.parse(s, va);
cout << s << endl;
cout << va << endl;
string s = va["key_string"].asString();
int i = va["key_number"].asInt();
bool b = va["key_boolean"].asBool();
double d = va["key_double"].asDouble();
cout <<s <<i<<b <<d<< endl;
}
return 0;
}