setenv, unsetenv, putenv

setenv, unsetenv, putenv

本文关键字:putenv unsetenv setenv      更新时间:2023-10-16

我正在为系统编程类编写一个自定义shell。我们被指示实现内置的setenv()unsetenv()命令,并提示检查putenv() 的手册页。

我的问题是,setenv(char*, char*, int)putenv(char*)似乎不工作在所有。我执行输入命令的代码如下:

//... skipping past stuff for IO redirection
pid = fork();
if(pid == 0){
    //child
    if(!strcmp(_simpleCommands[0]->_arguments[0],"printenv")){
        //check if command is "printenv"
        extern char **environ;
        int i;
        for(i = 0; environ[i] != NULL; i++){
            printf("%sn",environ[i]);
        }
        exit(0);
    }
    if(!strcmp(_simpleCommands[0]->_arguments[0],"setenv")){
        //if command is "setenv" get parameters char* A, char* B
        char * p = _simpleCommands[0]->_arguments[1];
        char * s = _simpleCommands[0]->_arguments[2];
        //putenv(char* s) needs to be formatted A=B; A is variable B is value
        char param[strlen(p) + strlen(s) + 1];
        strcat(param,p);
        strcat(param,"=");
        strcat(param,s);
        putenv(param);
        //setenv(p,s,1);
        exit(0);
    }
    if(!strcmp(_simpleCommands[0]->_arguments[0],"unsetenv")){
        //remove environment variable
        unsetenv(_simpleCommands[0]->_arguments[0]);
        exit(0);
    }
    //execute command
    execvp(_simpleCommands[0]->_arguments[0],_simpleCommands->_arguments);
    perror("-myshell");
    _exit(1);
}
//omitting restore IO defaults...

如果我运行printenv,它工作正常,但如果我尝试使用putenv()setenv()设置一个新的变量,我的printenv()命令返回完全相同的东西,所以它似乎不工作。

作为旁注,问题可能与函数或我如何调用它们无关,因为我的shell正在执行命令,好像它必须格式化通配符(*?),我不确定应该发生。

在检查命令行之前,您似乎无条件地调用了fork。但是一些shell内置命令需要在进程中运行,这样它们的效果才会持续。所有操纵环境的内置程序都属于这一类。

作为题外话,如果我正在编写shell,我不会尝试使用C库的环境操作函数。我会使用三个参数的main,将envp复制到我完全控制的数据结构中,然后将其反馈回execve。这部分是因为我是一个控制狂,部分是因为用setenv和/或putenv做任何复杂的事情而不发生内存泄漏几乎是不可能的。

你为什么认为它不起作用?我在下面写了一个简单的测试用例…

确保在相同的进程中调用seten和prientevn。

#include <stdlib.h>
#include <assert.h>
int main()
{

 char * s= "stack=overflow";
 int ret =  putenv(s);
 assert(ret == 0);
 //printout all the env
 extern char **environ;
        int i;
        for(i = 0; environ[i] != NULL; i++){
            printf("%sn",environ[i]);
        }

 return 0; 
}
pierr@ubuntu:~/workspace/so/c/env$ ./test | grep stack
stack=overflow