Linux:开发工具(2)

news2025/1/17 15:21:47

一、Linux编译器-gcc/g++使用

1.1 为什么我们可以用C/C++做开发呢?

       无论是在windows、还是Linux中,C++的开发环境不仅仅指的是vs、gcc、g++,更重要的是语言本身的头文件(函数的声明)和库文件(函数的实现)。所以我们在安装这些软件的时候,同时也选择了相关的开发包,会同步下载对应的头文件和库文件。

     所以任何一款编译型语言的使用,都必要需要安装相应的开发包(头文件和库文件)

查看头文件:ls /usr/include/

1.2 预处理(进行宏替换)

预处理阶段会涉及到很多操作

a、去注释

    注释我们一般用于对我们的代码进行解释说明,但并不参与编译,所以是可以直接去掉的,节省文件的大小。

b、头文件展开

     头文件里面包含了我们需要的一些函数的声明,由于在链接之前各个文件都是独立进行编译和转汇编的,所以头文件将函数声明展示出来其实就是为了在编译过程的时候告诉编译器,这个函数是存在的,一定要放行,而最后的函数定义一般得等到链接的时候才能找到

c、条件编译

     条件编译其实就是有选择的编译,比较常见的一种情况比如说我们要通过打印来观察代码的运行情况(调试),但是仅仅只是为了起到一个调试的作用,所以我们调试后还要删掉其实有点可惜,所以我们可以通过条件编译来对他进行保留,在必要的时候启动这段代码或者是去掉这段代码。

还有一种技巧就是 -D(宏)           直接在外面定义宏 

一种使用条件编译的场景举例——vs的社区版(免费)和专业版(付费)

      一般在下载VS的时候可能会存在这两种版本,而一般我们购买那一种是根据我们的需求,而专业版是付费的所以必然功能是更加健全的,但从开发人员的角度,社区版和专业版难道需要维护两份代码么??其实他们是一份代码,只不过对于社区版,他们通过条件编译去掉了一些功能!!

命令:gcc -E test.c -o test.i 

告诉gcc,从现在开始进行程序编译,从预处理工作就停下来,不要往前走了 

1.3 编译(生成汇编) 

       在这个阶段中,gcc 首先要检查代码的规范性、是否有语法错误等(有错误会直接崩掉),以确定代码的实际要做的工作,在检查 无误后,gcc 把代码翻译成汇编语言。其实头文件里包含的函数声明,其实就是告诉编译器,这个函数是存在的,你要放行!

命令:gcc -S test.i -o test.s

告诉gcc,从现在开始进行程序编译,将编译过程做完就停下来 

1.4 汇编(生成机器可识别代码) 

汇编阶段是把编译阶段生成的“.s”文件转成目标文件(可重定位目标二进制文件)

命令:gcc -c test.i -o test.o

告诉gcc,从现在开始进行程序编译,汇编结束就停下来

注意:有执行权限和具有可执行能力是两回事,就好比一个富二代,他有继承家产的权利,但是他未必就有管理家业的能力——>所以.o文件是无法运行的!!需要经过链接才行!! 

1.5 链接(生成可执行文件或库文件)

 将目标文件和库进行链接,就得到了可执行程序

命令:gcc test.o -o test 

明明已经生成了机器可以读懂的文件,为什么还需要链接才能运行呢???

     原因就在于我们当前的文件并只有函数调用、函数声明,缺没有函数方法,所以必须要和库(C语言标准库,本质上就是一个文件)链接,函数的方法就存在于库中(其实就是把源文件.c经过一定的编译,然后打包,最后只给你提供一个文件)这样做有两个好处:

1、方便源文件的隐藏(未来我们想把程序给别人用,但不希望他看到源码,也可以这样)

2、不让我们做重复工作(帮我们造好了轮子),站在巨人的肩膀上 。

所以软件=头文件的方法+库文件提供的方法实现+你的代码

      个人觉得在日常的学习中,我们要尽可能地去尝试自己造轮子,研究底层,扎实内功,这样才可能应对未来的一些更复杂的情况,而在以后工作的时候,尽可能用一些已经写好的高效的代码,避免重复工作。而如果需要自己造轮子,早期的学习就会给你提供很大的帮助。

1.6 .o和库是如何链接的(静态链接和动态链接)

1.6.1 动态库文件和静态库文件

在Linux中: .so(动态库)  .a(静态库)

命名规则:libname.so.XXX

在windows中:.dll(动态库)  .lib(静态库)

在Linux中,通过ls /usr/lib64/libc.so*  可以看到我们的动态库文件

通过指令ldd 可以查看该可执行程序所依赖的动态库

 还有之前我们知道其实指令的本质就是可执行程序,所以我们也可以去查看指令所依赖的动态库,我们会发现大部分都是用C的库

静态库的文件默认是没有安装的,需要通过以下指令去手动安装

C静态库:sudo yum install -y glibc-static

C++静态库:sudo yum install -y libstdc++-static

 1.6.2 动态库和静态库理解

1、假设我们考上了一所高中,这里不让上网,但是你是一个喜欢打游戏的人,你通常会将打游戏这件事列入到自己每天的计划表中,而你的学校附近恰好有一个网吧,你在将计划进行到规定时间的时候就会翻墙出去上网,然后再回来。

自己的计划表——可执行程序

学校——编译器

网吧——动态库

网吧里的电脑——库文件

      也就是说当程序执行到某个地方时,他会跳出到动态库继续执行,然后再回来,这个过程就是动态链接。  

2、假设有一天网吧老板突然被举报,并且没有营业执照而被迫查封,这个时候我的计划就无法如期进行了,并且影响的不只是你,还有学校其他喜欢打游戏的人!

     所以动态库不能缺失!!一旦缺失影响的不仅仅是一个程序,而是多个程序都会崩溃!

3、突然有一天学校允许学生上网了,这个时候网吧老板捕捉到了商机,把网吧的电脑搬出来开了一家二手电脑专卖店,然后喜欢打游戏的同学都过来购买了电脑了!!

二手电脑专卖店——静态库

二手电脑——库文件

     也就是说,静态库进行静态链接的时候,会将自己的方法拷贝到目标程序中,该程序以后不用再依赖静态库!

4、有一天那个老板又被举报查封了,但是这个时候学校里的学校却没有人知道这件事,因为大家都有自己的电脑了,店查不查封对他们没什么影响。

    所以静态链接的程序并不依赖库文件,即使静态库丢失了程序也可以正常运行!

1.6.3 控制链接方式的选择

当我们不做限制时,会默认使用动态链接。如果我们想要静态链接 就需要加 -static选项

1、如果没有-static,默认编译可能会出现三种情况:动态链接->静态链接->报错。而有了static,就会去掉第一种情况,即静态链接->报错

2、不一定是纯的动态链接或者静态链接,也可以是混合的

1.6.4 动态链接和静态链接的优缺点

动态库:

优点:动态库是共享库,可以有效节省资源(磁盘空间、内存空间、网络空间)

缺点:动态库一旦缺失,所有的程序将无法运行

静态库:

优点:不依赖库,程序可以独立运行

缺点:体积大,消耗资源

       一般来说,我们在实际应用中更倾向于使用动态链接,因为体积大所带来的影响是很大的,比方说你下个游戏要1G,但是用静态链接可能就需要上百G,所以无论是我们还是Linux默认,都是会尽量选择动态链接。

1.6.5 debug&&release 

       debug可以被追踪调试,是因为在形成可执行程序的时候被添加了debug信息,所以占用的空间会更大一点。而release一般作为发行版,重在用户体验,所以体积会较小一点,我们Linux下编译默认是release版本,-g选项可以帮助我们改成debug版本。

扩展:可执行程序形成的时候,不是无序的二进制构成,而是有自己的格式的——>可执行程序有自己的二进制格式!(ELF格式)

 readelf -S可以查看可执行程序的二进制构成

二、Linux项目自动化构建工具-make/Makefile

为什么我们会需要自动化构建工具????

        一个工程中的源文件不计数,其按类型、功能、模块分别放在若干个目录中,makefile定义了一系列的 规则来指定,哪些文件需要先编译,哪些文件需要后编译,哪些文件需要重新编译,甚至于进行更复杂的功能操作。

     所以,makefile带来的好处就是——“自动化编译”,一旦写好,只需要一个make命令,整个工程完全自动编译,极大的提高了软件开发的效率

说明:

1、make是一个命令工具,是一个解释makefile中指令的命令工具,一般来说,大多数的IDE都有这个命 令,比如:Delphi的make,Visual C++的nmake,Linux下GNU的make。可见,makefile都成为了一 种在工程方面的编译方法。

2、make是一条命令,makefile是一个文件,两个搭配使用,完成项目自动化构建。

2.1 依赖关系和依赖方法

       举个例子,到月底的时候,你会给你爸打电话要下个月的生活费,这是因为你们两个之间存在依赖关系,而讨要生活费就是你们之间的依赖方法,现实中人与人之间能建立联系其实就是因为有依赖关系和依赖方法。 

      而我们使用makefile文件也同样需要依赖关系和依赖方法

我们会发现这个过程其实有点像是函数调用,特别像递归,形成了makefile的自动推导

所以其实我们乱序也可以,因为他会自己去找,如果找不到就会报错。

 2.2 make clean

还是上面这个情景,我们一make就生成了很多文件

但是我们如果想要清理的话,就需要用clean(不需要依赖关系)

 

 

为什么make执行前面的,而make clean执行后面的呢???

         原因就是因为,如果make没有指定的话,默认就是从第一行开始扫描到第一个make就执行,所以建议将编译的工作放到最前面,然后把清理的工作写到后面!!

 2.3 为什么不会允许多次make呢??(重点)

我们会发现我们make一次之后,就不然我们继续make了。这是为啥呢?

 1、为了提高编译效率!如果我们的文件相较于之前没有发生过任何的变化,那么就不能进行make(因为系统觉得没必要,纯属浪费时间),如果相较于之前的内容有更新过,那么就会重新形成一次新的可执行程序覆盖掉

 2、那么我们的系统究竟是如何做到判断这个文件是否被更新过呢??难道是将文件扫描一遍??显然不现实。其原理如下:

     首先,可执行程序一定是由源文件形成的,所以源文件的最近修改时间一定比可执行程序的最近时间要老——>推到得出,如果我们的源文件的最近修改文件比可执行程序时间新,那么就说明这个文件被修改过!!!

总结:只需要比较可执行程序的时间和源文件的最近修改时间即可!

.exe比.c老,说明被修改过,需要重新编译

.exe比.c新,说明没有被修改过,就不需要重新编译(提高了编译效率!)

 2.4 stat指令

stat指令是查看文件或者文件系统的详细信息

 Access表示访问时间,任何增删查改都算访问操作

Modify表示修改文件内容的时间

Change表示修改文件属性的时间

1/  按道理来说,Access的修改频率应该是最高的,但有些时候是无意义的,并且由于磁盘是属于外部设备,如果频繁访问的话,效率是极低的,所以在设计的时候,Access的时间其实是取决于另外两个时间的!

(1)当仅读取或者访问文件文件时,只有Access会改变

(2)当修改文件内容时,Modify(文件内容)和Change(文件大小)都会改变,Access不一定改变

(3)当修改文件权限属性时, 只有Change会改变

2/ .PHONY:

    所以make会根据源文件和目标文件的新旧,判定是否需要重新执行依赖关系进行编译,那么如果我们希望依赖关系总是被执行,就需要.PHONY:(伪目标)

     .PHONY:其实就是告诉make,这个目标是我的朋友,只要他想编你就让他编,不要阻拦,所以这个关键词常用在clean的身上。

3、 在一个文件不存在的时候,touch的作用是将该文件新建出来,但是如果这个文件存在的话,touch就可以将三个时间都修改成当前的时候,或者是用-a -m -c 选项来强制修改其中一个。

2.5 特殊符号 

1、$@和$^  分别指代依赖关系的前一个和后一个

2、依赖方法最前面+@ ——>对应的依赖方法在make后不会回显到屏幕上

三、Linux小程序-进度条

3.1 回车换行

其实回车和换行是两个概念,换行仅仅表示从当前位置跳到下一行,而回车则是回到该行的起点

其实\n 默认是换行+回车  而\r才是回车(键盘上的enter图标)

       以前的老式打印机,通过打字他会自动从左往右写,而当我们写到最右边的时候,先往下跳(换行),再把他拉到最左边接着写(回车)

3.2 缓冲区 

 首先认识一下sleep函数 参数表示休息多少秒 头文件是<unistd.h> 方便我们观察

 他会输出在输出hello Linux的时候,然后休眠2秒。 

 

如果我们去掉换行,则hello Linux会在两秒后才出现

思考:去掉了\n,为什么会是先休眠再输出呢??

        首先我们要肯定的是,代码运行的顺序必然是从前往后的,所以sleep一定是在printf之后执行的,这说明在休眠的时候,hello Linux必然被保存在某个地方,其实就是缓冲区!!缓冲区是由C语言维护的一段内存,他默认是按行刷新回显到显示器上!!所以如果我们想要提前刷新的话,就需要用fflush

参数可以是:标准输出(stdin)、标准输入(stdout)、标准错误(stderr) 

 这样就可以将缓冲区的内容刷新出来了!!!

3.3 简单倒计时的设置

首先我们需要processbar.c  processbar.h  main.c 三个文件

接下里先实现一个简单倒计时

我们要考虑两个问题:

(1) 倒计时要覆盖之前的位置,所以需要回车/r

(2)缓冲区是按行刷新的,所以必须要提前刷新

 但是我们一旦将cnt变成10以上,就会出现这样的情况

 

      这是因为我们的显示器是字符显示设备,所以他都是一个字符一个字符去打印的,你以为10是一个整体,其实对于他来说,是字符1和字符0  所以为了解决这个问题,我们需要去设置格式。比如说%-2d表示两个字符,格式左对齐

3.4 进度条基础版本

 改进的地方:

(1)需要一个[ ]括起来 

(2)需要一个数字来展示进度

(3)最后希望能有一个类似转圈的东西

 #include"processBar.h"                                          
  2 #define NUM 102                                            
  3 const char* lable="|/-\\";//\\转义为1个                        
  4 void processbar()                                             
  5 {                                                              
  6   char bar[NUM];                                               
  7   memset(bar,'\0',sizeof(bar));                                                                                 
  8   int len=strlen(lable);                                       
  9   int cnt=0;                                                   
 10   while(cnt<=100)                                              
 11   {                                                            
 12     printf("[%-100s][%d%%][%c]\r",bar,cnt,lable[cnt%len]);  
 13     fflush(stdout); //需要刷新出来                             
 14     bar[cnt++]=STYLE;                                          
 15     usleep(100000);                                            
 16   }                                                           
 17   printf("\n");                                              
 18 }    

3.5 进度条外部调用版本

        以上进度条是我们去设置进度的,但是实际我们的这个进度条应该由外部去调用,根据软件的具体情况去调用进度条,而具体的进度应该是由外部软件决定的。

 #pragma once                            |  1 #include"processBar.h"                     1 #include"processBar.h"
  2 #include<stdio.h>                       |  2                                          |  2 #define NUM 102
  3 #include<unistd.h>                      |  3 int main()                               |  3 const char* lable="|/-\\";//\\转义为1个
  4 #include<string.h>                      |  4 {                                        |  4 char bar[NUM];
  5 void processbar(int rate);              |  5   //模拟当前正在下载一款软件             |  5 void processbar(int rate)
  6 #define STYLE '#'//风格                 |  6   int total=1000;                        |  6 {
  7                                         |  7   int cur=0;                             |  7   if(rate<0||rate>100) return;//防止你非
  8                                         |  8   while(cur<=total)                      |    法
~                                           |  9   {                                      |  8   int len=strlen(lable);             
~                                           | 10     processbar(cur*100/total);           |  9     printf("[%-100s][%d%%][%c]\r",bar,ra
~                                           | 11     cur+=10;                             |    te,lable[rate%len]);
~                                           | 12     usleep(50000);                       | 10     fflush(stdout); //需要刷新出来
~                                           | 13   }                                      | 11     bar[rate++]=STYLE;              
~                                           | 14   printf("\n");                          | 12 }                                       
~                                           | 15   return 0;                              | 13                                   
~                                           | 16 }                               

3.6 进度条回调版本

       有些时候我们希望能进行多任务下载,因此可以将该进度条设置成回调函数,这样不同的软件就可以传对应不同的进度条。

 #pragma once                                 1 #include"processBar.h"                 |  1 #include"processBar.h"
  2 #include<stdio.h>                       |    2                                        |  2 #define NUM 102
  3 #include<unistd.h>                      |    3 int main()                             |  3 const char* lable="|/-\\";//\\转义为1个
  4 #include<string.h>                      |    4 {                                      |  4 char bar[NUM];
  5 void processbar(int rate);              |E>  5   download(processbar);                |  5 void processbar(int rate)
  6 #define STYLE '#'//风格                 |    6   return 0;                            |  6 {
  7 typedef void(*callback_t)(int);//回调函>|    7 }                                      |  7   if(rate<0||rate>100) return;//防止你非
    数                                      |    8                                        |    法
  8 void downLoad(callback_t cb);           |  ~                                          |  8   int len=strlen(lable);
  9                                         |  ~                                          |  9     printf("[%-100s][%d%%][%c]\r",bar,ra
~                                           |  ~                                          |    te,lable[rate%len]);
~                                           |  ~                                          | 10     fflush(stdout); //需要刷新出来
~                                           |  ~                                          | 11     bar[rate++]=STYLE;
~                                           |  ~                                          | 12 }
~                                           |  ~                                          | 13 
~                                           |  ~                                          | 14 void download(callback_t cb)
~                                           |  ~                                          | 15 {
~                                           |  ~                                          | 16 
~                                           |  ~                                          | 17   int total=1000;
~                                           |  ~                                          | 18   int cur=0;
~                                           |  ~                                          | 19   while(cur<=total)
~                                           |  ~                                          | 20   {
~                                           |  ~                                          | 21     cb(cur*100/total);
~                                           |  ~                                            22     cur+=10;
~                                           |  ~                                          | 23     usleep(50000);
~                                           |  ~                                          | 24   }
~                                           |  ~                                          | 25   printf("\n");

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2052293.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

WPF动画

补间动画&#xff1a;动画本质就是在一个时间段内对象尺寸、位移、旋转角度、缩放、颜色、透明度等属性值的连续变化。也包括图形变形的属性。时间、变化的对象、变化的值 工业应用场景&#xff1a;蚂蚁线、旋转、高度变化、指针偏移、小车 WPF动画与分类 特定对象处理动画过…

本地项目上传github

一、先在github&#xff08;GitHub: Let’s build from here GitHub&#xff09;上创建仓库 1&#xff0c;登录github后&#xff0c;点击右上角头像&#xff0c;点击 Your repositories 2&#xff0c;点击new 3&#xff0c;填写仓库名&#xff0c;假设命名 testhub&#xff0…

【机器学习】全景指南:从基础概念到实战流程的全面解析

文章目录 1.引言1.1机器学习的重要性1.2机器学习的应用范围1.3本文的内容结构 2. 机器学习的基本概念与分类2.1 机器学习的定义2.2 机器学习的分类 4. 强化学习&#xff08;Reinforcement Learning&#xff09; 3. 机器学习的工作流程3.1 数据收集与准备1. 数据源与类型2. 数据…

win10怎么查看CPU多核占用率

想要看自己有几个CPU处理器&#xff0c;可以在设备管理器里查看&#xff1a; 查看多核占用率&#xff0c;搜索任务管理器&#xff0c;然后打开&#xff0c;任务管理器——性能——CPU 右下角就可以看到我的是1个CPU&#xff0c;6个内核&#xff0c;12线程 想要看每个CPU占用…

Unity3D 自定义窗口

Unity3D 自定义窗口的实现。 自定义窗口 Unity3D 可以通过编写代码&#xff0c;扩展编辑器的菜单栏和窗口。 简单的功能可以直接一个菜单按钮实现&#xff0c;复杂的功能就需要绘制一个窗口展示更多的信息。 编辑器扩展的脚本&#xff0c;需要放在 Editor 文件夹中。 菜单栏…

用Python爬取高德地图路径规划数据——01. 指定起终点爬取-Python程序及详解

这个Python程序旨在从高德地图API获取路径规划数据&#xff0c;解析这些数据&#xff0c;并最终将其保存到JSON和CSV文件中。下面&#xff0c;我将详细讲解每个部分的功能和实现方式。 1. 导入所需的模块 import requests import json import time import csvrequests: 用于发…

spring boot自动配置

Spring自动配置是Spring框架的一个核心特性&#xff0c;它允许开发者通过在类路径下的配置类发现bean&#xff0c;而无需在应用程序中显式地进行bean的声明。Spring Boot利用这一特性&#xff0c;通过starter依赖的机制和EnableAutoConfiguration注解&#xff0c;帮助开发者快速…

Adobe After Effects AE V2023-23.6.6.2 解锁版下载安装教程 (视频合成和特效制作)

前言 Adobe After Effects&#xff08;简称AE&#xff09;是一款专业的图形视频处理软件&#xff0c;数字影视特效合成软件&#xff0c;视频后期特效制作软件。主要用来创建动态图形和视觉特效&#xff0c;支持2D以及3D&#xff0c;是基于非线性编辑的软件&#xff0c;透过图层…

原生js用Export2Excel导出excel单级表头和多级表头数据方式实现

原生js用Export2Excel导出excel单级表头和多级表头数据方式实现 原生js用Export2Excel导出excel单级表头和多级表头数据方式实现HTML文件导入需要的文件HTML文件中实现导出函数HTML总代码实现汇总&#xff08;直接复制代码&#xff0c;注意js引入路径&#xff09; 原生js用Expo…

小区社区超市商城停车场管理系统-计算机毕设Java|springboot实战项目

&#x1f34a;作者&#xff1a;计算机毕设匠心工作室 &#x1f34a;简介&#xff1a;毕业后就一直专业从事计算机软件程序开发&#xff0c;至今也有8年工作经验。擅长Java、Python、微信小程序、安卓、大数据、PHP、.NET|C#、Golang等。 擅长&#xff1a;按照需求定制化开发项目…

Ai学术叫叫兽全网最新创新点改进系列:丰富文章的热力图如何制作,论文装逼必用神器!极大丰富文章内容,并提升论文实验效果及其质量!

Ai学术叫叫兽全网最新创新点改进系列&#xff1a;丰富文章的热力图如何制作&#xff0c;论文装逼必用神器&#xff01;极大丰富文章内容&#xff0c;并提升论文实验效果及其质量&#xff01; 所有改进代码均经过实验测试跑通&#xff01;截止发稿时YOLOv8、YOLOv10均已改进40&…

tekton什么情况下在Dockerfile中需要用copy

kaniko配置如下 如果docker中的workDir跟tekton中的workDir不一致需要copy。也可以通过mv&#xff0c;cp达到类似效果

大数据——Flink原理

摘要 Apache Flink是一个框架和分布式处理引擎&#xff0c;用于对无界和有界数据流进行有状态计算。Flink被设计在所有常见的集群环境中运行&#xff0c;以内存执行速度和任意规模来执行计算。 1. FLink特点 1.1. 事件驱动型(Event-driven) 事件驱动型应用是一类具有状态的应…

基于 NXP LPC5516 + MC33665 + MC33774 的菊花链 HVBMS 储能方案

在 ESS 储能系统中&#xff0c;HVBMS 一般会采用两级或三级架构&#xff0c;从而实现从电池模组到电池簇&#xff0c;再到电池堆的分级管理和控制。本方案则给大家讲解基于 LPC5516 MC33665 MC33774 菊花链 HVBMS 方案&#xff0c;涵盖了 BMU、BJB、CMU&#xff0c;构建 HVBM…

简洁清新个人博客网页模板演示学习

30多款个人博客、个人网站、个人主页divcss,html在线预览,个人静态页面模板(wordpress、帝国cms、typecho主题模板).这些简洁和优雅的博客网页模板,为那些想成为创建博客的个人或媒体提供灵感设计。网页模板可以记录旅游、生活方式、食品或摄影博客等网站。部分网页模板来源网友…

MySQL8.0.0.28数据库安装配置

MySQL8.0.0.28数据库安装配置 1. 安装前的准备工作 1.1 确认目前服务器上是否存在MySQL 命令&#xff1a;rpm -qa | grep mysql 说明&#xff1a;若返回空信息&#xff0c;就说明当前环境没有安装MySQL&#xff1b;直接跳到第4步操作后续。 1.2 检查当前环境是否有自带的m…

系统编程-文件属性和目录操作

4 文件属性和目录操作 一、目录操作 -- 目录操作主要的目的&#xff1a;让程序知道路径下有什么文件存在 -- 注&#xff1a;程序在哪里运行他的工作路径就在哪里 &#xff0c;程序中所有的相对路径的文件就是基于工作路径来实现的 1、获取当前程序运行的工作路径 -- 函数头…

爬取豆瓣TOP250电影详解

一.分析网页DOM树结构 1.分析网页结构及简单爬取 豆瓣&#xff08;Douban&#xff09;是一个社区网站&#xff0c;创立于2005年3月6日。该网站以书影音起家&#xff0c;提供关于书籍、电影、音乐等作品的信息&#xff0c;其作品描述和评论都是由用户提供&#xff08;User-Gen…

MATLAB 手动实现一种高度覆盖值提取建筑物点云的方法(74)

专栏往期文章,包含本章 MATLAB 手动实现一种高度覆盖值提取建筑物点云的方法(74) 一、算法介绍二、算法实现1.代码2.效果总结一、算法介绍 手动实现一种基于高度覆盖值的建筑物点云提取方法,适用于高大的城市建筑物,比只利用高度提取建筑物的方法更加稳定和具有价值,主要…

NC 丑数

系列文章目录 文章目录 系列文章目录前言 前言 前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。点击跳转到网站&#xff0c;这篇文章男女通用&#xff0c;看懂了就去分享给你的码吧。 描述 把只包含质因…