execv 系统调用未按预期运行

如何解决execv 系统调用未按预期运行

我使用 Linux,在编译任何 c 或 cpp 文件时,我分别在终端中使用 gcc 或 g++。 常用语法:g++ program.cpp

但现在我希望使用标志来编译文件。 例如:g++ -Wall -Wextra -std=c++11 program.cpp 我将使用更多 10 个标志来编译我的程序。但我不想在终端编译时记住和输入。

现在我希望创建一个涉及系统调用 (exec) 的 c 程序,以使用以下语法完成我的工作: ./compile program.cpp

但是在我下面的代码中使用 exec 时出现了一些问题

#include<fcntl.h>
#include<unistd.h>
#include<stdlib.h>
int main(int args,char* argv[]){
    char* arguments[10]={"-std=c++11","-Wall","-Wextra","-pedantic","-Wshadow","-fsanitize=address","-fsanitize=undefined","-fstack-protector"}; //consists of flags with which i will compile the program passed as argument
    printf("%s\t %s",argv[0],argv[1]);
    if(args==2){
        arguments[8]=argv[1];
        arguments[9]=(char*)NULL;
    }else{
        printf("only one argument allowed!");// just to check if i pass arguments correctly
    }
    printf("%s\t %s",arguments[8],arguments[9]);// just to check if my arguments array is correct
    if(execv("/bin/g++",arguments)==-1){ // to my suprise this line runs before above printing lines. What is the reason/solution?
        perror("execv Failed!");
        exit(1);
    }
    
    return 0;
}

以上代码编译成功,没有错误。 但我认为 execv 甚至在我在参数数组中插入传递的参数之前运行。 因此,程序运行时出现错误 execv Failed: no such file or directory 其次是 printfs。 请告诉我哪里出错了。

解决方法

于是我终于解决了上面代码中的歧义。我对代码进行了两次彻底的更改。

  1. 不是在声明字符串数组时直接分配参数字符串,
char* arguments[10]={"-std=c++11","-Wall","-Wextra","-pedantic","-Wshadow","-fsanitize=address","-fsanitize=undefined","-fstack-protector"};

我选择一个一个地分配字符串。

char* arguments[10];
        arguments[0]="g++";
        arguments[1]="-Wall";
        arguments[2]="-Wextra";
and so on

这修复了我代码中的分段错误。

  1. 这次我使用 execvp() 而不是 execv() 系统调用,因为我不需要显式声明命令 usr/bin/g++ 的完整路径等等。在 execvp 中,只有命令名就足够了。 所以我的新代码如下所示:
#include<stdio.h>
//#include<fcntl.h>
#include<unistd.h>
#include<stdlib.h>
int main(int args,char* argv[]){
        char* arguments[10];
        //printf("%s\t %s\n",argv[0],argv[1]); 
        arguments[0]="g++";
        arguments[1]="-Wall";
        arguments[2]="-Wextra";
        arguments[3]="-pedantic";
        arguments[4]="-Wshadow";
        arguments[5]="-fsanitize=address";
        arguments[6]="-fsanitize=undefined";
        arguments[7]="-fstack-protector";// to add more flags make changes in this array.
        if(args==2){
                arguments[8]=argv[1];
                arguments[9]=(char*)NULL;
                if(execvp(arguments[0],arguments)==-1){
                        perror("execv failed!");
                        exit(1);
                }

        }else{
                printf("->Only one argument(c/cpp file) allowed.\n->Requtired syntax: ./compile program.cpp\n");
        }
        return 0;
}
  1. 我遇到的另一个问题是,所有在 execv() 系统调用之前的 printfs 只会在 execv() 被执行后才会被打印出来。正如@Myousefi Sir 评论的那样,这是因为缓冲区未满。按照建议,在 printfs 中添加“\n”解决了问题。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其他元素将获得点击?
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。)
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbcDriver发生异常。为什么?
这是用Java进行XML解析的最佳库。
Java的PriorityQueue的内置迭代器不会以任何特定顺序遍历数据结构。为什么?
如何在Java中聆听按键时移动图像。
Java“Program to an interface”。这是什么意思?
Java在半透明框架/面板/组件上重新绘画。
Java“ Class.forName()”和“ Class.forName()。newInstance()”之间有什么区别?
在此环境中不提供编译器。也许是在JRE而不是JDK上运行?
Java用相同的方法在一个类中实现两个接口。哪种接口方法被覆盖?
Java 什么是Runtime.getRuntime()。totalMemory()和freeMemory()?
java.library.path中的java.lang.UnsatisfiedLinkError否*****。dll
JavaFX“位置是必需的。” 即使在同一包装中
Java 导入两个具有相同名称的类。怎么处理?
Java 是否应该在HttpServletResponse.getOutputStream()/。getWriter()上调用.close()?
Java RegEx元字符(。)和普通点?