Форум программистов, компьютерный форум CyberForum.ru

Использование CreateThread - C++

Восстановить пароль Регистрация
 
Рейтинг: Рейтинг темы: голосов - 35, средняя оценка - 4.71
alexey31415
 Аватар для alexey31415
59 / 59 / 3
Регистрация: 16.05.2010
Сообщений: 632
14.07.2012, 18:13     Использование CreateThread #1
Создаю функцию для потока,запускаю её с помощью _beginthread и всё нормально работает,но если использовать CreateThread,то возникают ошибки
функция,используемая с
_beginthread

C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
#include "StdAfx.h"
 
//-----------------------------------------------------------------------
//функция для установления соединения,получения данных и записи их в файл
//-----------------------------------------------------------------------
 
//объявляем буфер, для хранения возможной ошибки, размер определяется в самой библиотеке
static char errorBuffer[CURL_ERROR_SIZE];
//объявляем буфер принимаемых данных
//static string buffer;
 
 
typedef struct
{
    string buffer;
}DATA, *PDATA;
 
//функция обратного вызова
static int writer(char *data, size_t size, size_t nmemb, string *buffer)
{
  //переменная - результат, по умолчанию нулевая
  int result = 0;
 
  if (buffer != NULL)
  {
    //добавляем к буферу строки из data, в количестве nmemb
    buffer->append(data, size * nmemb);
    //вычисляем объем принятых данных
    result = size * nmemb;
  }
  //вовзращаем результат
  return result;
}
 
 
 
void connect(PVOID pvoid)
{
    PINFO pinfo = (PINFO)pvoid;
 
    setlocale(0,"");
 
    DWORD dwTlsIndex = TlsAlloc();
 
    TlsSetValue(dwTlsIndex,GlobalAlloc(GPTR,sizeof(DATA)));
 
    PDATA pdata;
 
    pdata = (PDATA)TlsGetValue(dwTlsIndex);
 
    //необходимые CURL объекты
    CURL *curl;
    CURLcode result;
    //инициализируем curl
    curl = curl_easy_init();
 
    //проверяем результат инициализации
  if (curl)
    {
    curl_easy_setopt(curl, CURLOPT_ERRORBUFFER, errorBuffer);
      //задаем все необходимые опции
      curl_easy_setopt(curl, CURLOPT_URL,pinfo->url);
     
      curl_easy_setopt(curl, CURLOPT_HEADER, 0);         // не возвращает заголовки
      curl_easy_setopt(curl, CURLOPT_FOLLOWLOCATION, 1);   // переходит по редиректам
      curl_easy_setopt(curl, CURLOPT_ENCODING, "");        // обрабатывает все кодировки
 
      curl_easy_setopt(curl, CURLOPT_CONNECTTIMEOUT, 120); // таймаут соединения
      curl_easy_setopt(curl, CURLOPT_TIMEOUT, 120);        // таймаут ответа
      curl_easy_setopt(curl, CURLOPT_MAXREDIRS, 10);       // останавливаться после 10-ого редиректа
 
 
      //указываем функцию обратного вызова для записи получаемых данных
      curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, writer);
      //указываем куда записывать принимаемые данные
      curl_easy_setopt(curl, CURLOPT_WRITEDATA, &(pdata->buffer));
      //запускаем выполнение задачи
      result = curl_easy_perform(curl);
 
      //проверяем успешность выполнения операции
      if (result == CURLE_OK)
        {
        string fn(pinfo->NameOfFile);
        ofstream f(fn.c_str());
        if(!f) { MessageBox(pinfo->hwnd, "Can't create file","Error",MB_OK);}
        else   { f<<pdata->buffer<<endl; f.close();}
        }
  }
    else
    //выводим сообщение об ошибке
    MessageBox(pinfo->hwnd, "Can't connect","Error",MB_OK);
 
     
  //завершаем сессию
  curl_easy_cleanup(curl);
 
  TlsFree(dwTlsIndex);
 
  _endthread();
}
,а вот с
CreateThread

C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
#include "StdAfx.h"
 
//-----------------------------------------------------------------------
//функция для установления соединения,получения данных и записи их в файл
//-----------------------------------------------------------------------
 
typedef struct
{
    string buffer;
    char errorBuffer[CURL_ERROR_SIZE];
    CURL *curl;
    CURLcode result;
}DATA, *PDATA;
 
//функция обратного вызова
static int writer(char *data, size_t size, size_t nmemb, string *buffer)
{
  //переменная - результат, по умолчанию нулевая
  int result = 0;
 
  if (buffer != NULL)
  {
    //добавляем к буферу строки из data, в количестве nmemb
    buffer->append(data, size * nmemb);
    //вычисляем объем принятых данных
    result = size * nmemb;
  }
  //вовзращаем результат
  return result;
}
 
 
DWORD WINAPI connect(LPVOID pvoid)
{
    PINFO pinfo = (PINFO)pvoid;
 
    setlocale(0,"");
 
    DWORD dwTlsIndex = TlsAlloc();
 
    TlsSetValue(dwTlsIndex,GlobalAlloc(GPTR,sizeof(DATA)));
 
    PDATA pdata;
 
    pdata = (PDATA)TlsGetValue(dwTlsIndex);
    
    
    //инициализируем curl
    pdata->curl = curl_easy_init();
 
    //проверяем результат инициализации
  if (pdata->curl)
    {
    curl_easy_setopt(pdata->curl, CURLOPT_ERRORBUFFER, pdata->errorBuffer);
      //задаем все необходимые опции
      curl_easy_setopt(pdata->curl, CURLOPT_URL,pinfo->url);
     
      curl_easy_setopt(pdata->curl, CURLOPT_HEADER, 0);         // не возвращает заголовки
      curl_easy_setopt(pdata->curl, CURLOPT_FOLLOWLOCATION, 1);   // переходит по редиректам
      curl_easy_setopt(pdata->curl, CURLOPT_ENCODING, "");        // обрабатывает все кодировки
 
      curl_easy_setopt(pdata->curl, CURLOPT_CONNECTTIMEOUT, 120); // таймаут соединения
      curl_easy_setopt(pdata->curl, CURLOPT_TIMEOUT, 120);        // таймаут ответа
      curl_easy_setopt(pdata->curl, CURLOPT_MAXREDIRS, 10);       // останавливаться после 10-ого редиректа
 
 
      //указываем функцию обратного вызова для записи получаемых данных
      curl_easy_setopt(pdata->curl, CURLOPT_WRITEFUNCTION, writer);
      //указываем куда записывать принимаемые данные
      curl_easy_setopt(pdata->curl, CURLOPT_WRITEDATA, &(pdata->buffer));
      //запускаем выполнение задачи
      pdata->result = curl_easy_perform(pdata->curl);
 
      //проверяем успешность выполнения операции
      if (pdata->result == CURLE_OK)
        {
        string fn(pinfo->NameOfFile);
        ofstream f(fn.c_str());
        if(!f) { MessageBox(pinfo->hwnd, "Can't create file","Error",MB_OK);}
        else   { f<<pdata->buffer<<endl; f.close();}
        }
  }
    else
    {
    //выводим сообщение об ошибке
    MessageBox(pinfo->hwnd, "Can't connect","Error",MB_OK);
    return false;
    }
     
  //завершаем сессию
  curl_easy_cleanup(pdata->curl);
 
  TlsFree(dwTlsIndex);
 
  return true;
}
После регистрации реклама в сообщениях будет скрыта и будут доступны все возможности форума.
Aneron
 Аватар для Aneron
157 / 156 / 12
Регистрация: 20.04.2010
Сообщений: 570
14.07.2012, 18:39     Использование CreateThread #2
какая ошибка7 экстрасенсов нет.
alexey31415
 Аватар для alexey31415
59 / 59 / 3
Регистрация: 16.05.2010
Сообщений: 632
14.07.2012, 21:34  [ТС]     Использование CreateThread #3
Цитата Сообщение от alexey31415 Посмотреть сообщение
curl_easy_setopt(pdata->curl, CURLOPT_ERRORBUFFER, pdata->errorBuffer);
вот здесь функция прерывается

Добавлено через 2 часа 50 минут
такж есть проблема с разделением потоков,после 3 вызовов CreateThread вызываю WaitForMultipleObjects,но вызываемые после этой функции 3 потока для выделения данных запускаются не по завершении первых трёх(для посылки запроса)
Somebody
2770 / 1583 / 141
Регистрация: 03.12.2007
Сообщений: 4,139
Завершенные тесты: 1
14.07.2012, 22:48     Использование CreateThread #4
_beginthread инициализирует сишную библиотеку (которая может использовать TLS), а CreateThread - нет. Если curl использует сишные функции, получаются проблемы.
Van111
кодер с++
208 / 187 / 4
Регистрация: 03.08.2011
Сообщений: 2,585
Записей в блоге: 12
14.07.2012, 23:29     Использование CreateThread #5
alexey31415, вот рабочий пример

создание потоков
Создание потоков


--------------------------------------------------------------------------------

Функция CreateThread создает для процесса новый поток. Созданный поток должен определить начальный адрес кода, с которого новый поток должен исполняться. Как правило, начальный адрес - это название функции, определенной в коде программы. Эта функция получает единственный параметр и возвращает значение типа DWORD. Процесс может иметь одновременно несколько потоков, выполняющих ту же самую функцию.

Нижеследующий пример демонстрирует, как создать новый поток, который выполняет локально определяемую функцию ThreadFunc.

#include <windows.h>
#include <conio.h>

DWORD WINAPI ThreadFunc(LPVOID lpParam)

{

char szMsg[80];

wsprintf(szMsg, "Parameter = %d", *(DWORD*)lpParam);

MessageBox( NULL, szMsg, " ThreadFunc", MB_OK );

return 0;

}

VOID main( VOID )

{

DWORD dwThreadId, dwThrdParam = 1;

HANDLE hThread;

char szMsg[80];
hThread = CreateThread(

NULL, // атрибуты безопасности по умолчанию

0, // размер стека используется по умолчанию

ThreadFunc, // функция потока

&dwThrdParam, // аргумент функции потока

0, // флажки создания используются по умолчанию

&dwThreadId); // возвращает идентификатор потока

// При успешном завершении проверяет возвращаемое значение.

if (hThread == NULL)

{
wsprintf( szMsg, "CreateThread failed." );
MessageBox( NULL, szMsg, "main", MB_OK );
}
else
{
_getch();
CloseHandle( hThread );
}
}


Для простоты, этот пример передает указатель на значение как на параметр функции потока. Это может быть указатель на любой тип данных или структуру, или это может быть пропущено совсем, при помощи передачи указателя NULL и удаления ссылок на параметр в ThreadFunc.

Опасно передавать адрес локальной переменной, если создающий поток заканчивает работу перед созданием нового потока, потому что указатель становится недопустимым. Вместо этого, или передайте указатель в динамически распределяемую память, или заставьте создающий поток ждать до тех пор, пока новый поток не завершит свое формирование. Данные могут также быть переданы и из создающего потока в новый поток, используя глобальные переменные. С глобальными переменными, обычно необходимо синхронизировать доступ ко многим потокам. Для получения дополнительной информации о синхронизации, см. статью Синхронизация исполнения многопоточного режима.

В процессах, где поток мог бы создать несколько потоков, чтобы исполнить некоторый код, неудобно использовать глобальные переменные. Например, процесс, который разрешает пользователю открыть одновременно несколько файлов на некоторое время, может создавать новый поток для каждого файла, с целью, чтобы каждый из потоков выполнял некоторую функцию потока. Создающий поток может передавать уникальную информацию (такую как имя файла) требующуюся для каждого экземпляра функции потока как параметр. Вы не можете использовать единственную глобальную переменную для этой цели, но Вы можете использовать динамически распределяемый строковый буфер.

Создающий поток может использовать параметры функции CreateThread, чтобы определить нижеследующее:

Атрибуты системы безопасности для дескриптора нового потока. Эти атрибуты защиты включают в себя флажок наследования, который устанавливает, может ли дескриптор быть унаследован дочерними процессами. Атрибуты системы безопасности к тому же включают в себя дескриптор безопасности, который система использует, чтобы выполнить доступ, который контролирует все последующие использования дескриптора потока прежде, чем предоставляется обращение к нему.
Начальный размер стека нового потока. Стек потока назначается автоматически в пространстве памяти процесса; система увеличивает стек насколько необходимо и освобождает его, когда поток заканчивает работу. Дополнительную информацию см. в статье Размер стека потока.
Флажок создания, который разрешает Вам создать поток в состоянии ожидания. Когда произошла приостановка, поток не запускается до тех пор, пока не будет вызвана функция ResumeThread.
Вы можете также создать поток и путем вызова функции CreateRemoteThread. Эта функция используется процессами отладчика, чтобы создать поток, который запускается в адресном пространстве отлаживаемого процесса.

Назад в оглавление темы
На главную страницу темы
alexey31415
 Аватар для alexey31415
59 / 59 / 3
Регистрация: 16.05.2010
Сообщений: 632
15.07.2012, 00:32  [ТС]     Использование CreateThread #6
Цитата Сообщение от Van111 Посмотреть сообщение
вот рабочий пример
я видел подобные примеры и сделал по аналогии,я ж потому тему и создал,что не нашёл решения
Avazart
 Аватар для Avazart
6904 / 5144 / 253
Регистрация: 10.12.2010
Сообщений: 22,629
Записей в блоге: 17
15.07.2012, 09:30     Использование CreateThread #7
Ну думаю проблема не в потоках

Если на быструю руку, то приблизительно так. DLoader.rar

Не знаю как парсинг, но чисто загрузка у меня работает нормально.
alexey31415
 Аватар для alexey31415
59 / 59 / 3
Регистрация: 16.05.2010
Сообщений: 632
15.07.2012, 12:30  [ТС]     Использование CreateThread #8
возможно у меня проблема возникает из-за добавления WaitForMultipleObjects,потому что мне нужно остановитть основной поток,пока работают потоки для загрузки,потом возобновить основной,запустить потоки для парсинга,опять остановить основной-дождаться и лишь затем заполнить списки
а у меня выходит так,запускаю программу,вылетает сообщение разрешение на установление соединения(это антивирус работает,ничего страшного),разрешаю-и создаётся лишь один файл,потом начинается парсинг и естетсвенно один открывается,а остальные два нет
Avazart
 Аватар для Avazart
6904 / 5144 / 253
Регистрация: 10.12.2010
Сообщений: 22,629
Записей в блоге: 17
15.07.2012, 17:47     Использование CreateThread #9
Ну так код в студию с указанным WaitForMultipleObjects
alexey31415
 Аватар для alexey31415
59 / 59 / 3
Регистрация: 16.05.2010
Сообщений: 632
15.07.2012, 23:19  [ТС]     Использование CreateThread #10
вот часть WndProc,где создаются потоки
WndProc
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
int i;
 
        //------------------------------------------------
        //получаем данные
        //------------------------------------------------
 
        for(i = 0;i < THREADCOUNT;i++)
        {
            aThread[i] = CreateThread(
                NULL,0,
                (LPTHREAD_START_ROUTINE)connect,
                NULL,
                0,
                &ThreadId);
        }
 
        WaitForMultipleObjects(THREADCOUNT, aThread, true,INFINITE);
 
        for(i = 0;i < THREADCOUNT;i++)
            CloseHandle(aThread[i]);
 
        //------------------------------------------------
        //извлекаем из xml файла
        //------------------------------------------------
    
        for(i = 0;i < THREADCOUNT;i++)
        {
            aThread[i] = CreateThread(
                NULL,0,
                (LPTHREAD_START_ROUTINE)parsing,
                NULL,
                0,
                &ThreadId);
        }
    
        WaitForMultipleObjects(THREADCOUNT, aThread, true,INFINITE);
 
        for(i = 0;i < THREADCOUNT;i++)
            CloseHandle(aThread[i]);

вот функция
connect
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
#include "StdAfx.h"
 
//-----------------------------------------------------------------------
//функция для установления соединения,получения данных и записи их в файл
//-----------------------------------------------------------------------
 
typedef struct
{
    string buffer;
    char errorBuffer[CURL_ERROR_SIZE];
    CURL *curl;
    CURLcode result;
}DATA, *PDATA;
 
//функция обратного вызова
static int writer(char *data, size_t size, size_t nmemb, string *buffer)
{
  //переменная - результат, по умолчанию нулевая
  int result = 0;
 
  if (buffer != NULL)
  {
    //добавляем к буферу строки из data, в количестве nmemb
    buffer->append(data, size * nmemb);
    //вычисляем объем принятых данных
    result = size * nmemb;
  }
  //вовзращаем результат
  return result;
}
 
 
DWORD WINAPI connect(LPVOID pvoid)
{
    PINFO pinfo = (PINFO)pvoid;
 
    DWORD dwTlsIndex = TlsAlloc();
 
    TlsSetValue(dwTlsIndex,GlobalAlloc(GPTR,sizeof(DATA)));
 
    PDATA pdata;
 
    pdata = (PDATA)TlsGetValue(dwTlsIndex);
 
    //инициализируем curl
    pdata->curl = curl_easy_init();
 
    //проверяем результат инициализации
  if (pdata->curl)
    {
    curl_easy_setopt(pdata->curl, CURLOPT_ERRORBUFFER, pdata->errorBuffer);
      //задаем все необходимые опции
      curl_easy_setopt(pdata->curl, CURLOPT_URL,pinfo->url);
     
      curl_easy_setopt(pdata->curl, CURLOPT_HEADER, 0);         // не возвращает заголовки
      curl_easy_setopt(pdata->curl, CURLOPT_FOLLOWLOCATION, 1);   // переходит по редиректам
      curl_easy_setopt(pdata->curl, CURLOPT_ENCODING, "");        // обрабатывает все кодировки
 
      curl_easy_setopt(pdata->curl, CURLOPT_CONNECTTIMEOUT, 120); // таймаут соединения
      curl_easy_setopt(pdata->curl, CURLOPT_TIMEOUT, 120);        // таймаут ответа
      curl_easy_setopt(pdata->curl, CURLOPT_MAXREDIRS, 10);       // останавливаться после 10-ого редиректа
 
 
      //указываем функцию обратного вызова для записи получаемых данных
      curl_easy_setopt(pdata->curl, CURLOPT_WRITEFUNCTION, writer);
      //указываем куда записывать принимаемые данные
      curl_easy_setopt(pdata->curl, CURLOPT_WRITEDATA, &(pdata->buffer));
      //запускаем выполнение задачи
      pdata->result = curl_easy_perform(pdata->curl);
 
      //проверяем успешность выполнения операции
      if (pdata->result == CURLE_OK)
        {
        string fn(pinfo->NameOfFile);
        ofstream f(fn.c_str());
        if(!f) { MessageBox(pinfo->hwnd, "Can't create file","Error",MB_OK);}
        else   { f<<pdata->buffer<<endl; f.close();}
        }
  }
    else
    {
    //выводим сообщение об ошибке
    MessageBox(pinfo->hwnd, "Can't connect","Error",MB_OK);
    return false;
    }
     
  //завершаем сессию
  curl_easy_cleanup(pdata->curl);
 
  TlsFree(dwTlsIndex);
 
  return true;
}

а вот
parsing
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
#include "StdAfx.h"
 
extern NEWS source[15];
 
typedef struct
{
    TiXmlDocument *doc;
}DATA, *PDATA;
 
DWORD WINAPI parsing(PVOID pvoid)
{
    PINFO pinfo;
 
    pinfo = (PINFO)pvoid;
 
    DWORD dwTlsIndex = TlsAlloc();
 
    TlsSetValue(dwTlsIndex,GlobalAlloc(GPTR,sizeof(DATA)));
 
    PDATA pdata;
 
    pdata = (PDATA)TlsGetValue(dwTlsIndex);
 
    int Start  =  pinfo->Amount;
    int Finish = pinfo->Amount + 5;
 
    string temp(pinfo->NameOfFile);
 
    pdata->doc = new TiXmlDocument(temp.c_str());
 
    bool LoadOK = pdata->doc->LoadFile();
 
    if(!LoadOK)
    {
        string error("Can't open file ");
        error += temp;
        MessageBox(pinfo->hwnd, error.c_str(), "Error",MB_OK);
        return false;
    }
 
    TiXmlElement *rss = pdata->doc->FirstChildElement("rss");
 
    TiXmlElement *channel = rss->FirstChildElement("channel");
 
    TiXmlElement *item = channel->FirstChildElement("item");
 
    MessageBox(pinfo->hwnd,"Start parsing","",MB_OK);
 
    while(Start < Finish)
    {
        TiXmlElement *title = item->FirstChildElement("title");
 
        wchar_t Temp[1024];
        char AnsiTemp[1024];
 
        MultiByteToWideChar(CP_UTF8,0,title->GetText(),
            -1,Temp,sizeof(Temp));
        WideCharToMultiByte(CP_ACP,0,Temp,-1,AnsiTemp,
            sizeof(AnsiTemp),NULL,NULL);
 
        source[Start].title = new char[strlen(AnsiTemp)+1]; 
        source[Start].title = AnsiTemp;
 
        TiXmlElement *link = item->FirstChildElement("link");
 
        source[Start].link = new char[strlen(link->GetText())+1];
        source[Start].link = const_cast<char *>(link->GetText());
 
        MessageBox(pinfo->hwnd,source[Start].link,"",MB_OK);
 
        Start++;
 
        item = item->NextSiblingElement();
    }
 
    TlsFree(dwTlsIndex);
 
    return true;
}
alexey31415
 Аватар для alexey31415
59 / 59 / 3
Регистрация: 16.05.2010
Сообщений: 632
17.07.2012, 01:43  [ТС]     Использование CreateThread #11
и что?никто не может помочь?
Avazart
 Аватар для Avazart
6904 / 5144 / 253
Регистрация: 10.12.2010
Сообщений: 22,629
Записей в блоге: 17
19.07.2012, 23:25     Использование CreateThread #12
А зачем ждать пока все скачается? А потом только парсить?
Почему сразу и потока закачки не вызывать поток парсинга?
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
20.07.2012, 11:35     Использование CreateThread
Еще ссылки по теме:

C/C++ Использование функций, использование break C++
Использование строк.Использование структур C++
C++ Struct в CreateThread

Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
alexey31415
 Аватар для alexey31415
59 / 59 / 3
Регистрация: 16.05.2010
Сообщений: 632
20.07.2012, 11:35  [ТС]     Использование CreateThread #13
всё равно создаётся лишь один файл и завершается лишь один поток
Yandex
Объявления
20.07.2012, 11:35     Использование CreateThread
Ответ Создать тему
Опции темы

Текущее время: 06:23. Часовой пояс GMT +3.
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2016, vBulletin Solutions, Inc.
Рейтинг@Mail.ru