吾爱破解 - 52pojie.cn

 找回密码
 注册[Register]

QQ登录

只需一步,快速开始

查看: 1410|回复: 7
收起左侧

[求助] 求助C语言的问题

[复制链接]
guanguanjujiu 发表于 2021-2-3 19:28
本帖最后由 guanguanjujiu 于 2021-2-3 19:41 编辑

[C] 纯文本查看 复制代码
#include <stdio.h>
#include <string.h>

int verifyCompute(char string[], int* time[]);
 
int main()
{
	const int MAX = 1000;
	
	int time[3];
	
	char string[MAX];
	char pd[] = "END";
	
	
	do{
		scanf("%s", string);
		
		if( verifyCompute(string,time))
		{
			printf("%2d:%2d:%2d", time[0], time[1], time[2]); 
		}else{
			printf("NULL");
		}
		
	}while( strcmp(string, pd) != 0);
	
	return 0;
}

int verifyCompute(char string[], int* time[])
{
	int ret = 1;
	
	int a = 0; //代表异或值;
	
	char b; //代表输入的异或值; 
	
	//计算异或值 
	for(int i=0; i<strlen(string); i++)
	{
		if(string[i] = '*')
		{
			break;
		}else if( string[i] != '$')
		{
			a ^= string[i]; 
		}
	}
	a %= 65536; 
	//取出异或值 
	b = strtok( string, "*");
	b = strtok( NULL, "*");
	
	if( b == a)
	{
		char time1;
		
		time1 = strtok(string,",");
		time1 = strtok(NULL,",");
		
		*time[0] = int(time1)/10000 + 8;
		
		if( *time[0] > 24)
		{
			*time[0] -= 24;
		}
		
		*time[1] = (time1 - time[0]*10000) / 100;
		
		*time[2] = (time1 - time[0]*10000 - time[1]*100);
		
	}else{
		ret = 0;
	}
	
	
	
	return ret;
}


行:  信息:
19   [Warning] passing argument 2 of 'verifyCompute' from incompatible pointer type
4    [Note] expected 'int **' but argument is of type 'int *'
52   [Warning] assignment makes integer from pointer without a cast
53   [Warning] assignment makes integer from pointer without a cast
59   [Warning] assignment makes integer from pointer without a cast
60   [Warning] assignment makes integer from pointer without a cast

62   [Error] expected expression before 'int'
69   [Error] invalid operands to binary * (have 'int *' and 'int')
71   [Error] invalid operands to binary * (have 'int *' and 'int')
71   [Error] invalid operands to binary * (have 'int *' and 'int')


大佬们,求助。。。。我快疯了。。。写了一个程序,BUG改不完啊!

下面是原题目:
题目内容:

NMEA-0183协议是为了在不同的GPS(全球定位系统)导航设备中建立统一的BTCM(海事无线电技术委员会)标准,由美国国家海洋电子协会(NMEA-The National Marine Electronics Associa-tion)制定的一套通讯协议。GPS接收机根据NMEA-0183协议的标准规范,将位置、速度等信息通过串口传送到PC机、PDA等设备。



NMEA-0183协议是GPS接收机应当遵守的标准协议,也是目前GPS接收机上使用最广泛的协议,大多数常见的GPS接收机、GPS数据处理软件、导航软件都遵守或者至少兼容这个协议。



NMEA-0183协议定义的语句非常多,但是常用的或者说兼容性最广的语句只有$GPGGA、$GPGSA、$GPGSV、$GPRMC、$GPVTG、$GPGLL等。



其中$GPRMC语句的格式如下:

    $GPRMC,024813.640,A,3158.4608,N,11848.3737,E,10.05,324.27,150706,,,A*50



这里整条语句是一个文本行,行中以逗号“,”隔开各个字段,每个字段的大小(长度)不一,这里的示例只是一种可能,并不能认为字段的大小就如上述例句一样。

    字段0:$GPRMC,语句ID,表明该语句为Recommended Minimum Specific GPS/TRANSIT Data(RMC)推荐最小定位信息

    字段1:UTC时间,hhmmss.sss格式

    字段2:状态,A=定位,V=未定位

    字段3:纬度ddmm.mmmm,度分格式(前导位数不足则补0)

    字段4:纬度N(北纬)或S(南纬)

    字段5:经度dddmm.mmmm,度分格式(前导位数不足则补0)

    字段6:经度E(东经)或W(西经)

    字段7:速度,节,Knots

    字段8:方位角,度

    字段9:UTC日期,DDMMYY格式

    字段10:磁偏角,(000 - 180)度(前导位数不足则补0)

    字段11:磁偏角方向,E=东W=西

    字段16:校验值

这里,“*”为校验和识别符,其后面的两位数为校验和,代表了“$”和“*”之间所有字符(不包括这两个字符)的异或值的十六进制值。上面这条例句的校验和是十六进制的50,也就是十进制的80。



提示:^运算符的作用是异或。将$和*之间所有的字符做^运算(第一个字符和第二个字符异或,结果再和第三个字符异或,依此类推)之后的值对65536取余后的结果,应该和*后面的两个十六进制数字的值相等,否则的话说明这条语句在传输中发生了错误。注意这个十六进制值中是会出现A-F的大写字母的。



现在,你的程序要读入一系列GPS输出,其中包含$GPRMC,也包含其他语句。在数据的最后,有一行单独的

    END

表示数据的结束。



你的程序要从中找出$GPRMC语句,计算校验和,找出其中校验正确,并且字段2表示已定位的语句,从中计算出时间,换算成北京时间。一次数据中会包含多条$GPRMC语句,以最后一条语句得到的北京时间作为结果输出。

你的程序一定会读到一条有效的$GPRMC语句。



输入格式:

多条GPS语句,每条均以回车换行结束。最后一行是END三个大写字母。



输出格式:

6位数时间,表达为:

    hh:mm:ss

其中,hh是两位数的小时,不足两位时前面补0;mm是两位数的分钟,不足两位时前面补0;ss是两位数的秒,不足两位时前面补0。



输入样例:

$GPRMC,024813.640,A,3158.4608,N,11848.3737,E,10.05,324.27,150706,,,A*50

END



输出样例:

10:48:13

发帖前要善用论坛搜索功能,那里可能会有你要找的答案或者已经有人发布过相同内容了,请勿重复发帖。

阿龙alone 发表于 2021-2-3 19:31
这种错误一个一个百度都能搜到的,
71   [Error] 'time2' undeclared (first use in this function)
例如第71行错误是time2没声明
75   [Error] expected ';' before '}' token
75行错误是少;
 楼主| guanguanjujiu 发表于 2021-2-3 19:36
阿龙alone 发表于 2021-2-3 19:31
这种错误一个一个百度都能搜到的,
71   [Error] 'time2' undeclared (first use in this function)
例如 ...

。。。。。
修改代码时,忘了改了,大佬,前面的错误和警告是怎么回事啊?
wangyujie96 发表于 2021-2-3 20:55
本帖最后由 wangyujie96 于 2021-2-3 21:04 编辑

int* time[] ,这是整数指针数组。整数数组就是int time[]。上面的错误大部分是因为这个
给数组的元素赋值只需要 time[0]=123;即可


能看出52行是试图取出十六进制的校验值,不过思路有问题。输入的校验值是十六进制文本,例如"50",这是文本型,在计算机内实际上是用两个字节存储的,35,30 ,即便你写 b = *strtok(string,","); 对字符指针解引用,b的值也是 35,长度是一个字节。这里涉及十六进制文本转整数,自行查阅:https://blog.csdn.net/nanfeibuyi/article/details/107110392 。或者将计算的校验值格式化为十六进制文本,用sprintf()函数来格式化字符串,%X 表示大写十六进制,然后strcmp字符串对比。

answdl 发表于 2021-2-3 21:03
我是个初学者,不知道能不能帮到。
在主函数上的变长数组,在VS2019上是错误的
在自定义函数上的for函数里的if(赋值运算)条件是始终成立的(没记错的话),即else if 不执行

//取出异或值   
    a %= 65536;
    b = strtok(string, "*");
    b = strtok(NULL, "*");
我完成不懂这里的重复赋值运算是什么意思

第二个if()函数
有一个强制类型转换在VS2019上是错误的。

第二个if()函数里的if()函数
对指针变量(地址)进行乘法运算???(我没有看懂)
wangyujie96 发表于 2021-2-3 21:15
抱歉,我刚开始不知道strbok函数的具体行为。strtok(NULL, "*");这种写法没错,是取出第二个分解字符串,但即使不考虑思路,你这语法也是错误的,应该解除引用, b = *strtok(... 。strtok 这个函数会改变传入的字符串!
最重要的,你这取出异或值的方法不对。

强制类型转换的写法是  a = (int) b;
mrzs10000 发表于 2021-2-3 22:19
本帖最后由 mrzs10000 于 2021-2-3 22:20 编辑

[C] 纯文本查看 复制代码
#include<stdio.h>
#include<string.h>
#include<ctype.h>
#include<stdlib.h>

int isInt(char* str){
        int len;
        len = strlen(str);
        int i=0;
        for(i;i<len;i++){
                if(!(isdigit(str[i]))){
                        return 0;
                }
        }
        return 1;
}

int verifyCompute(char string[], int time[]){
        int ret = 1;
        char a = string[1]; //代表异或值;不能赋错值,比如0什么的。需要自己修改以应对其他情况
        char* b; //代表输入的异或值;
        //计算异或值 
        for(int i=2; i<strlen(string); i++){
                if(string[i] == '*'){
                        break;
                }else if(string[i] != '$'){
                        a ^= string[i]; 
                }
        }
        a %= 65536;
        //取出异或值 
        b = strtok(string, "*");
        b = strtok(NULL, "*");
        int t = 0;
        sscanf(b, "%x", &t);//以十六进制读入
        if( t == a){
                char* time1;
                int tmp=0;
                time1 = strtok(string,",");
                time1 = strtok(NULL,",");
                time1=strtok(time1,".");
                if(isInt(time1)){//判断了一下是否全为数字
                        tmp=atoi(time1);
                }
                time[0] = tmp/10000;//后面要用来计算,现在不能加8
                if( time[0] >= 24){
                        time[0] -= 24;
                }
                time[1] = (tmp - time[0]*10000) / 100;
                time[2] = (tmp - time[0]*10000 - time[1]*100);
        }else{
                ret = 0;
        }
        return ret;
}

int main(){
        const int MAX = 1000;
        int time[3];
        char string[MAX];
        char pd[] = "END";
        while(true){
                scanf("%s", string);
                if(strcmp(string, pd)== 0){
                        break;
                }
                if(verifyCompute(string,time)){
                        printf("%2d:%2d:%2d", time[0]+8, time[1], time[2]); 
                }else{
                        printf("NULL");
                }
        }
        return 0;
}


好久没用C了,睡前随便改了一下,勉强能把你给的例子弄出来了,但考虑到题目要求,还需要修改很多地方,对输入格式和字符串的合法性做出判断,不能偷懒直接赋值。比如我代码里那个char a = string[1]。简单的说,就是要尽可能考虑到各种情况的输入并作出判断,处理各种可能出现异常。加油吧。
wangyujie96 发表于 2021-2-3 22:34
代码都写完了,网上一搜居然有原题:https://blog.csdn.net/kuangzuxiaon/article/details/77127010

总不能白写,还是贴上来:
[C] 纯文本查看 复制代码
#include <stdio.h>
#include <string.h>
#pragma warning(disable:4996)

int verifyCompute(char string[], int time[]);

int main()
{

	int time[3];

	char string[1000];
	char pd[] = "END";


	do {
		scanf("%s", string);

		if (verifyCompute(string, time))
		{
			printf("%02d:%02d:%02d", time[0], time[1], time[2]);  //不足两位填充0
		}
		else {
			printf("NULL");
		}

	} while (strcmp(string, pd) != 0);

	return 0;
}

int verifyCompute(char string[], int time[])
{
	int ret = 1;

	char result = 0; //代表异或值;

	int input; //代表输入的异或值; 

	//计算异或值 
	for (int i = 0; i < strlen(string); i++)
	{
		if (string[i] == '*')
		{
			break;
		}
		else if (string[i] != '$')
		{
			result ^= string[i];
		}
	}
	result %= 0xff;

	//取出输入的异或值 
	char* strSum = strstr(string, "*") + 1; //字符串搜索函数, +1 以忽略第一个字符“ * ”
	sscanf(strSum, "%2X", &input); // 将十六进制字符串转为整数

	if (input == result)
	{
		//校验通过
		char* strTime;

		strTime = strstr(string, ",") + 1;  // +1 忽略第一个字符“,”

		sscanf(strTime, "%2d%2d%2d", &time[0], &time[1], &time[2]);  //按两位两位整数的格式解析

		time[0] += 8;

		if (time[0] > 24)
		{
			time[0] -= 24;
		}

	}
	else {
		ret = 0;
	}



	return ret;
}
您需要登录后才可以回帖 登录 | 注册[Register]

本版积分规则

返回列表

RSS订阅|小黑屋|处罚记录|联系我们|吾爱破解 - LCG - LSG ( 京ICP备16042023号 | 京公网安备 11010502030087号 )

GMT+8, 2024-11-26 07:36

Powered by Discuz!

Copyright © 2001-2020, Tencent Cloud.

快速回复 返回顶部 返回列表