1054 求平均值 (sscanf(), sprintf()处理字符串)

本文介绍了一个复杂的编程问题,即在给定一系列实数输入时,如何过滤并计算合法数值的平均值。合法数值定义为在[-1000,1000]区间内且最多精确到小数点后两位的实数。文章分享了一种解决方案,通过使用sscanf(), sprintf()等函数来判断输入是否合法,并计算平均值。

本题的基本要求非常简单:给定 N 个实数,计算它们的平均值。但复杂的是有些输入数据可能是非法的。一个“合法”的输入是 [−1000,1000] 区间内的实数,并且最多精确到小数点后 2 位。当你计算平均值的时候,不能把那些非法的数据算在内。

输入格式:

输入第一行给出正整数 N(≤100)。随后一行给出 N 个实数,数字间以一个空格分隔。

输出格式:

对每个非法输入,在一行中输出 ERROR: X is not a legal number,其中 X 是输入。最后在一行中输出结果:The average of K numbers is Y,其中 K 是合法输入的个数,Y 是它们的平均值,精确到小数点后 2 位。如果平均值无法计算,则用 Undefined 替换 Y。如果 K 为 1,则输出 The average of 1 number is Y

输入样例 1:

7
5 -3.2 aaa 9999 2.3.4 7.123 2.35

输出样例 1:

ERROR: aaa is not a legal number
ERROR: 9999 is not a legal number
ERROR: 2.3.4 is not a legal number
ERROR: 7.123 is not a legal number
The average of 3 numbers is 1.38

输入样例 2:

2
aaa -9999

输出样例 2:

ERROR: aaa is not a legal number
ERROR: -9999 is not a legal number
The average of 0 numbers is Undefined

 

       说真的,这个题让我非常困扰。我的思路是不合法的数据有太多种类型了,那我就去判断合法的数据。题干描述太过于简洁,到底什么才是合法的数据,并没有给出详细说明。我个人的代码总是有一个测试点过不了,后来去看别人的博客,我发现就算是提交到PTA正确的代码,对于同一个其他的样例,也会有不同的结果。索性学习了小姐姐的代码,十分佩服。

#include<iostream>
#include<cstdio>
#include<string.h>

using namespace std;

int main(){
	int n, cnt = 0;
	char a[50], b[50];
	double temp, sum = 0.0;
	cin >> n;
	for(int i = 0; i < n; i++){
		scanf("%s", a);
		sscanf(a, "%lf", &temp);
		sprintf(b, "%.2lf", temp);
		int flag = 0;
		for(int j = 0; j < strlen(a); j++){
			if(a[j] != b[j]) flag = 1;
		}
		if(flag || temp < -1000 || temp > 1000){
			printf("ERROR: %s is not a legal number\n", a);
                        continue;
		}else{
			sum += temp;
			cnt++;
		}
	}
	if(cnt == 1){
            printf("The average of 1 number is %.2lf", sum);
        }else if(cnt > 1){
            printf("The average of %d numbers is %.2lf", cnt, sum / cnt);
        }else {
            printf("The average of 0 numbers is Undefined");
        }
}

sscanf():从一个字符串中读入指定格式的数据。

sprintf():字符串格式化命令,主要功能是把格式化的数据写入某个字符串中。

你写的这一段: ```c "二,设备状态监控 \n 样机1: 内存剩余: xxxKB, cpu平均值xx%%, cpu峰值xx%%%\n" ``` 意图是在邮件中加入设备监控信息,并且想用 `snprintf` 之类的函数来填充具体数值。我们来 **逐个分析是否可行、`%` 是否会出问题**。 --- ## ✅ 回答:可以这样写,但必须正确处理 `%` 转义! ### 🔴 问题关键:在 C 字符串中使用 `snprintf` 或 `printf` 时,`%` 是格式化占位符 如果你写: ```c char buf[1024]; snprintf(buf, sizeof(buf), "样机1: 内存剩余: %dKB, cpu平均值%d%, cpu峰值%d%"); ``` 👉 这会导致 **未定义行为(undefined behavior)**! 因为: - `%d` 是合法的 → 替换为整数 - 但单独一个 `%` 后面没有合法转换符(如 `s`, `d`, `f` 等)→ 编译器认为格式字符串错误 - 比如 `%` 后跟 `\n` → 不认识 → 可能崩溃或打印乱码 --- ## ✅ 正确做法:用 `%%` 表示字面上的百分号 `%` | 你想输出的内容 | 错误写法 | 正确写法 | |----------------|----------|----------| | 输出一个 `%` | `"%"` | `"%%"` | 所以你应该这样写: ```c char monitor_info[512]; int mem_free = 10240; // 示例:剩余内存 KB int cpu_avg = 65; // CPU 平均使用率 int cpu_peak = 90; // CPU 峰值 snprintf(monitor_info, sizeof(monitor_info), "二,设备状态监控 \n" "样机1: 内存剩余: %d KB, cpu平均值 %d%%, cpu峰值 %d%%\n", mem_free, cpu_avg, cpu_peak); ``` ✅ 输出结果: ``` 二,设备状态监控 样机1: 内存剩余: 10240 KB, cpu平均值 65%, cpu峰值 90% ``` ✔️ 完美无误! --- ## 📌 在你的邮件正文中整合这段内容的方法 假设你现在生成完整邮件正文如下: ```c // 先生成测试报告正文(来自 XML 解析) char* test_report = parse_xml_generate_report("test_results.xml"); if (!test_report) { fprintf(stderr, "XML解析失败\n"); return -1; } // 生成设备监控信息 char monitor_info[512]; int mem_free = get_memory_usage(); // 你需要实现这个函数 int cpu_avg = get_cpu_average(); int cpu_peak = get_cpu_peak(); snprintf(monitor_info, sizeof(monitor_info), "二,设备状态监控 \n" "样机1: 内存剩余: %d KB, cpu平均值 %d%%, cpu峰值 %d%%\n", mem_free, cpu_avg, cpu_peak); // 组合最终邮件正文 char email_body[8192]; snprintf(email_body, sizeof(email_body), "%s\n%s", test_report, monitor_info); // 把两部分拼起来 // 使用 email_body 发送邮件... ``` --- ## ❓ 如何获取这些系统数据?(Linux 下示例) ### ✅ 获取内存使用情况(/proc/meminfo) ```c int get_memory_usage() { FILE *fp = fopen("/proc/meminfo", "r"); int memfree = 0; if (fp) { char line[256]; while (fgets(line, sizeof(line), fp)) { if (sscanf(line, "MemAvailable: %d kB", &memfree) == 1) { break; } } fclose(fp); } return memfree; // 单位 KB } ``` --- ### ✅ 获取 CPU 使用率(简化版:读取 /proc/stat 一次,不能算实时) > ⚠️ 注意:CPU 使用率需要两次采样才能计算。如果只是想上报“最近的平均负载”,可以用 `getloadavg()`。 #### 方法一:获取 1 分钟平均负载(简单粗略) ```c double loadavg[3]; if (getloadavg(loadavg, 3) != -1) { printf("1分钟平均负载: %.2f\n", loadavg[0]); } ``` #### 方法二:精确采集 CPU 利用率(需前后两次读 `/proc/stat`) 由于较复杂,建议封装成函数。这里给一个简化版本(假设有前次数据): ```c typedef struct { unsigned long user, nice, system, idle, iowait, irq, softirq; } cpu_stats_t; int get_cpu_percent(cpu_stats_t *prev, cpu_stats_t *now) { unsigned long long prev_total = prev->user + prev->nice + prev->system + prev->idle + prev->iowait + prev->irq + prev->softirq; unsigned long long now_total = now->user + now->nice + now->system + now->idle + now->iowait + now->irq + now->softirq; unsigned long long prev_idle = prev->idle + prev->iowait; unsigned long long now_idle = now->idle + now->iowait; unsigned long long diff_total = now_total - prev_total; unsigned long long diff_idle = now_idle - prev_idle; if (diff_total == 0) return 0; return (int)((double)(diff_total - diff_idle) / diff_total * 100); } ``` 👉 需要先采集一次作为基准,过几秒再采一次计算差值。 --- ## ✅ 最终推荐方案(实用主义) 如果你不需要实时性,可以这样写死近似值或从 shell 获取: ```c int get_cpu_average() { double load[3]; return getloadavg(load, 1) > 0 ? (int)(load[0] * 10) : 50; // 粗略估计 } int get_cpu_peak() { // 实际中可记录历史最大值,或返回 top 命令抓取的结果 return 95; // 示例值 } ``` 或者直接调用命令: ```c int run_command_int(const char *cmd) { FILE *fp = popen(cmd, "r"); int result = -1; if (fp) { fscanf(fp, "%d", &result); pclose(fp); } return result; } // 使用示例 int cpu_avg = run_command_int("top -bn1 | head -10 | grep 'Cpu(s)' | awk '{print $2}' | cut -d'%' -f1 | xargs printf '%.0f'"); int mem_free = run_command_int("grep 'MemAvailable' /proc/meminfo | awk '{print $2}'"); ``` ⚠️ 注意:`popen` 和 `top` 可能耗费资源,适合定时任务而非高频调用。 --- ## ✅ 总结:你的写法可以吗? | 项目 | 是否正确 | 说明 | |------|----------|------| | `"cpu平均值xx%"` | ❌ 危险 | 单独 `%` 会被当作格式符 | | `"cpu平均值%d%%"` | ✅ 正确 | `%%` 输出一个 `%` 字符 | | 加入到邮件正文 | ✅ 可行 | 建议用 `snprintf` 动态生成后拼接 | --- ## ✅ 推荐最终代码结构 ```c char* generate_monitor_section() { static char buf[512]; int mem_kb = get_memory_usage(); int cpu_avg = 65; // 或动态获取 int cpu_peak = 90; snprintf(buf, sizeof(buf), "二,设备状态监控 \n" "样机1: 内存剩余: %d KB, cpu平均值 %d%%, cpu峰值 %d%%\n", mem_kb, cpu_avg, cpu_peak); return buf; } ``` 然后合并进邮件体: ```c sprintf(final_body, "%s\n%s", test_report, generate_monitor_section()); ``` --- ##
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值