- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在用 C 语言模拟聊天服务,我向服务器端发送一条消息,例如“U:用户名,密码;登录用户,但是当我询问谁在线时,它会打印 n 次(令 n 为用户数)在线)询问的用户名!什么不是预期的行为。有人可以告诉我为什么吗?
服务器.c
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <string.h>
#include <termios.h>
#include <linux/stat.h>
#define FIFO_FILE "MYFIFO"
typedef struct online {
char* user;
struct online* next;
} online;
typedef struct users{
char* name;
char* password;
} users;
online* begin=NULL;
online* end=NULL;
FILE *fp;
void start_server();
void add_online(char* user);
void realocate_end();
void remove_online(char* user);
char* user_access(char* s);
void who_online();
void free_all();
int find_username(char* user);
void add_user_regist(char* user,char* password);
users* find_user_regist(char* user, char* password);
int main(int argc, char *argv[])
{
if(argc>1){
char password[50];
char password2[50];
printf("Password? ");
scanf("%s",password);
printf("Repita a password? ");
scanf("%s",password2);
if(strcmp(password,password2)!=0){
printf("Passwords diferentes!\n");
exit(1);
}
if(find_username(argv[1])){
printf("Utilizador ja existe\n");
exit(1);
}
add_user_regist(argv[1],password);
printf("Utilizador %s adicionado.\n",argv[1]);
exit(0);
}
char readbuf[250];
/* Create the FIFO if it does not exist */
umask(0);
mknod(FIFO_FILE, S_IFIFO|0666, 0);
printf("Servidor iniciado.\nEm modo de espera de mensagens\n");
while(1)
{
fp = fopen(FIFO_FILE, "r");
fgets(readbuf, 250, fp);
fclose(fp);
if(readbuf[0]=='U')
user_access(readbuf);
if(readbuf[0]=='W')
who_online();
printf("Received string: %s\n",readbuf);
}
return(0);
}
void realocate_end(){
online* tmp=begin;
while(tmp!=NULL){
if(tmp->next==NULL)
end=tmp;
tmp=tmp->next;
}
}
void remove_online(char* user){
online* tmp=begin;
char path[100];
sprintf(path,"/tmp/chatroom_%s",user);
unlink(path);
while(tmp!=NULL){
if(tmp->user==user && tmp->user == end->user){
free(end);
end=NULL;
realocate_end();
}
else if(tmp->user==user && tmp->user == begin->user ){
begin=begin->next;
free(tmp);
break;
}
else if(tmp->next->user ==user){
online* tmp2 = tmp->next->next;
free(tmp->next);
tmp->next = tmp2;
realocate_end();
break;
}
tmp=tmp->next;
}
}
void add_online(char* user){
online* tmp = (online*)malloc(sizeof(online));
tmp->user = user;
tmp->next = NULL;
if(begin==NULL){
begin = tmp;
end=tmp;
}
else{
end->next=tmp;
end=end->next;
}
}
void who_online(){
online* tmp = begin;
while(tmp!=NULL){
printf("%s\n",tmp->user);
tmp=tmp->next;
}
}
void free_all(){
while(begin!=NULL){
online* tmp = begin->next;
free(begin);
begin = tmp;
}
}
int find_username(char* user){
FILE* fp;
char line[50];
fp=fopen("./regist", "r");
while(fscanf(fp,"%s",line)!=EOF){
char* name=strtok(line,",");
if(strcmp(name,user)==0){
fclose(fp);
return 1;
}
}
return 0;
}
char* user_access(char* s){
int i=0 ,begin=2;
strtok(s,":");
char* username=strtok(NULL,",");
char* password=strtok(NULL,";");
add_online(username);
}
users* find_user_regist(char* user, char* password){
FILE* fp;
char line[50];
users* tmp = (users*) malloc(sizeof(users));
fp=fopen("./regist", "r");
while(fscanf(fp,"%s",line)!=EOF){
tmp->name =strtok(line,",");
tmp->password = strtok(NULL,";");
if(strcmp(tmp->name,user)==0 && strcmp(tmp->password,password)==0 ){
fclose(fp);
return tmp;
}
}
fclose(fp);
return NULL;
}
void add_user_regist(char* user,char* password){
FILE* fp;
fp=fopen("./regist", "a");
if(fp==NULL)
fp=fopen("./regist", "w");
fprintf(fp,"%s,%s;\n",user,password);
fclose(fp);
}
客户端.c
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/stat.h>
#include <unistd.h>
#include <termios.h>
#include <linux/stat.h>
#define FIFO_FILE "MYFIFO"
int main(int argc, char *argv[])
{
FILE *fp;
char user[50];
char password[50];
if(argc==1){
printf("Não escolheu utilizador\n");
exit(0);
}
sprintf(user,"%s",argv[1]);
printf("Password:\n");
scanf("%s",password);
getchar();
char mensagem[250];
if((fp = fopen(FIFO_FILE, "w")) == NULL) {
perror("fopen");
exit(1);
}
char path [100] ;
strcpy(path,".");
strcat(path,user);
umask(0);
mknod(path, S_IFIFO|0666, 0);
sprintf(mensagem,"U:%s,%s;",user,password);
fputs(mensagem, fp);
fclose(fp);
while(1){
printf("**Menu**\n1) Listar utilizadores online\n2) Mandar SMS a um utilizador\n3) Logout\n\n");
char opcao;
scanf("%c",&opcao);
char lixo=getchar();
if(opcao!='1' && opcao!='2' && opcao!='3')
continue;
if(opcao=='1'){
if((fp = fopen(FIFO_FILE, "w")) == NULL) {
perror("fopen");
exit(1);
}
sprintf(mensagem,"W:%s;",user);
fputs(mensagem, fp);
fclose(fp);
}
else if(opcao=='2'){
if((fp = fopen(FIFO_FILE, "w")) == NULL) {
perror("fopen");
exit(1);
}
memset(mensagem,0,sizeof(mensagem));
scanf("%s",mensagem);
char lixo=getchar();
fputs(mensagem, fp);
fclose(fp);
}
else if(opcao=='3'){
break;
}
}
return(0);
}
最佳答案
这可能是由于您忘记分配新内存来存储每个用户的用户名造成的。现在的做法是为每个用户的字符串指针分配新的内存,而不是实际的字符串。
如果您通过函数调用跟踪存储在结构中的值的路径,您将看到每个记录都指向 readbuf
数组
考虑一下:
char readbuf[250] = "U:name,password;"
user_access(readbuf);
在user_access中,char *s
指向字符串的开头,即U
。第二次调用 strtok
返回指向用户名 n
开头的指针,并用空终止符替换 ,
。
在调用 add_online
时,缓冲区现在看起来像这样 "U\0name\0password\0\0"
和 username
code> 指向仍存储在 readbuf
中的 c 字符串 "name"
当您在链表中创建新节点时,您只需将指针分配给 readbuf 中的 c 字符串,因此用户名将根据缓冲区中的内容而变化。由于 W 命令只有一个字符,发送到服务器的最后一个用户名将在缓冲区中完好无损。
要解决此问题,您只需为每个用户名分配新内存并复制字符串即可。
void add_online(char* user){
online* tmp = (online*) malloc(sizeof(online));
tmp->user = (char*) malloc(strlen(user)*sizeof(char));
strcpy(tmp->user, user);
tmp->next = NULL;
if (begin==NULL){
begin = tmp;
end=tmp;
}
else {
end->next=tmp;
end=end->next;
}
}
同样,在释放结构时,您应该释放为用户名分配的内存
关于c - 为什么我的模拟聊天服务写了错误的内容?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37269380/
我正在为我的按钮使用 onClick 功能。我的按钮代码如下所示: Button 1 Button 2 我的 JS 函数如下所示: function fadeNext(selectedId, spee
首先,我想提一下,我理解每个人在不提供至少一些试验或错误的情况下提出问题的感受,但这纯粹是一种知识需求,话虽如此,我会去提前问。 我一直无法弄清楚如何将保存在 MySQL 表中的 600-1000 个
我想做的事情有点令人困惑,而且我英语不太好,所以我先把代码贴在这里,这样你就可以很容易地理解: 以下是表单内容: Testing for Stackoverflow Option1
我学习 SDL 二维编程已有一段时间了,现在我想创建一个结合使用 SDL 和 OpenGL 的程序。我是这样设置的: SDL_Init(SDL_INIT_VIDEO); window = SDL_Cr
我创建了 2 个 data-* 标签。数据类别和数据标签。单击 href 标签后,我想复制该数据类别和数据标签以形成输入。我的代码是:
我想用 CSS 换行。我正在使用内容。 td:before { content: "Test\A Test2"; } 它不工作。如何正确
这个问题已经有答案了: Java Class that implements Map and keeps insertion order? (8 个回答) 已关闭 6 年前。 我有一个 HashMap
我正在尝试使用 JMeter 执行端到端测试。测试涉及写入SFTP文件夹并从另一个SFTP文件夹读取写入操作生成的文件。 我能够使用 JMeter SSH SFTP 插件连接到 SFTP 文件夹,并能
您好,我有带有标准服务器端 Servlet 的 GWT 客户端。 我可以从 GWT 客户端上传文件并在服务器端读取其内容 我可以将其作为字符串发送回客户端 但是 我有 GWT FormPanel与操作
我在 Plone 4.3.9 中创建了一个自定义类型的灵巧性,称为 PersonalPage,必须只允许在特定文件夹中使用 成员文件夹/用户文件夹 . 在他的 FTI 中,默认情况下 False .
在新(更新)版本的应用程序中更改小部件布局的最佳做法是什么?当新版本提供更新、更好的小部件时,如何处理现有小部件? 最佳答案 我认为您必须向用户显示一个弹出窗口,说明“此版本中的新功能”并要求他们重新
在我的应用程序中,我使用支持 View 寻呼机和 PagerTabStrip。进入查看寻呼机我有一些 fragment ,进入其中一个我正在使用支持卡片 View 。运行应用程序后,所有卡片 View
我有以下布局文件。基本上我有谷歌地图,在左上角我有一个 TextView,我需要在其中每 15 秒保持一次计数器以刷新 map 。布局很好。
我使用如下结构: HashMap > > OverallMap 如果我这样做: OverallMap . clear ( ) clear() 丢弃的所有内容(HashMap 对象、Integer 对
我在数据库中有 1000 张图像。在页面加载时,我随机显示 60 张图片,当用户滚动时,我通过 AJAX 请求添加 20 张图片。 第一种方法 我所做的是将所有图像加载到一个容器中,然后隐藏所有图像并
我正在使用 woocommerce 创建一个网上商店。 我想在每个产品上添加一个包含产品信息的表格,例如颜色、交货时间等等。 但是当我添加这张表时。本产品消失后的所有内容。 我的表的代码: td {
This question already has an answer here: What does an empty value for the CSS property content do?
因此,我正在与我的 friend 一起为 Google Chrome 开发一个扩展程序,对于大多数功能(即日历、设置等),我们打开一个模式,这样我们就不必重定向到另一个页面。当您在内容之外单击时,我们
我将可变高度的 CSS 框设置为在更大的 div 中向左浮动。现在我想添加一个标题,其中文本在框的左侧垂直显示(旋转 90 度),如下面的链接所示(抱歉还不能发布图片)。 http://imagesh
相关页面位于 www.codykrauskopf.com/circus 如果您查看我页面的右侧,在半透明容器和浏览器窗口边缘之间有一个间隙。我看了看,出于某种原因,wrap、main、content
我是一名优秀的程序员,十分优秀!