C_and_CPP 板


LINE

※ 引述《hth9494 (掰掰惹 仙度瑞拉)》之铭言: : 开发平台(Platform): (Ex: Win10, Linux, ...) : Linux kali 5.18.0-kali2-amd64 #1 SMP PREEMPT_DYNAMIC Debian 5.18.5-1kali1 : (2022-06-20) x86_64 GNU/Linux : 编译器(Ex: GCC, clang, VC++...)+目标环境(跟开发平台不同的话需列出) : gcc (Debian 11.3.0-3) 11.3.0 : 额外使用到的函数库(Library Used): (Ex: OpenGL, ...) : 问题(Question): : 我有两个process开启同一个file,且两个process同时向file写入字串 : 一个写入"a",一个写入"b",各自重复写入200次 : 每一次写入,两个process就会printf出当前ftell的值 : 问题: : 我期待看到file中ab会交替出现 : 但并没有,而是a全部出现完才换b,或者b出现完才换a : 不过从console上印出的ftell值却又显示 : 这两个process显然是交替执行着,并不是一个执行完才换另一个 : 既然如此,为什麽file的内容不是交替的显示a和b呢 : 请问要如何做才能看到ab交替的结果 : 谢谢 : 喂入的资料(Input): : 预期的正确结果(Expected Output): : file中a和b交替显示 : 错误结果(Wrong Output): : 200个a全部显示完才换b,或者200个b显示完才换a : 程式码(Code):(请善用置底文网页, 记得排版,禁止使用图档) : #include <stdio.h> : #include <stdlib.h> : #include <string.h> : int main() : { : FILE *pfile; : int id = fork(); : if (id == 0) { : char *str = "a"; : pfile = fopen("testfile", "a"); : if (pfile) { : int i; : for (i = 0; i < 200; i++) { : printf("a = %d\n", ftell(pfile)); : fwrite(str, 1, strlen(str), pfile); : } : } : } : else if (id > 0) { : printf("id = %d\n", id); : char *str = "b"; : pfile = fopen("testfile", "a"); : if (pfile) { : int i; : for (i=0;i<200;i++) { : printf("b = %d\n", ftell(pfile)); : fwrite(str, 1, strlen(str), pfile); : } : } : } : fclose(pfile); : return 0; : } : 补充说明(Supplement): 抱歉,我原先的问题描述不准确 我原文所谓的交替是指,我预期a和b会有不规则穿插的效果 像是aaabbaaaaaabbbbbabababaaa 而非abababababababababababab 因为本来只是想练习fork()和感受一下race condition的效果 结果因为用fwrite导致档案内容不如预期才上来发问 但看了板友的推文,我就也想练习一下要怎麽做才能abababababababab 因为目前还不会IPC或condition variable之类的观念 所以只想到笨方法 现在变成有两个档案,原先的testfile和一个新的check 当process a写入testfile後,会在check写下一个属於a的标记,表示已经写入testfile 当process b写入testfile後,亦然 process a 在写入档案前 先不断去判断check,看看b是否有写入标记了 如果确认b已写入,才会写入testfile process b 也做一样的动作 我执行後这样就能达成ababab的效果 前提要先建立check并且给一个初始的标记 我大概知道这样会有一些问题,像是a跟b要不停去检查check,而非等待通知再执行 导致浪费CPU资源等等 但我这个作法除了这个问题外,是否已能保证执行结果一定是abababab呢 是否要给check加锁什麽的,否则在check上也会发生race condition的问题 导致结果不是abababab? 谢谢各位 #include <stdio.h> #include <unistd.h> #include <fcntl.h> #include <stdlib.h> #include <string.h> int main() { FILE *pfile = NULL; int id = fork(); if (id == 0) { // process a, keep writting a char *str = "a"; pfile = fopen("testfile", "a"); if (pfile) { int fd = fileno(pfile); int i; for (i = 0; i < 200; i++) { // keep checking until b is written char c = 0; while (c != 'b') { FILE *pcheck = fopen("check", "r"); int fdcheck = fileno(pcheck); read(fdcheck, &c, 1); fclose(pcheck); } // write a printf("a = %d\n", ftell(pfile)); write(fd, str, 1); // update check FILE *pcheck = fopen("check", "w"); int fdcheck = fileno(pcheck); write(fdcheck, str, 1); fclose(pcheck); } } } else if (id > 0) { // process b , keep writting b char *str = "b"; pfile = fopen("testfile", "a"); if (pfile) { int fd = fileno(pfile); int i; for (i=0;i<200;i++) { // keep checking until a is written char c = 0; while (c != 'a') { FILE *pcheck = fopen("check", "r"); int fdcheck = fileno(pcheck); read(fdcheck, &c, 1); fclose(pcheck); } // write b printf("b = %d\n", ftell(pfile)); write(fd, str, 1); // update check FILE *pcheck = fopen("check", "w"); int fdcheck = fileno(pcheck); write(fdcheck, str, 1); fclose(pcheck); } } } if (pfile) fclose(pfile); return 0; } --



※ 发信站: 批踢踢实业坊(ptt.cc), 来自: 125.229.74.160 (台湾)
※ 文章网址: https://webptt.com/cn.aspx?n=bbs/C_and_CPP/M.1657513524.A.C18.html
1F:推 qscgy4: 如果只是想看 abbababbbaa 那你直接printf或cout都能看 07/11 13:45
2F:→ qscgy4: 如果想要在档案看到,那就多写一个thread来接收 07/11 13:45
3F:→ qscgy4: 那个thread就按接收顺序写入 07/11 13:46
4F:→ qscgy4: 这样应该最简单 07/11 13:47
5F:推 wulouise: 你的实作在两个producer的时候没事,多个的时候不好处理 07/12 07:50
6F:→ longlongint: 有试过 sync write 吗 07/12 14:08
7F:→ WPC001: 要不规则穿插... 除非写入内容很长... 或是每写一个byte就 07/15 21:23
8F:→ WPC001: 主动睡下去... 07/15 21:23
9F:→ WPC001: 基本上你有fclose... 应该是已经被flush了,多加个sleep吧 07/15 21:25
10F:→ choosin: file本身就是IPC的一种 要abab更简单 开档检查结尾写回就 07/19 11:49
11F:→ choosin: 好 依照你本来想写200a跟200b 那最後如果没有200a+200b那 07/19 11:49
12F:→ choosin: 也可以看到race condition的发生 07/19 11:50







like.gif 您可能会有兴趣的文章
icon.png[问题/行为] 猫晚上进房间会不会有憋尿问题
icon.pngRe: [闲聊] 选了错误的女孩成为魔法少女 XDDDDDDDDDD
icon.png[正妹] 瑞典 一张
icon.png[心得] EMS高领长版毛衣.墨小楼MC1002
icon.png[分享] 丹龙隔热纸GE55+33+22
icon.png[问题] 清洗洗衣机
icon.png[寻物] 窗台下的空间
icon.png[闲聊] 双极の女神1 木魔爵
icon.png[售车] 新竹 1997 march 1297cc 白色 四门
icon.png[讨论] 能从照片感受到摄影者心情吗
icon.png[狂贺] 贺贺贺贺 贺!岛村卯月!总选举NO.1
icon.png[难过] 羡慕白皮肤的女生
icon.png阅读文章
icon.png[黑特]
icon.png[问题] SBK S1安装於安全帽位置
icon.png[分享] 旧woo100绝版开箱!!
icon.pngRe: [无言] 关於小包卫生纸
icon.png[开箱] E5-2683V3 RX480Strix 快睿C1 简单测试
icon.png[心得] 苍の海贼龙 地狱 执行者16PT
icon.png[售车] 1999年Virage iO 1.8EXi
icon.png[心得] 挑战33 LV10 狮子座pt solo
icon.png[闲聊] 手把手教你不被桶之新手主购教学
icon.png[分享] Civic Type R 量产版官方照无预警流出
icon.png[售车] Golf 4 2.0 银色 自排
icon.png[出售] Graco提篮汽座(有底座)2000元诚可议
icon.png[问题] 请问补牙材质掉了还能再补吗?(台中半年内
icon.png[问题] 44th 单曲 生写竟然都给重复的啊啊!
icon.png[心得] 华南红卡/icash 核卡
icon.png[问题] 拔牙矫正这样正常吗
icon.png[赠送] 老莫高业 初业 102年版
icon.png[情报] 三大行动支付 本季掀战火
icon.png[宝宝] 博客来Amos水蜡笔5/1特价五折
icon.pngRe: [心得] 新鲜人一些面试分享
icon.png[心得] 苍の海贼龙 地狱 麒麟25PT
icon.pngRe: [闲聊] (君の名は。雷慎入) 君名二创漫画翻译
icon.pngRe: [闲聊] OGN中场影片:失踪人口局 (英文字幕)
icon.png[问题] 台湾大哥大4G讯号差
icon.png[出售] [全国]全新千寻侘草LED灯, 水草

请输入看板名称,例如:iOS站内搜寻

TOP