用GTK和socket实现简单的聊天室,出现“段错误”。是哪里的问题??
麻烦看看我的程序,我要用GTK图形界面做一个简单的聊天室,在两台机器上实现通信。现在程序可以编译过去,但是运行时,当客户段弹出登录界面时,输入用户名再点击确定的时候,就在终端显示“段错误”。我是新手,刚刚入门,向前辈们请教,盼回复。
服务器端:
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#define OUTPORT 3333
#define MAX_USERS 8
struct _client{
gint sd;
gboolean in_use;
gchar name[64];
gchar buf[1024];
};
typedef struct _client client;
client user[MAX_USERS];
void do_service(gpointer id)
{
gint j;
char tobuf[1024];
while(read(user[GPOINTER_TO_INT(id)].sd,
user[GPOINTER_TO_INT(id)].buf,1024)!=-1)
{
sprintf(tobuf,"%s: %s\n",user[GPOINTER_TO_INT(id)].name,
user[GPOINTER_TO_INT(id)].buf);
for(j=0;j {
if(user[j].in_use)
{
write(user[j].sd,tobuf,1024);
g_printf("%s",tobuf);
}
}
}
user[GPOINTER_TO_INT(id)].in_use=FALSE;
close(user[GPOINTER_TO_INT(id)].sd);
//exit(0);
}
int main(int agrv,char *argv[])
{
gint sd,newsd;
struct sockaddr_in *my_addr; /* 本机地址信息 */
struct sockaddr_in *remote_addr; /* 客户端地址信息 */
gint slen;
gint count=0;
gint flags;
gchar buf[1024];
gchar tobuf[1024];
gint length,i,j;
if(!g_thread_supported())
g_thread_init(NULL);
else
g_print("thread not support\n");
sd=socket(AF_INET,SOCK_STREAM,0);
if(sd==-1)
{
g_print("Creat socket error!\n");
return -1;
}
my_addr=g_new(struct sockaddr_in,1);
my_addr->sin_family=AF_INET;
my_addr->sin_port=htons(OUTPORT);
my_addr->sin_addr.s_addr = INADDR_ANY;
bzero( &(my_addr->sin_zero),8);
slen=sizeof(struct sockaddr_in);
if(bind(sd,(struct sockaddr *) my_addr,slen)<0)
{
g_print("bind error\n!");
return -1;
}
if(listen(sd,8)<0)
{
g_print("listen error!\n");
}
for(i=0;i user.in_use=FALSE;
flags=fcntl(sd,F_GETFL);
fcntl(sd,F_GETFL,flags&~O_NDELAY);
for(;;)
{
newsd=accept(sd,(struct sockaddr *) remote_addr, &slen);
if(newsd==-1)
{
g_print("accept error\n");
break;
}
else
{
if(count>=MAX_USERS)
{
sprintf(buf,"用户数量过多,服务器不能连接!\n");
write(newsd,buf,1024);
close(newsd);
}
else
{
flags=fcntl(user.sd,F_GETFL);
fcntl(user.sd,F_SETFL,O_NONBLOCK);
user[count].sd=newsd;
user[count].in_use=TRUE;
read(newsd,user[count].name,64);
g_thread_create((GThreadFunc)do_service,
(gpointer)count,TRUE,NULL);
count++;
}
}
}//for(;;)
close(sd);
g_free(my_addr);
}
服务器端:
#include
#include
#include
#include
#include
#include
#include
#include
#include
#define OUTPORT 3333
gint sd;
struct sockaddr_in s_in;
struct hostent *host;
gchar username[64];
gchar buf[1024];//read
gchar get_buf[1048];//write
gboolean isconnected=FALSE;
static GtkWidget *text;
static GtkTextBuffer *buffer;
static GtkWidget *message_entry;
static GtkWidget *name_entry;
static GtkWidget *login_button;
void get_message(void)
{
GtkTextIter iter;
gchar get_buf[1024];
gchar buf[1024];
while(read(sd,buf,1024)!=-1)
{
sprintf(get_buf,"%s",buf);
gdk_threads_enter();
gtk_text_buffer_get_end_iter(buffer,&iter);
gtk_text_buffer_insert(buffer,&iter,get_buf,-1);
gdk_threads_leave();
}
}
void on_destroy(GtkWidget *widget,GdkEvent *event,gpointer data)
{
sprintf(username,"guest");
if(do_connect()==TRUE)
{
gtk_widget_set_sensitive(login_button,FALSE);
g_thread_create((GThreadFunc)get_message,NULL,FALSE,NULL);
}
gtk_widget_destroy(widget);
}
void on_button_clicked(GtkButton *button,gpointer data)
{
const gchar *name;
name=gtk_entry_get_text(GTK_ENTRY(name_entry));
sprintf(username,"%s",name);
if(do_connect())
{
gtk_widget_set_sensitive(login_button,FALSE);
g_thread_create((GThreadFunc)get_message,NULL,FALSE,NULL);
}
gtk_widget_destroy(data);
}
void creat_win(void)
{
GtkWidget *win,*vbox;
GtkWidget *button;
win=gtk_window_new(GTK_WINDOW_TOPLEVEL);
gtk_window_set_title(GTK_WINDOW(win),"输入用户名");
gtk_container_set_border_width(GTK_CONTAINER(win),10);
g_signal_connect(G_OBJECT(win),"delete_event",
G_CALLBACK(on_destroy),NULL);
gtk_window_set_modal(GTK_WINDOW(win),TRUE);
gtk_window_set_position(GTK_WINDOW(win),GTK_WIN_POS_CENTER);
vbox= gtk_vbox_new(FALSE,0);
gtk_container_add(GTK_CONTAINER(win),vbox);
name_entry=gtk_entry_new();
gtk_box_pack_start(GTK_BOX(vbox),name_entry,TRUE,TRUE,5);
button=gtk_button_new_from_stock(GTK_STOCK_OK);
g_signal_connect(G_OBJECT(button),"clicked",
G_CALLBACK(on_button_clicked),win);
gtk_box_pack_start(GTK_BOX(vbox),button,FALSE,FALSE,5);
gtk_widget_show_all(win);
}
gboolean do_connect(void)
{
GtkTextIter iter;
gint slen;
sd=socket(AF_INET,SOCK_STREAM,0);
if(sd<0)
{
gtk_text_buffer_get_end_iter(buffer,&iter);
gtk_text_buffer_insert(buffer,&iter,"打开套接字时出错!\n",-1);
return FALSE;
}
s_in.sin_family=AF_INET;
s_in.sin_port=htons(OUTPORT);
s_in.sin_addr=*((struct in_addr *)host->h_addr);
bzero(&(s_in.sin_zero),8);
slen=sizeof(s_in);
if(connect(sd,(struct sockaddr *) &s_in,slen)<0)
{
gtk_text_buffer_get_end_iter(buffer,&iter);
gtk_text_buffer_insert(buffer,&iter,"连接服务器时出错!\n",-1);
return FALSE;
}
else
{
gtk_text_buffer_get_end_iter(buffer,&iter);
gtk_text_buffer_insert(buffer,&iter,username,-1);
gtk_text_buffer_get_end_iter(buffer,&iter);
gtk_text_buffer_insert(buffer,&iter,"\n成功于服务器连接!\n",-1);
//
write(sd,username,64);
//
isconnected=TRUE;
return TRUE;
}
}
void on_send(GtkButton *button,gpointer data)
{
const char *message;
if(isconnected==FALSE) return;
message=gtk_entry_get_text(GTK_ENTRY(message_entry));
sprintf(buf,"%s\n",message);
write(sd,buf,1024);
gtk_entry_set_text(GTK_ENTRY(message_entry),"");
}
void on_login(GtkWidget *widget,GdkEvent *weent,gpointer data)
{
creat_win();
}
void on_delete_event(GtkWidget *widget,GdkEvent *event,gpointer data)
{
close(sd);
gtk_main_quit();
}
int main(int argc,char *argv[])
{
GtkWidget *window;
GtkWidget *vbox,*hbox,*button,*label,*view;
if(!g_thread_supported())
g_thread_init(NULL);
gtk_init(&argc,&argv);
window=gtk_window_new(GTK_WINDOW_TOPLEVEL);
gtk_window_set_title(GTK_WINDOW(window),"客户端");
g_signal_connect(G_OBJECT(window),"delete_event",
G_CALLBACK(on_delete_event),NULL);
gtk_container_set_border_width(GTK_CONTAINER(window),10);
vbox=gtk_vbox_new(FALSE,0);
gtk_container_add(GTK_CONTAINER(window),vbox);
hbox=gtk_hbox_new(FALSE,0);
gtk_box_pack_start(GTK_BOX(vbox),hbox,FALSE,FALSE,5);
label=gtk_label_new("点击登录按钮连接服务器");
gtk_box_pack_start(GTK_BOX(hbox),label,FALSE,FALSE,5);
login_button=gtk_button_new_with_label("登录");
gtk_box_pack_start(GTK_BOX(hbox),login_button,FALSE,FALSE,5);
g_signal_connect(G_OBJECT(login_button),"clicked",
G_CALLBACK(on_login),NULL);
view=gtk_scrolled_window_new(NULL,NULL);
gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(view),
GTK_POLICY_AUTOMATIC,GTK_POLICY_AUTOMATIC);
text=gtk_text_view_new();
gtk_box_pack_start(GTK_BOX(vbox),view,TRUE,TRUE,5);
gtk_container_add(GTK_CONTAINER(view),text);
buffer=gtk_text_view_get_buffer(GTK_TEXT_VIEW(text));
hbox=gtk_hbox_new(FALSE,0);
gtk_box_pack_start(GTK_BOX(vbox),hbox,FALSE,FALSE,5);
label=gtk_label_new("输入:");
gtk_box_pack_start(GTK_BOX(hbox),label,FALSE,FALSE,5);
message_entry=gtk_entry_new();
gtk_box_pack_start(GTK_BOX(hbox),message_entry,FALSE,FALSE,5);
button=gtk_button_new_with_label("发送");
gtk_box_pack_start(GTK_BOX(hbox),button,FALSE,FALSE,5);
g_signal_connect(G_OBJECT(button),"clicked",
G_CALLBACK(on_send),NULL);
gtk_widget_show_all(window);
gdk_threads_enter();
gtk_main();
gdk_threads_leave();
return FALSE;
}