Форум программистов, компьютерный форум, киберфорум
Visual Basic .NET
Войти
Регистрация
Восстановить пароль
Карта форума Темы раздела Блоги Сообщество Поиск Заказать работу  
 
 
Рейтинг 4.63/158: Рейтинг темы: голосов - 158, средняя оценка - 4.63
203 / 203 / 13
Регистрация: 14.10.2011
Сообщений: 227
1
.NET 4.x

Технология клиент-сервер. Классы клиента и сервера. Обсуждение

16.10.2011, 16:02. Показов 32749. Ответов 102
Метки нет (Все метки)

Author24 — интернет-сервис помощи студентам
В данном теме выкладываю свои классы клиента и сервера, а так же простеньких примеров реализованных на этих классах. Жду конструктивной критики, идей, предложений по улучшению, оптимизации и прочего.
Так же прошу поправлять меня, если комментарии в коде не соответствуют реальной действительности. Комментировал то как я это понимаю.
P.S. с орфографией тоже туговато, так что исправляем


Класс СЕРВЕРА.
VB.NET
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
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
Imports System.Net.Sockets
 
Public Class Class_Server
    Public Delegate Sub StatusInvoker(ByVal t As String) ' делегат для синхронизации при выводе сообщений
 
 
    Private port As Integer = 13000 ' порт сервера на котором он работает
    Private server_name As String ' название сервера. Например: Game, Authorization server . Просто для удобства
    Private flag As Byte ' в текущий версии не используется. В будущем планируется клиентов с другим флагом сразу отключать, что бы только "правильные" клиенты могли подключаться к серверу.
 
    Private server_Listener As TcpListener ' Сам TcpListener который и прослушивает нужный порт
    Private clientsList As New Hashtable ' Список клиентов. (Уникальный индификатор ID,сам класс клиента)
 
    Public Event OnRead(ByVal _client As handleClinet, ByVal _data As String) ' Событие вызываемое при получении данных от клиентов
    Public Event Connect(ByVal _client As handleClinet) ' Событие вызываемое при подключении клиента
    Public Event Disconnect(ByVal _client As handleClinet, ByVal _reson As String) ' Событие вызываемое при отключении клиента
    Public Event messege(ByVal text As String) ' событие вызываемое, когда нужно передать информацию из класса
 
    Public Sub New(ByVal _Port As Integer, ByVal _Name As String, ByVal _Flag As Byte) ' инициализация класса. Порт, название, флаг
        port = _Port
        server_name = _Name
        flag = _Flag
    End Sub
 
    Public Sub start() ' Запуск сервера.
        Dim server_Thread As Threading.Thread = New Threading.Thread(AddressOf startServ)
        server_Thread.Start() ' Запуск отдельного потока на прослушивание порта и подключений к серверу
    End Sub
 
    Public Sub stop_server() ' Процедура для закрытия сервера. 
        Dim Item As DictionaryEntry
        Dim list As Hashtable = clientsList
        For Each Item In list ' Перебор всех клиентов в списке
            CType(Item.Value, handleClinet).closeClient() ' Закрываем каждого клиента, что бы поток клиента завершился.
        Next
 
        server_Listener.Stop() ' Остановка прослушивания порта
    End Sub
 
    Public ReadOnly Property User_List() As Hashtable ' Возращаем список клиентов в виде хэш таблицы.
        Get
            Return clientsList
        End Get
    End Property
 
    Private Sub Client_OnRead(ByVal client As handleClinet, ByVal data As String) ' Процедура, вызываемая при получении сервером данных от клиента
        RaiseEvent OnRead(client, data) ' вызываем событие сервера, что клиент прислал данные
    End Sub
    Private Sub Client_Disconnect(ByVal client As handleClinet, ByVal reson As String) ' Процедура, вызываемая при отключении клиента
        clientsList.Remove(client.ID) ' удаляем клиента из списка клиентов
        RaiseEvent Disconnect(client, reson) ' вызываем событие сервера, что отключился клиент
    End Sub
 
    Public Sub Send_user(ByVal client As handleClinet, ByVal data As String) ' Процедура отправки данных нужному клиенту
        Dim tcp As TcpClient = client.TCP() ' получаем TcpClient нужный для получения потока записи
        Dim stream As NetworkStream = tcp.GetStream ' получение потока записи
 
        Dim mess() As Byte = System.Text.Encoding.UTF8.GetBytes(data) ' преобразование сообщения в байты для потока
 
        stream.Write(mess, 0, mess.Length) ' отсылка данных в виде байт
        stream.Flush() ' честно не знаю, что это делает. Работает и с этим и без, но в большинстве примеров есть, поэтому оставил.
    End Sub
    Public Sub Send_all(ByVal data As String) ' Процедура посылки сообщения всем клиентам
        Dim Item As DictionaryEntry
        Dim list As Hashtable = clientsList
 
        For Each Item In list ' Перебор всех клиентов и рассылка через Send_user
            Send_user(CType(Item.Value, handleClinet), data)
        Next
    End Sub
 
 
    Private Sub startServ() ' процедура служащая для прослушивания порта и подключении клиентов
        Dim client As TcpClient
        Dim count As Integer ' Кол-во подключений
 
        Try
            server_Listener = New TcpListener(port)
            server_Listener.Start() ' Начинаем прослушивать порт
            msg("=========Сервер " & Me.server_name & " запущен...=========")
 
 
            While True ' Бесконечный цикл, что бы клиенты могли подключаться всегда
                count += 1 ' Новое подключение, значит счетчик +1
 
                client = server_Listener.AcceptTcpClient() ' Ждем подключения. Когда клиент подключается, запоминаем его в client
 
                Dim temp_client As New handleClinet(client, count) ' Создаем новый класс клиента и передаем ему подключившегося клиента и его номер подключения
                temp_client.startClient(Me) ' Запускаем клиента и прослушивание входящих сообщений. Ссылку на класс сервера передаем, что бы клиент мог добавить своё событие на Client_OnRead
                clientsList.Add(temp_client.ID, temp_client) ' Добавляем клиента в список клиентов
 
                RaiseEvent Connect(temp_client) ' Вызываем событие, что клиент подключился
            End While
        Catch ex As Exception
            msg("=========Ошибка сервера " & Me.server_name & ". Причина: " & ex.Message) ' Сообщаем об ошибке
        End Try
 
        ' Так как сюда попасть можно, только после ошибки, значит сервер остановлен. Сообщаем об этом.
        msg("=========Сервер " & Me.server_name & " остановлен!=========")
 
    End Sub
 
    Public Sub msg(ByVal t As String) ' Процедура отправки сообщения
        Try
            RaiseEvent messege(t) ' Вызываем событие о новом сообщении
        Catch ex As Exception
            MsgBox(ex.Message, , "Msg Сервера")
        End Try
    End Sub
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
    Public Class handleClinet ' Класс клиента в сервере. Нужен для индификации и базовых дейсвий с клиентом 
        Public Event OnRead(ByVal _client As handleClinet, ByVal _data As String) ' Событие, что текущий клиент получил данные
        Public Event Disconnect(ByVal _client As handleClinet, ByVal _reson As String) ' Событие при отключении текущего клиента
 
        Private mgID As Guid = Guid.NewGuid ' Получаем уникальный индификатор клиента для хэш таблицы списка клиентов
        Private clientSocket As TcpClient ' Сокет клиента
        Private stream As NetworkStream ' Поток данных клиента
        Private clNo As String ' Номер клиента
        ' Private serv As Class_Server
        Private myIP As String ' Храним тут ип этого клиента
 
        Public Sub New(ByVal inClientSocket As TcpClient, ByVal count As Integer) ' Инициализация клиента
            Me.clientSocket = inClientSocket
            Me.stream = clientSocket.GetStream()
            Me.clNo = count
        End Sub
 
        Public ReadOnly Property IP() As String ' Получение IP
            Get
                If myIP = "" Then myIP = clientSocket.Client.RemoteEndPoint.ToString
                Return myIP
            End Get
        End Property
 
        Public ReadOnly Property ID() As String ' Получение ID
            Get
                Return mgID.ToString
            End Get
        End Property
        Public ReadOnly Property Number() As String ' Получение номера клиента 
            Get
                Return clNo.ToString
            End Get
        End Property
        Public ReadOnly Property TCP() As TcpClient ' Получение TcpClient клиента 
            Get
                Return clientSocket
            End Get
        End Property
        Public Sub closeClient() ' Процедура принудительного закрытия подключения клиента
            stream.Close()
            clientSocket.Close()
        End Sub
 
        Public Sub startClient(ByVal _serv As Class_Server) ' Процедура запуска клиента
            Dim client_Thread As Threading.Thread = New Threading.Thread(AddressOf doListen)
            client_Thread.Start() ' Запуск прослушивания новых данных текущим клиентом в отедьном потоке
 
            AddHandler Me.OnRead, AddressOf _serv.Client_OnRead ' При срабатывании события OnRead в клиенте, вызываем процедуру Client_OnRead на самом сервере
            AddHandler Me.Disconnect, AddressOf _serv.Client_Disconnect ' При срабатывании события Disconnect в клиенте, вызываем процедуру Client_Disconnect на самом сервере
        End Sub
        Private Sub doListen()
            Dim bytes(1048576) As Byte ' 1мб буфер, что бы данные не дробились.
            Dim data As String
            Dim i As Int32
 
            Try
                i = stream.Read(bytes, 0, bytes.Length) ' Ждем новых данных от клиентского приложения. При получении заносим данные в буфер bytes и длину данных в i
                While (i <> 0) ' Цикл бесконечный, пока длина считанных данных не окажется=0, что ознает, что клиент оключился.
                    data = System.Text.Encoding.UTF8.GetString(bytes, 0, i) ' Переводим байты в текст
                    RaiseEvent OnRead(Me, data) ' Вызываем событие о получении новых данных
 
                    i = stream.Read(bytes, 0, bytes.Length) ' Ждем новых данных от клиентского приложения. При получении заносим данные в буфер bytes и длину данных в i
                End While
                closeClient() ' Так как вышли из цикла, то клиентское приложение закрыло соединение. Следовательно закрываем и мы.
                RaiseEvent Disconnect(Me, "Клиент решил отключиться от сервера.") ' Вызываем событие, что клиент отключился и передаем причину.
 
            Catch ex As Exception
                closeClient() ' Ошибка, значит отключаем клиента.
                RaiseEvent Disconnect(Me, "Проблема соединения: " & ex.Message) ' Вызываем событие, что клиент отключился и передаем причину.
 
            End Try
 
        End Sub
    End Class
End Class



Класс КЛИЕНТА
VB.NET
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
Imports System.Net.Sockets
 
Public Class Class_client
    Public Delegate Sub StatusInvoker(ByVal t As String) ' делегат для синхронизации при выводе сообщений
 
    Dim client As TcpClient ' Текущий TcpClient связывающий клиента и сервер
    Dim stream As NetworkStream ' Поток данных связывающий клиента и сервер
    Dim flag As Byte ' Пока не используется. Нужен для индификации сервером "правильного" клиента
 
    Dim connection As Boolean = False ' Состояние подключения
 
 
    Public Event OnRead(ByVal _data As String) ' Событие, при получении данных от сервера
    Public Event Disconnected(ByVal _reson As String)  ' Событие, при отключении от сервера
    Public Event messege(ByVal text As String)  ' Событие, при необходимости передать сообщение из класса
    Dim s As Object
 
    Public Function Connect(ByVal server As String, ByVal port As Integer, ByVal _Flag As Byte) As Boolean ' Функция подключения клиента к серверу. Вернет True в случае удачи и False при ошибке.
        If connection = False Then ' Проверка не подключены ли мы уже?
            Try
                Me.client = New TcpClient(server, port)
                Me.stream = client.GetStream()
                Me.flag = _Flag
                Dim client_Thread As Threading.Thread = New Threading.Thread(AddressOf Me.doListen)
                client_Thread.Start() ' Запуск отдельного потока на прослушивание данных с сервера.
                connection = True ' Состояние=Подключен
                Return True
            Catch ex As Exception
                msg(ex.Message) ' Сообщаем об ошибке
                Return False
            End Try
        Else
            Return False ' Если уже подключены, тогда новое подключение не удалось
        End If
    End Function
 
    Public Sub Disconnect(Optional ByVal s As String = "") ' Процедура отключения от сервера. Возможно указать коментарий.
        If connection Then
            connection = False ' Состояние=Отключен
            stream.Close()
            client.Close()
            If s = "" Then
                RaiseEvent Disconnected("Отключен") ' Вызываем событие отключения с коментарием "по умолчанию"
            Else
                RaiseEvent Disconnected(s) ' Вызываем событие отключения с коментарием указанным при вызове текущей процедуры
            End If
 
        End If
    End Sub
 
    Public Function Send(ByVal data As String) As Boolean ' Функция отправки сообщения серверу. Вернет True, если посылка удалась или False в обратном случаее
        If connection Then
            Try
                Dim databyte() As Byte = System.Text.Encoding.UTF8.GetBytes(data) ' Создаем массив байтов для отправки из Data
                stream.Write(databyte, 0, databyte.Length) ' Отправляем наш массив байт.
                stream.Flush() ' честно не знаю, что это делает. Работает и с этим и без, но в большинстве примеров есть, поэтому оставил.
                Return True ' Говорим, что посылка удалась
            Catch ex As Exception
                msg(ex.Message) ' Сообщаем об ошибке
                Disconnect("Проблема при отправке.") ' Отключаемся
                Return False
            End Try
        Else
            Return False ' Если не подключены, то отправка не удалась.
        End If
    End Function
 
 
    Private Sub doListen() ' Процедура прослушки сообщений от сервера
 
 
        Dim bytes(1048576) As Byte ' 1мб буфер, чтобы данные не дробились.
        Dim data As String
        Dim i As Int32
 
        Try
            i = stream.Read(bytes, 0, bytes.Length) ' Ждем новых данных от сервера. При получении заносим данные в буфер bytes и длину данных в i
            While (i <> 0) ' Цикл бесконечный, пока длина считанных данных не окажется=0, что ознает, что соединение с сервером прекращенно.
                data = System.Text.Encoding.UTF8.GetString(bytes, 0, i) ' Переводим байты в текст
                RaiseEvent OnRead(data) ' Вызываем событие о получении новых данных
 
                i = stream.Read(bytes, 0, bytes.Length) ' Ждем новых данных от сервера. При получении заносим данные в буфер bytes и длину данных в i
            End While
            Disconnect() ' Так как вышли из цикла, то сервер закрыло соединение. Следовательно отключаемся.
        Catch ex As Exception
            Disconnect(ex.Message) ' Ошибка, значит отключаемся, указав причину.
        End Try
 
    End Sub
 
 
 
    Public Sub msg(ByVal t As String) ' Процедура отправки сообщения
        Try
            RaiseEvent messege(t)  ' Вызываем событие о новом сообщении
        Catch ex As Exception
            MsgBox(ex.Message)
        End Try
    End Sub
End Class


Сообщение в разработке. Примеры будут чуть позже.
35
Programming
Эксперт
94731 / 64177 / 26122
Регистрация: 12.04.2006
Сообщений: 116,782
16.10.2011, 16:02
Ответы с готовыми решениями:

Как вызвать Disconnect клиента (Технология Клиент-Сервер)
Как на сервере вызвать дисконнект клиента на пример когда произойдет тик таймера, пробовал многими...

Чат "Сервер-клиент". На сервер не могу отправить сообщение с клиента
Не могу понять как сделать, чтоб сервер ещё прослушивал и сообщения... Нет ли входящих данных....

Http сервер и клиент. При попытке получить ответ клиента от сервера ошибка: Поток был недоступен для чтения
Здравствуйте, пытаюсь разобраться с написанием клиент-серверного приложения используя сокеты....

Технология создания приложений на основе многоуревневой архитектуры клиент - Web-сервер - сервер баз данных
Технология создания приложений на основе многоуревневой архитектуры клиент Web-сервер – сервер...

102
Студент :)
895 / 328 / 12
Регистрация: 29.01.2011
Сообщений: 1,679
17.10.2011, 19:08 21
Author24 — интернет-сервис помощи студентам
Treals, Да.
0
203 / 203 / 13
Регистрация: 14.10.2011
Сообщений: 227
17.10.2011, 19:11  [ТС] 22
VB.NET
1
Т.е без перенастройки сервера могут зайти как с локалки так и с внешки?
Да.
VB.NET
1
server_Listener = New TcpListener(port)
Если изменить на
VB.NET
1
server_Listener = New TcpListener(ip,port)
То можно будет слушать только этот ип
3
30 / 30 / 3
Регистрация: 25.05.2011
Сообщений: 308
17.10.2011, 23:39 23
А такой вопрос... Как отправить определенную команду на клиент который был запущен с определенного IP?

Допустим клиент запустили с 5-ти разных компов... На сервер мне пришли их IP... И мне надо на определенный IP допустим послать команду на закрытие того клиента...
0
203 / 203 / 13
Регистрация: 14.10.2011
Сообщений: 227
19.10.2011, 11:12  [ТС] 24
Лучший ответ Сообщение было отмечено как решение

Решение

VB.NET
1
2
3
4
5
6
7
Dim Item As DictionaryEntry
        Dim list As Hashtable = server.User_List
 dim client as server.handleClinet
        For Each Item In list ' Перебор всех клиентов и рассылка через Send_user
client=CType(Item.Value, server.handleClinet)
if client.ip = "Нужный ип" then server.Send_user(client, "нужная команда")
        Next
5
30 / 30 / 3
Регистрация: 25.05.2011
Сообщений: 308
20.10.2011, 19:57 25
Простите мою тупрость... Но я так и не понял как при этих классах написать команду от сервера клиенту что бы клиент закрывался ...
0
203 / 203 / 13
Регистрация: 14.10.2011
Сообщений: 227
20.10.2011, 20:21  [ТС] 26
Лучший ответ Сообщение было отмечено как решение

Решение

Цитата Сообщение от Treals Посмотреть сообщение
Простите мою тупрость... Но я так и не понял как при этих классах написать команду от сервера клиенту что бы клиент закрывался ...
Меняем код клиента.
VB.NET
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
Private Sub OnRead(ByVal data As String)
        ' Процедура расшифровки принятых данных
        Dim d() As String
        ' Каждый отдельный пакет должен заканчиваться ";",
        ' что бы была возможность в случаее сильной нагрузки на канал,
        ' принимать по нескольку пакетов одновременно
 
        d = Split(data, ";") ' Разбиваем сообщение по ";"
        For x = 0 To d.Length - 1 ' Перебираем все полученные пакеты
            If d(x) <> "" Then ' Если пакет не пустой делаем его обработку
 
 
                If Mid(d(x), 1, 9) = "PING TEST" Then ' Если пакет проверки пинга,  
                    st.Stop() 'то останавливаем стопвотчер
                    msg("Пинг: " & st.ElapsedMilliseconds) 'и выводим пинг в милисекундах
                Elseif Mid(d(x), 1, 5) = "CLOSE" ' Если пакет закрытия, то закрываем клиент.
                    end ' закрываем клиент.
                Else ' Иначе
                    msg("Сервер: " & d(x)) ' просто выводим сообщение с сервера
                End If
 
 
            End If
        Next
 
    End Sub

А на сервере просто отправляем команду "CLOSE;" всем или нужному клиенту.
Клиент получив и проверив пакет, делает нужные действия. Обработав в данном случае
VB.NET
1
2
Elseif Mid(d(x), 1, 5) = "CLOSE" ' Если пакет закрытия, то закрываем клиент.
                    end ' закрываем клиент.
клиент закроется.
3
37 / 37 / 2
Регистрация: 11.11.2011
Сообщений: 423
17.02.2012, 14:11 27
Я так понимаю,нужно привязывать каждую процедуру к элементу управления.Это так?
0
12 / 12 / 0
Регистрация: 06.05.2011
Сообщений: 250
19.02.2012, 00:57 28
почему когда я закрываю сервер он в процесе висит?
0
_
2364 / 1243 / 78
Регистрация: 28.10.2009
Сообщений: 4,331
19.02.2012, 01:15 29
chiribas, при закрытии формы - событие FormClosing закрывай все сокеты и соединения, да и до закрытия формы тоже старайися закрыть все подключения
1
12 / 12 / 0
Регистрация: 06.05.2011
Сообщений: 250
19.02.2012, 11:58 30
Почему на одном компьютере соединение устанавливается, а на разных нет?
На клиенте
VB.NET
1
CLIENT.Connect("внутренний ip сервера", 10000, AscW("+"))
пробовал и внешний ip, и имя компьютера.
0
203 / 203 / 13
Регистрация: 14.10.2011
Сообщений: 227
19.02.2012, 14:18  [ТС] 31
Почему на одном компьютере соединение устанавливается, а на разных нет?
В локальной сети? На сервере порт 10000 прослушивается? Порты в сети открыты?
0
12 / 12 / 0
Регистрация: 06.05.2011
Сообщений: 250
19.02.2012, 15:30 32
Цитата Сообщение от KingManiya Посмотреть сообщение
В локальной сети?
в глобальной
0
203 / 203 / 13
Регистрация: 14.10.2011
Сообщений: 227
20.02.2012, 13:26  [ТС] 33
Нужны хотя бы ошибки, больше информации. В интернете данные клиент\сервер проверялись, всё работало. Смотрите открытость портов. Не находится ли сервер за NATом.
0
12 / 12 / 0
Регистрация: 06.05.2011
Сообщений: 250
20.02.2012, 14:41 34
Если после того как я запустил сервер в котором
VB.NET
1
CLIENT.Connect("ip", 34567, AscW("+"))
а потом на этом сайте http://speed-tester.info/check_port.php ввожу 34567 и написано "Порт 34567 закрыт", то в сервере чтото не правильно?

Добавлено через 52 минуты
а если сервер находится за nat'ом, то что делать?(у меня usb мегафон-модем)
0
203 / 203 / 13
Регистрация: 14.10.2011
Сообщений: 227
21.02.2012, 03:37  [ТС] 35
а если сервер находится за nat'ом, то что делать?(у меня usb мегафон-модем)
Скорее всего, сервер не сможешь работать за NATом. Так какой такой ип как у сервера, будет и у других компьютеров в сети. Поэтому подключатся не известно куда клиент не будет. А вот клиент быть за NATом может, так как ответ от сервера идет не по ип клиента, а по уже установленному соединению.

а потом на этом сайте http://speed-tester.info/check_port.php ввожу 34567 и написано "Порт 34567 закрыт", то в сервере чтото не правильно?
Порт закрыт пишет либо из за NATа, либо из за закрытых портов в брандмауэре. В любом случае клиент, не сможет подключится, при таком сообщении.
1
43 / 43 / 5
Регистрация: 12.11.2011
Сообщений: 444
11.03.2012, 22:06 36
У меня иногда крашится сервер
вылетает на этой строке такое сообщение Ошибка Коллекция была изменена; невозможно выполнить операцию перечисления.

VB.NET
1
2
3
4
5
6
7
8
    Public Sub Send_all(ByVal data As String) ' Процедура посылки сообщения всем клиентам
        Dim Item As DictionaryEntry
        Dim list As Hashtable = clientsList
 
        For Each Item In list ' Перебор всех клиентов и рассылка через Send_user
            Send_user(CType(Item.Value, handleClinet), data)
        Next
    End Sub
и иногда на

VB.NET
1
2
3
4
5
6
7
8
9
10
    Public Sub Send_user(ByVal client As handleClinet, ByVal data As String) ' Процедура отправки данных нужному клиенту
            Dim tcp As TcpClient = client.TCP() ' получаем TcpClient нужный для получения потока записи
            Dim stream As NetworkStream = tcp.GetStream ' получение потока записи
 
            Dim mess() As Byte = System.Text.Encoding.UTF8.GetBytes(data) ' преобразование сообщения в байты для потока
 
            stream.Write(mess, 0, mess.Length) ' отсылка данных в виде байт
 
            stream.Flush()
    End Sub
stream.Flush() 'Вот тут вылетает: Операция не разрешена на неподключенных сокетах
0
2 / 2 / 0
Регистрация: 28.11.2010
Сообщений: 14
14.03.2012, 18:49 37
Классы бесподобны, одна радость!
но...

Сколько не бился, не могу заставить выполнять какие-либо действия с объектами на форме, или чем то подобным..
например,
обрабатываю событие OnRead на сервере, смотрю, если data = HIDE, выполняю Form2.Hide, из-за чего клиент грандиозно отрубается, а сервер ругается:
Проблема соединения: An error occurred creating the form. See Exception.InnerException for details. The error is: ActiveX control '6bf52a52-394a-11d3-b153-00c04f79faa6' cannot be instantiated because the current thread is not in a single-threaded apartment.

ну или например если я хочу задать компоненту AxMediaPlayer свойство settings.volume=int(data)
такой же вылет..

при этом
msgbox(data)
или form2.textbox1.text=data, к примеру работает.
но, если же в этом случает отслеживать событие onchanged и передавать текст дальше в свои функции-опять вылет...


Каааааак, объясните мне, как мне победить эту запару с потоками??
может попробовать реализовать псевдомультипоточность через таймер, вместо бесконечного цикла в отдельном потоке? подскажите пожалуйста!
0
203 / 203 / 13
Регистрация: 14.10.2011
Сообщений: 227
14.03.2012, 19:26  [ТС] 38
может попробовать реализовать псевдомультипоточность через таймер, вместо бесконечного цикла в отдельном потоке? подскажите пожалуйста!
Без потока, во время ожидания подключения происходит зависание.

Сколько не бился, не могу заставить выполнять какие-либо действия с объектами на форме, или чем то подобным..
например,
обрабатываю событие OnRead на сервере, смотрю, если data = HIDE, выполняю Form2.Hide, из-за чего клиент грандиозно отрубается, а сервер ругается:
Проблема соединения: An error occurred creating the form. See Exception.InnerException for details. The error is: ActiveX control '6bf52a52-394a-11d3-b153-00c04f79faa6'
По подобию с
VB.NET
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
    Private Sub Disconnect(ByVal reson As String)
        ' Процедура при отключении клиента
        msg("Подключение разорванно! Причина: " & reson)
        Button1.Enabled = True
        Button2.Enabled = False
        Button3.Enabled = False
        Button4.Enabled = False
    End Sub
    Private Sub Disconnect_Invoke(ByVal t As String) ' Процедура отключения от сервера с синхронизацией потоков
        Try
            If Me.IsHandleCreated Then
                Me.Invoke(New Class_client.StatusInvoker(AddressOf Me.Disconnect), t)
            End If
        Catch ex As Exception
            MsgBox(ex.Message)
        End Try
    End Sub
разве не получается?
0
2 / 2 / 0
Регистрация: 28.11.2010
Сообщений: 14
14.03.2012, 19:58 39
VB.NET
1
2
3
4
5
6
7
8
9
Private Sub OnRead(ByVal client As Class_Server.handleClinet, ByVal data As String)
        If (data = "HIDE ;") Then
Form1.Invoke(New StatusInvoker(AddressOf test2222))
        End If
End Sub
 
Public Sub test2222()
Form1.Hide()
End Sub
нет, вылетает

Добавлено через 21 минуту
VB.NET
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
    Private Sub OnRead(ByVal client As Class_Server.handleClinet, ByVal data As String)
        If (data = "HIDE ;") Then
            hider_invoke()
        End If
    End Sub
 
    Sub hider_invoke(ByVal t As String)
        Try
            If Me.IsHandleCreated Then
                Me.Invoke(Class_Server.(AddressOf Me.hider), t)
            End If
        Catch ex As Exception
            MsgBox(ex.Message)
        End Try
    End Sub
 
    Sub hider(ByVal t As String)
        Form1.Hide()
    End Sub
Это все дело на форме
Я на верном пути? что должно быть в "t"?
0
203 / 203 / 13
Регистрация: 14.10.2011
Сообщений: 227
14.03.2012, 20:11  [ТС] 40
Я на верном пути?
вроде да...
что должно быть в "t"?
Это просто вспомогательная переменная, для передачи информации в процедуру.
VB.NET
1
   hider_invoke("")
Или можно везде удалить t, вплоть до делегата..

И возможно(!):
VB.NET
1
2
3
4
5
6
7
8
9
 Sub hider_invoke(ByVal t As String)
        Try
            If Form1.IsHandleCreated Then
                Form1.Invoke(Class_Server.(AddressOf me.hider), t)
            End If
        Catch ex As Exception
            MsgBox(ex.Message)
        End Try
    End Sub
Так как нужно синхронизировать поток той формы, в которой происходит изменение.
0
14.03.2012, 20:11
IT_Exp
Эксперт
87844 / 49110 / 22898
Регистрация: 17.06.2006
Сообщений: 92,604
14.03.2012, 20:11
Помогаю со студенческими работами здесь

Технология клиент-сервер
Кто может подробно объяснить клиент-сервер технологию? Я не понимаю некоторые вещи . Например , я...

Клиент-сервер, передать строку с клиента на сервер
Подскажите плиз Есть код он передает серверу то что мы пишем с клавиатуры а как передать строку...

Клиент-Сервер. Распознавание клиента
День добрый. Подскажите пожалуйста, как сделать так, чтобы при подключении к серверу не одного, а...

Получить ip адрес клиента (асинхронный клиент - сервер)
Не могу получить ip адрес клиента (асинхронный клиент - сервер) при получении сообщения на стороне...


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

Или воспользуйтесь поиском по форуму:
40
Ответ Создать тему
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2024, CyberForum.ru