guanguanjujiu 发表于 2021-2-3 19:28

求助C语言的问题

本帖最后由 guanguanjujiu 于 2021-2-3 19:41 编辑

#include <stdio.h>
#include <string.h>

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

int main()
{
        const int MAX = 1000;
       
        int time;
       
        char string;
        char pd[] = "END";
       
       
        do{
                scanf("%s", string);
               
                if( verifyCompute(string,time))
                {
                        printf("%2d:%2d:%2d", time, time, time);
                }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 = '*')
                {
                        break;
                }else if( string != '$')
                {
                        a ^= string;
                }
        }
        a %= 65536;
        //取出异或值
        b = strtok( string, "*");
        b = strtok( NULL, "*");
       
        if( b == a)
        {
                char time1;
               
                time1 = strtok(string,",");
                time1 = strtok(NULL,",");
               
                *time = int(time1)/10000 + 8;
               
                if( *time > 24)
                {
                        *time -= 24;
                }
               
                *time = (time1 - time*10000) / 100;
               
                *time = (time1 - time*10000 - time*100);
               
        }else{
                ret = 0;
        }
       
       
       
        return ret;
}

行:信息:
19    passing argument 2 of 'verifyCompute' from incompatible pointer type
4    expected 'int **' but argument is of type 'int *'
52    assignment makes integer from pointer without a cast
53    assignment makes integer from pointer without a cast
59    assignment makes integer from pointer without a cast
60    assignment makes integer from pointer without a cast
62    expected expression before 'int'
69    invalid operands to binary * (have 'int *' and 'int')
71    invalid operands to binary * (have 'int *' and 'int')
71    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 'time2' undeclared (first use in this function)
例如第71行错误是time2没声明
75    expected ';' before '}' token
75行错误是少;

guanguanjujiu 发表于 2021-2-3 19:36

阿龙alone 发表于 2021-2-3 19:31
这种错误一个一个百度都能搜到的,
71    'time2' undeclared (first use in this function)
例如 ...

。。。。。
修改代码时,忘了改了,大佬,前面的错误和警告是怎么回事啊?

wangyujie96 发表于 2021-2-3 20:55

本帖最后由 wangyujie96 于 2021-2-3 21:04 编辑

int* time[] ,这是整数指针数组。整数数组就是int time[]。上面的错误大部分是因为这个
给数组的元素赋值只需要 time=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, "*");
我完成不懂这里的重复赋值运算是什么意思{:301_983:}

第二个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 编辑

#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))){
                        return 0;
                }
      }
      return 1;
}

int verifyCompute(char string[], int time[]){
      int ret = 1;
      char a = string; //代表异或值;不能赋错值,比如0什么的。需要自己修改以应对其他情况
      char* b; //代表输入的异或值;
      //计算异或值
      for(int i=2; i<strlen(string); i++){
                if(string == '*'){
                        break;
                }else if(string != '$'){
                        a ^= string;
                }
      }
      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 = tmp/10000;//后面要用来计算,现在不能加8
                if( time >= 24){
                        time -= 24;
                }
                time = (tmp - time*10000) / 100;
                time = (tmp - time*10000 - time*100);
      }else{
                ret = 0;
      }
      return ret;
}

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

好久没用C了,睡前随便改了一下,勉强能把你给的例子弄出来了,但考虑到题目要求,还需要修改很多地方,对输入格式和字符串的合法性做出判断,不能偷懒直接赋值。比如我代码里那个char a = string。简单的说,就是要尽可能考虑到各种情况的输入并作出判断,处理各种可能出现异常。加油吧。

wangyujie96 发表于 2021-2-3 22:34

代码都写完了,网上一搜居然有原题:https://blog.csdn.net/kuangzuxiaon/article/details/77127010

总不能白写,还是贴上来:
#include <stdio.h>
#include <string.h>
#pragma warning(disable:4996)

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

int main()
{

        int time;

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


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

                if (verifyCompute(string, time))
                {
                        printf("%02d:%02d:%02d", time, time, time);//不足两位填充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 == '*')
                {
                        break;
                }
                else if (string != '$')
                {
                        result ^= string;
                }
        }
        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, &time, &time);//按两位两位整数的格式解析

                time += 8;

                if (time > 24)
                {
                        time -= 24;
                }

        }
        else {
                ret = 0;
        }



        return ret;
}
页: [1]
查看完整版本: 求助C语言的问题