Форум программистов, компьютерный форум, киберфорум
C++ Builder
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 5.00/8: Рейтинг темы: голосов - 8, средняя оценка - 5.00
0 / 0 / 1
Регистрация: 12.08.2013
Сообщений: 23

Не работает Synchronize() в DLL при сборке без рантайм библиотек

31.08.2017, 13:41. Показов 1666. Ответов 11
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Добрый день. Есть библиотека, в ней есть поток TThread. Библиотека должна использоваться на компьютере без IDE. Когда библиотека собирается с выставленными галочками Link with runtime packages и Link with dynamic RTL, всё работает хорошо. Убираю галочки - программа виснет на методе Synchronize, не вызывая полученную функцию. Код:

NetThread.h
Кликните здесь для просмотра всего текста
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
//---------------------------------------------------------------------------
 
#ifndef NetworkH
#define NetworkH
#include <System.Classes.hpp>
#include <IdBaseComponent.hpp>
#include <IdComponent.hpp>
#include <IdUDPClient.hpp>
#include "OMSDefs.h"
typedef void TCallback(OMSDeviceT Dev, unsigned char * Data, unsigned int Size);
class NetThread : public TThread
{
    public:
        __fastcall NetThread(OMSDeviceT Dev, unsigned int Port);
        TIdUDPClient * udp;
        void WriteReg(unsigned int Address, unsigned char * Value, unsigned int Size);
        void ReadReg(unsigned int Address, unsigned int Size, TCallback * Callback);
        void StartFlow(TCallback * Callback);
        void StopFlow();
    protected:
        void __fastcall Execute();
    private:
        //synchronize methods
        void __fastcall FlowResponse();
        void __fastcall SingleResponse();
 
        OMSDeviceT _dev;
        unsigned int _port;
 
 
        void (*_callbackFlow)(OMSDeviceT, unsigned char *, unsigned int);
        void (*_callbackSingle)(OMSDeviceT, unsigned char *, unsigned int);
 
};
//---------------------------------------------------------------------------
#endif


Network.cpp
Кликните здесь для просмотра всего текста
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
//---------------------------------------------------------------------------
 
#pragma hdrstop
 
#include "Network.h"
#include "OMSSensorSimple.h"
//---------------------------------------------------------------------------
#pragma package(smart_init)
__fastcall NetThread::NetThread(OMSDeviceT Dev, unsigned int Port) :
    TThread(true), _dev(Dev), _port(Port)
{
 
}
void NetThread::StartFlow(TCallback * Callback)
{
    _callbackFlow = Callback;
    Priority = tpNormal;
    Start();
}
void __fastcall NetThread::Execute()
{
 
    while(!Terminated)
    {
        Sleep(500);
        Synchronize(FlowResponse);
    }
}
void NetThread::WriteReg(unsigned int Address, unsigned char * Value, unsigned int Size)
{
 
}
void NetThread::ReadReg(unsigned int Address, unsigned int Size, TCallback * Callback)
{
    _callbackSingle = Callback;
}
void __fastcall NetThread::FlowResponse()
{
    byte a[2];
    a[0] = 1;
    a[1] = 2;
    _callbackFlow(_dev,a,2);
}
void NetThread::StopFlow()
{
 
}



Создание потока:
C++
1
2
3
4
5
6
7
8
std::vector<HANDLE> vDevs;
HANDLE __stdcall OMSConnect(OMSDeviceT & Dev,unsigned int Port)
{
    NetThread * nt = new NetThread(Dev,Port);
    HANDLE h = (void*)nt;
    vDevs.push_back(h);
    return vDevs.back();
}
Запуск потока:

C++
1
2
3
4
5
void __stdcall OMSStartFlow(HANDLE Dev,  CallbackFlowT * CallbackFlow)
{
    NetThread * dev = (NetThread*)Dev;
    dev->StartFlow(CallbackFlow);
}
Сам callback:
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
//---------------------------------------------------------------------------
void CallbackFlow(OMSDeviceT Dev, unsigned char * Data, unsigned int Size)
{
    switch(Dev.devIP)
    {
        case 0xC0A80002:                            //ip = 192.168.0.2
            Form6->Memo2->Lines->Add("flow 1");
            break;
        case 0xC0A80003:
            Form6->Memo3->Lines->Add("flow 2");     //ip = 192.168.0.3
            break;
        default:
            break;
    }
 
}
Что я делаю не так?
0
Лучшие ответы (1)
IT_Exp
Эксперт
34794 / 4073 / 2104
Регистрация: 17.06.2006
Сообщений: 32,602
Блог
31.08.2017, 13:41
Ответы с готовыми решениями:

включение библиотек при сборке
Прошу помочь. Написал прогу в Builder6, собрал, перенес на другую машину - не работает . Просит файлы. В настройках редактора неактивно...

При статической сборке Qt требует 3 dll
Я собрал Qt статически и он всё равно требует 3 dll libgcc_s_dw2-1.dll libstdc++-6.dll libwinpthread-1.dll

Возникают ошибки при сборке dll qwt
значит хочу себе поставить qwt 6.0.1, на копме стоит WIN XP SP3, MS VS 2010, Qt 4.8.1. Все делал так, ка написано здесь...

11
Эксперт С++
 Аватар для Avazart
8484 / 6151 / 615
Регистрация: 10.12.2010
Сообщений: 28,683
Записей в блоге: 30
31.08.2017, 13:51
Цитата Сообщение от romasv Посмотреть сообщение
Что я делаю не так?
Всё.

Сам идея дохлая.
Начиная с использование экспорта классов из DLL.
И заканчивая Synchronize в dll в котором нет основного потока gui с которым нужно синхронизироваться.
Я уже не говорю - нафига оно вообще сдалось пихать в dll
0
0 / 0 / 1
Регистрация: 12.08.2013
Сообщений: 23
31.08.2017, 14:03  [ТС]
Класс не экспортируется из DLL. Экспортируются OMSConnect и StartFlow. Коннект возвращает хэндл на поток, СтартФлоу принимает хэндл и запускает по этому адресу поток. Может это неправильно...

GUI есть, оно вызывает экспортные функции, и передает в DLL коллбэки на свои функции. Получается, GUI и DLL работают в одном потоке, но в DLL есть второй поток, который и должен синхронизоваться с потоком GUI.

Как лучше сделать?
0
Эксперт С++
 Аватар для Avazart
8484 / 6151 / 615
Регистрация: 10.12.2010
Сообщений: 28,683
Записей в блоге: 30
31.08.2017, 14:08
Цитата Сообщение от romasv Посмотреть сообщение
Класс не экспортируется из DLL.
Что тогда такое OMSDeviceT?
Цитата Сообщение от romasv Посмотреть сообщение
GUI есть, оно вызывает экспортные функции, и передает в DLL коллбэки на свои функции.
Я не вижу кода GUI в dll...
Цитата Сообщение от romasv Посмотреть сообщение
Получается, GUI и DLL работают в одном потоке,
Не канает, поток в dll не знает ничего о GUI где-то во вне.
0
0 / 0 / 1
Регистрация: 12.08.2013
Сообщений: 23
31.08.2017, 14:10  [ТС]
Код GUI:
Кликните здесь для просмотра всего текста
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
//---------------------------------------------------------------------------
 
#include <vcl.h>
#pragma hdrstop
 
#include "tester.h"
//---------------------------------------------------------------------------
#pragma package(smart_init)
#pragma resource "*.dfm"
TForm6 *Form6;
//---------------------------------------------------------------------------
__fastcall TForm6::TForm6(TComponent* Owner)
    : TForm(Owner)
{
}
//---------------------------------------------------------------------------
void __fastcall TForm6::FormCreate(TObject *Sender)
{
    OMSDeviceT * omsDevice;
    OMSDeviceT dev;
    HANDLE hDev1,hDev2;
    unsigned int n;
    unsigned int type;
    unsigned int ip;
    OMSHello(&omsDevice,&n);
    for(int i = 0; i < n; i++)
    {
        type = omsDevice->devType;
        ip = omsDevice->devIP;
        dev.devType = type;
        dev.devIP = ip;
        vDevs.push_back(dev);
        String s = "Device"+IntToStr(i)+": Type = " + UIntToStr(type) + " IP = " + UIntToStr(ip);
        Memo1->Lines->Add(s);
        omsDevice++;
 
    }
 
    Memo1->Lines->Add("Connect device 1");
    hDev1 = OMSConnect(vDevs[0],5005);
    Memo1->Lines->Add("Start flow for device 1");
    OMSStartFlow(hDev1,*CallbackFlow);
 
//  Memo1->Lines->Add("Connect device 2");
//  hDev2 = OMSConnect(vDevs[1],5023);
//  Memo1->Lines->Add("Start flow for device 2");
//  OMSStartFlow(hDev2,*CallbackFlow);
}
//---------------------------------------------------------------------------
void CallbackFlow(OMSDeviceT Dev, unsigned char * Data, unsigned int Size)
{
    switch(Dev.devIP)
    {
        case 0xC0A80002:                            //ip = 192.168.0.2
            Form6->Memo2->Lines->Add("flow 1");
            break;
        case 0xC0A80003:
            Form6->Memo3->Lines->Add("flow 2");     //ip = 192.168.0.3
            break;
        default:
            break;
    }
 
}
//---------------------------------------------------------------------------
void CallbackSingle(OMSDeviceT Dev, unsigned char * Data, unsigned int Size)
{
 
}
0
Эксперт С++
 Аватар для Avazart
8484 / 6151 / 615
Регистрация: 10.12.2010
Сообщений: 28,683
Записей в блоге: 30
31.08.2017, 14:20
Лучший ответ Сообщение было отмечено romasv как решение

Решение

Цитата Сообщение от romasv Посмотреть сообщение
Как лучше сделать?
1. Не пихать все в dll (А нафига?)
2. Не использовать Synchronize.
3. Не использовать стат линковку.

итд...

Цитата Сообщение от romasv Посмотреть сообщение
Код GUI:
Это код приложения? Или dll ?
Если приложения - читайте что я написал выше, поток в dll ничего не знает о вашем приложении.

В динамике оно работает лишь по тому что грузятся одни и те же библиотеки и используются совместно
(глобальные объекты, менеджер памяти идр вещи скрытые от глаз.)

Добавлено через 6 минут
В обще это мутная тема где все не очевидно и непонятно.

Можете почитать:
Создать DLL, которая создаст кнопку на форме
1
0 / 0 / 1
Регистрация: 12.08.2013
Сообщений: 23
31.08.2017, 15:42  [ТС]
Хм, довольно интересно всё расписали, спасибо!

Цитата Сообщение от Avazart Посмотреть сообщение
Это код приложения? Или dll ?
Если приложения - читайте что я написал выше, поток в dll ничего не знает о вашем приложении.
Код приложения. Он же и передает указатель на свой метод в библиотеку.

В DLL заказчик просил запихать... Dll должна принимать запросы с gui-программки, и принимать udp-поток данных. То, что она приняла, она должна пихнуть обратно в gui. Заказчику видимо лень писать свой поток.

Добавлено через 1 час 0 минут
И спасибо за советы! В общем, убрал Synchronize, теперь вызываю метод напрямую. Всё работает, и никак не глючит... Не понимаю, почему, ведь какой-то совершенно левый поток что-то меняет в другой программе, да ещё и на форме, и ему это прекрасно удается! Магия...
0
Эксперт С++
 Аватар для Avazart
8484 / 6151 / 615
Регистрация: 10.12.2010
Сообщений: 28,683
Записей в блоге: 30
31.08.2017, 16:04
Цитата Сообщение от romasv Посмотреть сообщение
теперь вызываю метод напрямую.
Это как ? Без синхронизации? Если да - то это нифига не работает.
0
0 / 0 / 1
Регистрация: 12.08.2013
Сообщений: 23
31.08.2017, 16:08  [ТС]
Да, без синхронизации...

C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
void __fastcall NetThread::Execute()
{
    while(!Terminated)
    {
        Sleep(500);
//      Synchronize(FlowResponse);      //don't use synchronize in static linked dll (won't be working)
        FlowResponse();
    }
}
void __fastcall NetThread::FlowResponse()
{
    byte a[2];
    a[0] = 1;
    a[1] = 2;
    _callbackFlow(_dev,a,2);
}
И колбэк-функции спокойно пишут в Memo на форму, при этом я могу сам что-то в тот же Memo писать. Пытался заглючить прогу тем, что кинул на форму таймер и заставил его каждую миллисекунду писать текст в тот же Memo. Никаких сбоев. Очень странно...
0
Эксперт С++
 Аватар для Avazart
8484 / 6151 / 615
Регистрация: 10.12.2010
Сообщений: 28,683
Записей в блоге: 30
31.08.2017, 17:22
Цитата Сообщение от romasv Посмотреть сообщение
Да, без синхронизации...
Тогда в топку.

Добавлено через 32 секунды
Цитата Сообщение от romasv Посмотреть сообщение
Пытался заглючить прогу тем, что кинул на форму таймер и заставил его каждую миллисекунду писать текст в тот же Memo. Никаких сбоев.
Хреново пытались.
0
0 / 0 / 1
Регистрация: 12.08.2013
Сообщений: 23
01.09.2017, 08:30  [ТС]
Тогда какой выход? Использовать WINAPI-поток? Насколько безопасно будет в нем использовать TIdUDPClient? И как быть с синхронизацией?
0
Эксперт С++
 Аватар для Avazart
8484 / 6151 / 615
Регистрация: 10.12.2010
Сообщений: 28,683
Записей в блоге: 30
01.09.2017, 10:15
Цитата Сообщение от romasv Посмотреть сообщение
И как быть с синхронизацией?
Можете попробовать использовать системную очередь сообщений, для синхронизации.
Но как я сказал лучше отказаться от такой затеи вовсе или же ликоваться динамически.
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
BasicMan
Эксперт
29316 / 5623 / 2384
Регистрация: 17.02.2009
Сообщений: 30,364
Блог
01.09.2017, 10:15
Помогаю со студенческими работами здесь

Как изменить место создания dll-библиотек в своём проекте при компиляции
Здравствуйте, возник вопрос. В моём решении используются dll-библиотеки. Все они как обычно подключены через референсы. При компиляции...

Почему при сборке Releas'а прога без VC++ не идёт?
Прога использует библиотеки ДАО. Собираю как Release, на машине Win 98 она выдаёт ошибку MS VC++ Runtime Error.

Не работает Synchronize в TThread
Доброго времени суток! .h class TMyThread : public TThread { private: void __fastcall send_mess(); protected: void...

Как лучше копировать память, обмениваться указателями и т.д. при работе с кучей динамических библиотек (DLL) ?
Привет! Вот есть много DLL-к, каждая работает со своей внутренней динамической памятью, а так же есть необходимость работать с объектами...

При сборке работает только полпроекта
при сборке работает только полпроекта. Что делать? в свойствах все так же как и для отладки, а именно подсистема - винда и точка входа -...


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

Или воспользуйтесь поиском по форуму:
12
Ответ Создать тему
Новые блоги и статьи
моя боль
iceja 24.01.2026
Выложила интерполяцию кубическими сплайнами www. iceja. net REST сервисы временно не работают, только через Web. Написала за 56 рабочих часов этот сайт с нуля. При помощи perplexity. ai PRO , при. . .
Модель сукцессии микоризы
anaschu 24.01.2026
Решили писать научную статью с неким РОманом
http://iceja.net/ математические сервисы
iceja 20.01.2026
Обновила свой сайт http:/ / iceja. net/ , приделала Fast Fourier Transform экстраполяцию сигналов. Однако предсказывает далеко не каждый сигнал (см ограничения http:/ / iceja. net/ fourier/ docs ). Также. . .
http://iceja.net/ сервер решения полиномов
iceja 18.01.2026
Выкатила http:/ / iceja. net/ сервер решения полиномов (находит действительные корни полиномов методом Штурма). На сайте документация по API, но скажу прямо VPS слабенький и 200 000 полиномов. . .
Расчёт переходных процессов в цепи постоянного тока
igorrr37 16.01.2026
/ * Дана цепь(не выше 3-го порядка) постоянного тока с элементами R, L, C, k(ключ), U, E, J. Программа находит переходные токи и напряжения на элементах схемы классическим методом(1 и 2 з-ны. . .
Восстановить юзерскрипты Greasemonkey из бэкапа браузера
damix 15.01.2026
Если восстановить из бэкапа профиль Firefox после переустановки винды, то список юзерскриптов в Greasemonkey будет пустым. Но восстановить их можно так. Для этого понадобится консольная утилита. . .
Сукцессия микоризы: основная теория в виде двух уравнений.
anaschu 11.01.2026
https:/ / rutube. ru/ video/ 7a537f578d808e67a3c6fd818a44a5c4/
WordPad для Windows 11
Jel 10.01.2026
WordPad для Windows 11 — это приложение, которое восстанавливает классический текстовый редактор WordPad в операционной системе Windows 11. После того как Microsoft исключила WordPad из. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru