Форум программистов, компьютерный форум, киберфорум
C++: Сети
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 5.00/11: Рейтинг темы: голосов - 11, средняя оценка - 5.00
4 / 4 / 6
Регистрация: 17.09.2012
Сообщений: 325

Сервер ожидает recv, хотя клиент вроде посылает

27.10.2014, 22:14. Показов 2376. Ответов 9
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
добрый день, можете подсказать пожалуйста что может быть не так, вообщем у меня есть клиент и сервер и вроде все фурычит как мне нужно, только вот есть проблемка, такое ощушение что у меня в цикле на сервере когда он принял первый recv и по второму кругу в цикле доходит до этого же рековера, то ждет пока будет принято, хотя клиент вроде как посылает, на всякий случай выложу целый цикл, но мне кажется проблема в recv ведь когда цикл проходит 1 раз то все делается как надо, а потом как будто ждет
цикл на клиенте
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
 bool onethread=false;
  while(1)
  {
    std::cout<<"Hello"<<"\n"<<"What do you want? \n1.Share file's \n2.Download file's \n3.Exit program \nPlease insert you choise: ";
    std::cin>>Client.choise;
    if(Client.choise=='1')
    {
        std::cout<<"What file you want to share?\nEnter the name: ";       
        std::cin>>Client.NameFile;
        send(CreateClient,(char *)&Client,sizeof(Packet),0);
        if(!onethread)
        {
            onethread=true;
            DWORD thID;
            HANDLE threadforwait = CreateThread(NULL, NULL, SharedFile, &CreateClient, NULL, &thID);
            CloseHandle(threadforwait);
        }
    }
    else if(Client.choise=='2')
    {
        std::cout<<"What file you want download?\nEnter the name: ";
        std::cin>>Client.NameFile; 
        send(CreateClient,(char *)&Client,sizeof(Packet),0);
        std::vector<char> file(sizeof(Packet)); 
        recv(CreateClient,&file.front(),sizeof(Packet),0);
        Client = *(reinterpret_cast<Packet*>(&file.front()));
        if(Client.choise=='5')
        {
            sockaddr_in ForDownload;  
            ForConnect.sin_family=AF_INET;
            ForConnect.sin_port=htons(3035);
            ForConnect.sin_addr.S_un.S_addr=inet_addr(Client.ReturnAddr);
            connect(CreateClient,(sockaddr*)&ForDownload,sizeof(ForDownload));
            //download...
        }
        else
        {
            std::cout<<"This file not found\r\n";
        }
    }
    else if(Client.choise=='3'){break;}
    else {std::cout<<"Incorrect input\n";}
  }
цикл на сервере
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
    std::vector<char> file(sizeof(Packet));       
    while(1)
    {
        if(recv(NS,&file.front(),sizeof(Packet),0)<=0)
        {
            Database.erase(GetData.ReturnAddr);
            break;  
        }        
        GetData = *(reinterpret_cast<Packet*>(&file.front()));        
        if (GetData.choise=='1')
        { 
            Database.insert(std::make_pair(GetData.ReturnAddr, &GetData.NameFile[0]));
            for (auto it = Database.begin(); it != Database.end(); ++it)
            {
                std::cout << it->first << " : " << it->second << std::endl;
            }
        }
        else if(GetData.choise=='2')
        {  
            for (auto it = Database.begin(); it != Database.end(); it++)           
            {
                if(it->second==GetData.NameFile)
                {
                    strcpy(GetData.ReturnAddr,Iter->first.c_str());
                    GetData.choise='5';
                    send(NS,(char *)&GetData,sizeof(Packet),0);
                }
                else
                {
                    GetData.choise='0';
                    send(NS,(char *)&GetData,sizeof(Packet),0);
                }
            }
        }
    }
0
IT_Exp
Эксперт
34794 / 4073 / 2104
Регистрация: 17.06.2006
Сообщений: 32,602
Блог
27.10.2014, 22:14
Ответы с готовыми решениями:

Клиент/сервер: клиент посылает серверу слово; определить, является ли это слово палиндромом
Осуществить взаимодействие клиента и сервера на основе протокола TCP/IP. Функционирование клиента и сервера реализовать следующим образом:...

Клиент посылает строку, а сервер заменяет каждый 4 символ на %
Серверная часть #include &quot;stdafx.h&quot; #include &lt;winsock2.h&gt; #include &lt;iostream&gt; #include &lt;stdio.h&gt; #include &lt;stdlib.h&gt; #pragma...

Сервер выдаёт неправильный ответ, хотя, вроде, всё работает
Дана задача: №2 Друзья Саша и Паша живут на одной улице. Как то раз они договорились порешать задачи Открытой Московской...

9
Ушел с форума
Эксперт С++
 Аватар для Убежденный
16481 / 7444 / 1187
Регистрация: 02.05.2013
Сообщений: 11,616
Записей в блоге: 1
27.10.2014, 22:35
Цитата Сообщение от итернал Посмотреть сообщение
C++
1
if(recv(NS,&file.front(),sizeof(Packet),0)<=0)
Все. После этой строчки можно дальше не смотреть.

Эта ошибка появляется настолько часто, что ей давно уже пора
сделать какую-нибудь закрепленную тему, где большими
красными буквами (Arial, 250px) было бы написано:

Кликните здесь для просмотра всего текста


В TCP НЕТ ПАКЕТОВ !
В TCP ПАКЕТОВ НЕТ !
ПАКЕТОВ НЕТ В TCP !
ПАКЕТОВ В TCP НЕТ !
НЕТ В TCP ПАКЕТОВ !
НЕТ ПАКЕТОВ В TCP !



Читаем описание функции recv на том же MSDN:

recv function
http://msdn.microsoft.com/en-u... 85%29.aspx
For connection-oriented sockets (type SOCK_STREAM for example), calling recv will
return as much data as is currently available—up to the size of the buffer specified.
"as much data as is currently available" !!!

То есть, "recv(Socket, Buffer, 1000, Flags)" может вернуть байт меньше 1000.
Например, 998, или 356, или вообще только 1.

Когда исправите это вопиющее недоразумение, можно будет продолжить анализ кода.
0
4 / 4 / 6
Регистрация: 17.09.2012
Сообщений: 325
27.10.2014, 23:04  [ТС]
сделать просто
C++
1
recv(NS,&file.front(),sizeof(Packet),0);
?
но тогда при запуске у меня начинает выводить множество раз то что я отправил первый раз
или вы что-то другое имели введу? тогда щас буду розбиратся
0
Эксперт С++
 Аватар для schdub
3073 / 1411 / 425
Регистрация: 19.01.2009
Сообщений: 3,894
27.10.2014, 23:48
Цитата Сообщение от итернал Посмотреть сообщение
но тогда при запуске у меня начинает выводить множество раз то что я отправил первый раз
или вы что-то другое имели введу? тогда щас буду розбиратся
Нужно всегда проверять сколько байт было отослано send() и сколько принято recv() - нужно проверять возвращаемое значение фукций. Когда по TCP нужно получить / отослать фиксированное количество байт, обычно используют такого рода функции:

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
//...
 
// ///////////////////////////////////////////////////////////////////////// //
 
int tcpsocket::send_all(SOCKET socket, const char* pbuffer, int buf_size) {
    int send_size = 0;
    int num = 0;
    while (buf_size) {
        num = send(socket, pbuffer + send_size, buf_size, 0);
        if (num <= 0) return num;
        send_size += num;
        buf_size  -= num;
    }
    return send_size;
}
 
// ///////////////////////////////////////////////////////////////////////// //
 
int tcpsocket::recv_all(SOCKET socket, char* pbuffer, int buf_size) {
    int recv_size = 0;
    int num = 0;
 
    while (buf_size) {
        num = recv(socket, pbuffer + recv_size, buf_size, 0);
        if (num <= 0) break;
        recv_size += num;
        buf_size  -= num;
    }
 
    return recv_size;
}
 
// ///////////////////////////////////////////////////////////////////////// //
 
//...
но обратите внимание, даже они не гарантируют, что будут отосланы / приняты все требуемые байты.
0
4 / 4 / 6
Регистрация: 17.09.2012
Сообщений: 325
28.10.2014, 17:40  [ТС]
Цитата Сообщение от итернал Посмотреть сообщение
но тогда при запуске у меня начинает выводить множество раз то что я отправил первый раз
так я заметил что у меня
C++
1
recv(NS,&file.front(),sizeof(Packet),0);
почему-то выполняется на втором проходе в цикле сразу, пока на клиенте нету send
0
4 / 4 / 0
Регистрация: 13.10.2013
Сообщений: 183
08.05.2015, 18:29
Сейчас пытаюсь разобраться в аналогичной ситуации.
Прошу отнестись к моему непониманию с пониманием, так как опыта нету никакого.

Суть того что я не понимаю:

C++
1
num = recv(socket, pbuffer + recv_size, buf_size, 0);
находится в бесконечном цикле.
Если в данный момент в сокет клиент ничего нового не кинул, а всё что было до этого кинуто давно прочтено, то по идее num должен быть приравнен нулю. Вместо этого он висит и ждёт когда в сокет что нибудь кинут.
Что может прервать цикл?
0
Ушел с форума
Эксперт С++
 Аватар для Убежденный
16481 / 7444 / 1187
Регистрация: 02.05.2013
Сообщений: 11,616
Записей в блоге: 1
08.05.2015, 19:26
Цитата Сообщение от Kapitan79 Посмотреть сообщение
Если в данный момент в сокет клиент ничего нового не кинул, а всё что было до этого кинуто давно прочтено, то по идее num должен быть приравнен нулю.
Нет. Для блокирующих сокетов recv не возвращает управление, пока в
буфер не придут данные (ret value = к-во прочитанных байт) или пока на
другом конце не разомкнут соединение (ret value = 0) или пока не
возникнет какая-либо ошибка (ret value = -1).
0
4 / 4 / 0
Регистрация: 13.10.2013
Сообщений: 183
08.05.2015, 19:51
Тогда получается что вариант один - разрывать соединение со стороны клиента после передачи данных чтобы выйти из цикла и затем делать отдельный сервер который будет принимать со стороны клиента ответ?

Или есть какой то флаг или признак что все байты например одной фотографии успешно переданы и можно запускать их дальнейшую обработку после чего послать ответ клиенту?
0
Ушел с форума
Эксперт С++
 Аватар для Убежденный
16481 / 7444 / 1187
Регистрация: 02.05.2013
Сообщений: 11,616
Записей в блоге: 1
08.05.2015, 20:00
Ну во-первых, никто не запрещает использовать асинхронные/неблокирующие сокеты.
Во-вторых, можно же передавать длину данных перед самими данными...
1
4 / 4 / 0
Регистрация: 13.10.2013
Сообщений: 183
08.05.2015, 20:15
Ясно, спасибо, почитаю про асинхронные сокеты.
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
BasicMan
Эксперт
29316 / 5623 / 2384
Регистрация: 17.02.2009
Сообщений: 30,364
Блог
08.05.2015, 20:15
Помогаю со студенческими работами здесь

Сервер выдаёт неправильный ответ, хотя, вроде, всё работает
Дана задача: №1 Геннадию интересно, сможет ли слон с поля x1, y1 побить пешку x2, y2 за неограниченное количество ходов. ...

Взаимодействие WinCC с Labview 2013, как сервер-клиент, так и клиент-сервер
Здравствуйте. Интересует информация о взаимодействии WinCC с Labview 2013, как сервер-клиент, так и клиент-сервер через ОРС-инфтерфейс. ...

Сервер посылает данные клиенту
Здравствуйте, подскажите пожалуйста. Есть callback api вконтакте который посылает данные на сервер, нужно чтобы сервер каким-то...

Сервер и recv
Вообщем такая вот проблема,есть клиент и сервер,все работает нормально,так сказать я не сильно еще разбираюсь в сетях на c++,поэтому...

Клиент посылает произвольный набор латинских букв серверу и получает их назад упорядоченными по алфавиту
Помогите пожалуйста!!!Клиент посылает произвольный набор латинских букв серверу и по-лучает их назад упорядоченными по алфавиту.


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

Или воспользуйтесь поиском по форуму:
10
Ответ Создать тему
Новые блоги и статьи
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
/ * Дана цепь постоянного тока с 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 из. . .
Classic Notepad for Windows 11
Jel 10.01.2026
Old Classic Notepad for Windows 11 Приложение для Windows 11, позволяющее пользователям вернуть классическую версию текстового редактора «Блокнот» из Windows 10. Программа предоставляет более. . .
Почему дизайн решает?
Neotwalker 09.01.2026
В современном мире, где конкуренция за внимание потребителя достигла пика, дизайн становится мощным инструментом для успеха бренда. Это не просто красивый внешний вид продукта или сайта — это. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru