系统函数学习

lseek函数用于改变文件的当前文件偏移量(cfo),它可以设置为文件开始、当前位置或文件末尾的特定字节位置。对于普通文件,cfo是非负整数,而对于特殊设备可能为负数。如果偏移量大于文件长度,写操作会创建文件空洞。argc和argv在命令行参数传递中使用,argc记录参数个数,argv是一个字符串指针数组,包含了命令行的所有参数。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1. lseek 函数用法

        所有打开的文件都有一个当前文件偏移量(current file offset),以下简称cfo。cfo通常是一个非负整数,用于表明文件开始处到文件当前位置的字节数。读写操作通常开始于cfo,并且使cfo增大,增量为读写的字节数。文件被打开时,cfo会被初始化为0,除非使用了O_APPEND。

        使用lseek函数可以改变文件的cfo。

#include <unistd.h>
#include <sys/type.h>
off_t lseek(int fields, off_t offset, int whence)

返回值:新的偏移量(成功),-1(失败)

参数offset的含义取决于参数whence:

  1. 如果whence是SEEK_SET,文件偏移量将被设置为offset。
  2. 如果whence是SEEK_CUR,文件偏移量将被设置为cfo加上offset,offset可以为正也可以为负。
  3. 如果whence是SEEK_END,文件偏移量将设置为文件长度加上offset,offset可以设置为正也可以为负。

lseek的以下用法返回当前的偏移量;

off_t currpos;
currpos = lseek(fd, 0, SEEK_CUR);

这个技巧也可用于判断我们是否可以改变某个文件的偏移量。如果参数fd(文件描述符)指定的pipe(管道)、FIFO 或者socket,lseek返回 -1 并且置errno 为 ESPIPE。

对于普通文件(regular file),cfo是一个非负整数。但是对于特殊设备,cfo有可能是负数。因此,我们不能简单地测试lseek的返回值是否小于0来判断lseek成功与否,而应该测试lseek的返回值是否等于-1来判断lseek成功与否。

        lseek仅将cfo保存于内核中,不会导致任何IO操作。这个cfo将被用于之后的读写操作。 

        如果offset比文件的当前长度更大,下一个写操作就会把文件“撑大(extend)”。这就是所谓的在文件里创造“空洞(hole)”。没有被实际写入文件的所以字节由重复0表示。空洞是否占用硬盘空间是由文件系统觉得的。

  #include <stdio.h>
        /* Unix header */
        #include <fcntl.h>
        #include <unistd.h>
        #include <sys/stat.h>

        char buf1[] = "abcdefghij";
        char buf2[] = "ABCDEFGHIJ";

        int main(void)
        {
            int fd, size;

            if ((fd = creat("file.hole", S_IRUSR|S_IWUSR)) < 0)
            {
                printf("creat error\n");
                return -1;
            }

            size = sizeof(buf1) - 1;
            if (write(fd, buf1, size) != size)
            {
                printf("buf1 write error\n");
                return -1;
            }
            /* offset now = 10 */

            if (lseek(fd, 16384, SEEK_SET) == -1)
            {
                printf("lseek error\n");
                return -1;
            }
            /* offset now = 16384 */

            size = sizeof(buf2) - 1;
            if (write(fd, buf2, size) != size)
            {
                printf("buf2 write error\n");
                return -1;
            }
            /* offset now = 16394 */

            return 0;
        }

2. int main(int argc, char *argv[])

1. argc、argv的具体含义

        argc和argv参数用在命令行编译程序时有用,main(int argc, char **env)中,

第一个参数,int型argc,为整型,用来统计程序运行时发送给main函数的命令行参数的个数

第二个参数,char* 型的argv[],为字符串数组,用来存放指向字符串参数的指针数组,每个元素指向一个参数。各个成员含义如下:

  • argv[0] 指向程序运行的全路径名
  • argv[1] 指向在DOS命令行中执行程序名后的第一个字符串。
  • argv[2] 指向执行程序名后的第二个字符串
  • argv[3] 指向执行程序名后的第三个字符串
  • argv[argc] 为NULL

第三个参数,char** 型的env,为字符串数组,env[] 的每一个元素都包含ENVVAR=value形式的字符串,其中ENVVAR为环境变量,value为其对应的值。平时使用的比较少。

调试方法:

        方法一:

#include <stdio.h>

int main(int argc, char **argv) {
        int i;
        for(int i = 0; i < argc; i++) {
                printf("Argument %d is %s\n", i, argv[i]);
        }
        return 0;
}
sy@LAPTOP-LE7N15EN:~/sy/manage_memory$ ./main_proctice 1 2 3 4
Argument 0 is ./main_proctice
Argument 1 is 1
Argument 2 is 2
Argument 3 is 3
Argument 4 is 4

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值