Linux高并发服务器开发学习笔记1(环境编程)

目录

第一章:静态库动态库

1.静态库

2.动态库(共享库)

第二章:Makefile

1.概述

2.Makefile文件的制作

3.其他 

 第三章:GDB调试

1.概述

2.命令 

第四章:系统编程其他笔记

open

stat


第一章:静态库动态库

1.静态库

 命名规则:

制作静态库:

先通过汇编转为.o文件:

gcc add.c div.c mult.c  sub.c  -c 

再制作静态库 

~/Desktop/linux/calc$ ar rcs libwl.a add.o div.o sub.o mult.o

使用静态库:

└── library
    ├── include
    │   └── head.h
    ├── lib
    │   └── libwl.a
    ├── main.c
    └── src
        ├── add.c
        ├── div.c
        ├── mult.c
        └── sub.c
 gcc ./library/main.c -o app2 -I ./library/include/ -L ./library/lib/ -l wl
main.c:
#include <stdio.h>
#include "head.h"//这个为相对路径,在同级目录下自动找不到,需要-I ./library/include/  需要在这个路径里面找.h

int main()
{
    int a = 20;
    int b = 12;
    printf("a = %d, b = %d\n", a, b);
    printf("a + b = %d\n", add(a, b));
    printf("a - b = %d\n", subtract(a, b));
    printf("a * b = %d\n", multiply(a, b));//这些函数定义在lib中,-L找库路径 -l指定库名字
    printf("a / b = %f\n", divide(a, b));
    return 0;
}

-L ./library/lib/:指定库路径 

-l wl :指定用这个库(定义了加减乘除的函数)

libwl.a :head.h中的函数声明函数定义打包。

head.h

#ifndef _HEAD_H
#define _HEAD_H
int add(int a, int b);
int subtract(int a, int b);
int multiply(int a, int b);
double divide(int a, int b);
#endif
// #ifndef.....#define.....#endif可以避免编译时的重复定义,但是无法避免多个文件链接时的重复定义。头文件应该尽量少的include其他头文件。

定义函数(记得包含head.h):

#include <stdio.h>
#include "head.h"
int add(int a, int b)
{
    return a+b;
}

2.动态库(共享库)

使用动态库和使用动态库一样 ,但执行可执行文件的时候会报错:

 找原因,先看原理:

 动态库GCC链接的时候,只能是告诉main “运行时需要libwl.so的信息”,但是信息没有写进去,需要在运行的时候去搜索libwl.so的路径然后加载到内存中。

办法1:临时配置(第二次开启需要重新配置)

export LD_LIBRARY_PATH=$LD_LIBRART_PATH:/home/wuliwlll/Desktop/dongtaiku/library/lib/

在LD_LIBRARY_PATH添加libwl.so的路径

办法2:内置配置(用户级)

1.在wuliwlll用户目录下配置.bashrc

2.shift+g 在最后一行添加如下。

3.然后启动 

 4.运行即可

办法3:内置配置(管理级)

在此文件最后一行加入路径,如方法2所示,然后启动。

第二章:Makefile

1.概述

make指令可以自动解释makefile文件内容,达到自动化编译的效果

2.Makefile文件的制作

wuliwlll@wuliwlll-virtual-machine:~/Desktop/makefile$ make
gcc sub.c add.c mult.c div.c main.c -o app

make自动解析Makefile,执行里面的命令,然后.c文件的头文件会自动调用当前目录的头文件,如果头文件不在当前目录,需要如下操作:

.
├── add.c
├── div.c
├── head
│   └── head.h
├── main.c
├── Makefile
├── mult.c
└── sub.c

Makefile里面需要添加头文件目录 (此头文件里是函数声明)

app2:sub.c add.c mult.c div.c main.c
        gcc sub.c add.c mult.c div.c main.c -o app2 -I ./head/

然后make,执行

3.其他 

关于变量

自定义变量:

模式匹配:

makefile中的函数:

src这个变量,右边用了函数,返回.c的所有文件名,默认空格隔开,是字符串

objs:右边表示用模式匹配法的函数来替换,用.o替换.c的文件,返回值为所有名称匹配上的.o文件名字符串

 第三章:GDB调试

1.概述

是GNU软件系统社区提供的调试工具

2.命令 

 

 

第四章:系统编程其他笔记

每一个进程会被分配虚拟地址,通过MMU会将虚拟地址的地址映射到物理地址 

open

#include <sys/types.h>
    #include <sys/stat.h>
    #include <fcntl.h>

    // 打开一个已经存在的文件
    int open(const char *pathname, int flags);
        参数:
            - pathname:要打开的文件路径
            - flags:对文件的操作权限设置还有其他的设置
              O_RDONLY,  O_WRONLY,  O_RDWR  这三个设置是互斥的
        返回值:返回一个新的文件描述符,如果调用失败,返回-1

    errno:属于Linux系统函数库,库里面的一个全局变量,记录的是最近的错误号。

    #include <stdio.h>
    void perror(const char *s);作用:打印errno对应的错误描述
        s参数:用户描述,比如hello,最终输出的内容是  hello:xxx(实际的错误描述)
    
    
    // 创建一个新的文件 mode是权限限制  如0777  最终为0777 & ~umask   
    int open(const char *pathname, int flags, mode_t mode);//umask是为了抹除一些权限,有保护和限制的作用

为什么flags是用 | 呢。

因为flag是个有32位的标志位(int类型),每一位的1和0代表是否生效,若第一位为1是 为O_RDWD权限开启,第四位为O_CREAT,两个flag或运算后相当于均生效。 

lseek read write不笔记了。(14条消息) Linux系统调用四、lseek()函数详解_lseek函数_Mindtechnist的博客-CSDN博客

stat

实现ll -l案例,实现./ls  a.txt


#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <pwd.h>
#include <grp.h>
#include <time.h>
#include <string.h>

// 模拟实现 ls -l 指令
// -rw-rw-r-- 1 nowcoder nowcoder 12 12月  3 15:48 a.txt
int main(int argc, char * argv[]) {
//argc: 整数,用来统计你运行程序时送给main函数的命令行参数的个数   比如./app text  就是两个参数
//*argv[ ]: 指针数组,用来存放指向你的字符串参数的指针,每一个元素指向一个参数
//argv[0] 指向程序运行的全路径名
//argv[1] 指向在DOS命令行中执行程序名后的第一个字符串
//argv[2] 指向执行程序名后的第二个字符串
    // 判断输入的参数是否正确
    if(argc < 2) {
        printf("%s filename\n", argv[0]);
        return -1;
    }

    // 通过stat函数获取用户传入的文件的信息
    struct stat st;
    int ret = stat(argv[1], &st);
    if(ret == -1) {
        perror("stat");
        return -1;
    }

    // 获取文件类型和文件权限
    char perms[11] = {0};   // 用于保存文件类型和文件权限的字符串

    switch(st.st_mode & S_IFMT) {//跟掩码与运算后,后面位数清零,只保留前四位
        case S_IFLNK://通过前四位判断类型  比如S_IFLNK为0120000,如果st.mode前四位也是0 (0是八进制标志位)12000前四位,则输出'l'
            perms[0] = 'l';
            break;
        case S_IFDIR:
            perms[0] = 'd';
            break;
        case S_IFREG:
            perms[0] = '-';
            break; 
        case S_IFBLK:
            perms[0] = 'b';
            break; 
        case S_IFCHR:
            perms[0] = 'c';
            break; 
        case S_IFSOCK:
            perms[0] = 's';
            break;
        case S_IFIFO:
            perms[0] = 'p';
            break;
        default:
            perms[0] = '?';
            break;
    }

    // 判断文件的访问权限

    // 文件所有者
    perms[1] = (st.st_mode & S_IRUSR) ? 'r' : '-';
    perms[2] = (st.st_mode & S_IWUSR) ? 'w' : '-';
    perms[3] = (st.st_mode & S_IXUSR) ? 'x' : '-';

    // 文件所在组
    perms[4] = (st.st_mode & S_IRGRP) ? 'r' : '-';
    perms[5] = (st.st_mode & S_IWGRP) ? 'w' : '-';
    perms[6] = (st.st_mode & S_IXGRP) ? 'x' : '-';

    // 其他人
    perms[7] = (st.st_mode & S_IROTH) ? 'r' : '-';
    perms[8] = (st.st_mode & S_IWOTH) ? 'w' : '-';
    perms[9] = (st.st_mode & S_IXOTH) ? 'x' : '-';

    // 硬连接数
    int linkNum = st.st_nlink;

    // 文件所有者
    char * fileUser = getpwuid(st.st_uid)->pw_name;

    // 文件所在组
    char * fileGrp = getgrgid(st.st_gid)->gr_name;

    // 文件大小
    long int fileSize = st.st_size;

    // 获取修改的时间
    char * time = ctime(&st.st_mtime);

    char mtime[512] = {0};
    strncpy(mtime, time, strlen(time) - 1);

    char buf[1024];
    sprintf(buf, "%s %d %s %s %ld %s %s", perms, linkNum, fileUser, fileGrp, fileSize, mtime, argv[1]);

    printf("%s\n", buf);

    return 0;
}

 lstat是获取软连接信息

readdir中的返回值

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值