C语言头文件组织与包含原则

2023-05-16

文章目录

  • 0. 说明
  • 1. 头文件作用
  • 2. 头文件组织原则
  • 3. 头文件包含原则
  • 4. 代码文件组织原则
  • 5. 注解
  • 题外话

转载自博客园——C语言头文件组织与包含原则

0. 说明

本文假定读者已具备基本的C编译知识。

如非特殊说明,文中“源文件”指*.c文件,“头文件”指*.h文件,“引用”指包含头文件。

1. 头文件作用

C语言里,每个源文件是一个模块,头文件为使用该模块的用户提供接口。

  • 接口指一个功能模块暴露给其他模块,用以访问具体功能的方法。
  • 使用源文件实现模块的功能,使用头文件暴露单元的接口。
  • 用户只需包含相应的头文件就可使用该头文件中暴露的接口。(头文件及对应的lib或者dll库文件)

头文件中的方法可以将程序中的各功能模块联系起来,有利于模块化程序设计:

  • 通过头文件调用库功能。
    • 实际工作中,源代码不便(或不准)向用户公布,只向用户提供头文件和二进制库。
    • 用户只需按照头文件中的接口声明来调用库功能,不必关心接口如何实现。
    • 编译器会从库中提取相应的代码。
  • 头文件能加强类型安全检查。
    • 若某个接口的实现或使用方式与头文件中的声明不一致,编译器就会指出错误。这一简单的规则能大大减轻程序员调试、改错的负担。
    • 在预处理阶段,编译器将源文件包含的头文件内容复制到包含语句(#include)处。
    • 在源文件编译时,连同被包含进来的头文件内容一起编译,生成目标文件(.obj)。如果所包含的头文件非常庞大,则会严重降低编译速度(使用GCC的-E选项可获得并查看最终预处理完的文件)。
    • 因此,在源文件中应仅包含必需的头文件,且尽量不要在头文件中包含其它头文件。

2. 头文件组织原则

  • 源文件src中实现变量、函数的定义,并指定链接范围。
  • 头文件inc中书写外部需要使用的全局变量、函数声明及数据类型和宏的定义。
  • 可以看看一些著名的C++库的组织方式,例如:Github——OpenCV。
    • 最外层是整体的opencv.hpp,这里包含下一级的头文件,例如:core.hpp,core.hpp又继续包含自己的下一级头文件,例如:base.hpp等

建议组织头文件内容时遵循以下原则:

  1. 头文件划分原则:

    • 类型定义、宏定义尽量与函数声明相分离,分别位于不同的头文件中。
    • 内部函数声明头文件与外部函数声明头文件相分离,内部类型定义头文件与外部类型定义头文件相分离。
    • 注意,类型和宏定义有时无法分拆为不同文件,比如结构体内数组成员的元素个数用常量宏表示时。因此仅分离类型宏定义与函数声明,且分别置于*.th*.fh文件(并非强制要求)。
  2. 头文件的语义层次化原则:

    • 头文件需要有语义层次。
    • 不同语义层次的类型定义不要放在一个头文件中,不同层次的函数声明不要放在一个头文件中。
  3. 头文件的语义相关性原则:

    • 同一头文件中出现的类型定义、函数声明应该是语义相关的、有内部逻辑关系的,避免将无关的定义和声明放在一个头文件中。
  4. 头文件名应尽量与实现功能的源文件相同

    • 即module.c和module.h。
    • 但源文件不一定要包含其同名的头文件。
    • 我觉得源文件和头文件名字一样,主要是为了程序可读性高,不至于程序猿自己都搞不清楚
  5. 头文件中不应包含本地数据,以降低模块间耦合度。

    • 即只有源文件自己使用的类型、宏定义和变量、函数声明,不应出现在头文件里。
    • 作用域限于单文件的私有变量和函数应声明为static,以防止外部调用。
    • 将私有类型置于源文件中,会提高聚合度,并减少不必要的格式外漏。
  6. 头文件内不允许定义变量和函数,只能有宏、类型(typedef/struct/union/enum等)及变量和函数的声明。特殊情况下可extern基本类型的全局变量,源文件通过包含该头文件访问全局变量。但头文件内不应extern自定义类型(如结构体)的全局变量,否则将迫使本不需要访问该变量的源文件包含自定义类型所在头文件[1]。

  7. 说明性头文件不需要有对应的源文件。此类头文件内大多包含大量概念性宏定义或枚举类型定义,不包含任何其他类型定义和变量或函数声明。此类头文件也不应包含任何其他头文件。

  8. 使用#pragma onceheader guard(亦称include guard或macro guard)避免头文件重复包含。#pragma once是一种非标准但已被现代编译器广泛支持的技巧,它明确告知预处理器“不要重复包含当前头文件”。而header guard则通过预处理命令模拟类似行为:

    1 #ifndef  _PRJ_DIR_FILE_H  //必须确保header guard宏名永不重名
    2 #define  _PRJ_DIR_FILE_H
    3 
    4 //<头文件内容>
    5 
    6 #endif
    

    使用#pragma once相比header guard具有两个优点[2]:

    • 更快。编译器不会第二次读取标记#pragma once的文件,但却会读若干遍使用header guard 的文件(寻找#endif);
    • 更简单。不再需要为每个文件的header guard取名,避免宏名重名引发的“找不到声明”问题。

    缺点则是:

    • #pragma once保证物理上的同一个文件不会被包含多次,无法对头文件中的一段代码作#pragma once声明。若某个头文件具有多份拷贝(内容相同的多个文件),pragma不能保证它们不被重复包含。当然,这种重复包含很容易被发现并修正。
  9. C++中要引用C函数时,函数所在头文件内应包含extern “C”[3]。被extern "C"修饰的变量和函数将按照C语言方式编译和连接,否则编译器将无法找到C函数定义,从而导致链接失败。

    //.h文件头部
    #ifdef  __cplusplus
    extern "C" {
    #endif
    
    //<函数声明>
    
    //.h文件尾部
    #ifdef  __cplusplus
    }
    #endif
    
  10. 头文件内要有面向用户的充足注释,从应用角度描述接口暴露的内容。

3. 头文件包含原则

在实际编程中,常常因头文件包含不当而引发编译时报告符号未定义的错误或重复定义的警告。要消除符号未定义的编译错误,只需在引用符号(变量、函数、数据类型及宏等)前确保它已被声明或定义[4]。要消除重复定义的警告,则需合理设计头文件包含顺序和层次。

建议包含头文件时遵循以下原则:

  1. 源文件内的头文件包含顺序应从最特殊到一般,如:

    #include "通用头文件"  //内部可能定义本模块数据类型别名	
    #include "源文件同名头文件"	
    #include "本模块其他头文件"	
    #include "自定义工具头文件"	
    #include "第三方头文件"	
    #include "平台相关头文件"	
    #include "C++库头文件"	
    #include "C库头文件"
    

    优点是每个头文件必须include需要的关联头文件,否则会报错。同时,源文件同名头文件置于包含列表前端便于检查该头文件是否自完备,以及类型或函数声明是否与标准库冲突。

  2. 减少头文件的嵌套和交叉引用,头文件仅包含其真正需要显式包含的头文件。

    • 头文件的嵌套和交叉引用会使程序组织结构和文件组织变得混乱,同时造成潜在的错误。
    • 大型工程中,原有头文件可能会被多个其他(源或头)文件包含,在原有头文件中添加新的头文件往往牵一发而动全身。
    • 若头文件中类型定义需要其他头文件时,可将其提出来单独形成一个全局头文件。
  3. 头文件应包含哪些头文件仅取决于自身,而非包含该头文件的源文件。
    例如,编译源文件时需要用到头文件B,且源文件已包含头文件A,而索性将头文件B包含在头文件A中,这是错误的做法。

  4. 尽量保证用户使用此头文件时,无需手动包含其他前提头文件,即此头文件内已包含前提头文件。
    例如,面积相关操作的头文件Area.h内已包含关于点操作的头文件Point.h,则用户包含Area.h后无需再手动包含Point.h。这样用户就不必了解头文件的内在依赖关系。

  5. 头文件应是自完备的,即在任一源文件中包含任一头文件而不会产生编译错误。

  6. 源文件中包含的头文件尽量不要有顺序依赖。

  7. 尽量在源文件中包含头文件,而非在头文件中。且源文件仅包含所需的头文件。

  8. 头文件中若能前置声明(亦称前向声明[5]),就不要包含另一头文件。仅当前置声明不能满足或过于麻烦时才使用include,如此可减少依赖性方面的问题。示例如下:

    struct T_MeInfoMap;  //前置声明
    struct T_OmciMsg;    //前置声明
    
    typedef FUNC_STATUS (*OmciChkFunc)(struct T_MeInfoMap *ptMeInfo, struct T_OmciMsg *ptMsg, struct T_OmciMsg *ptAckMsg);
    
    //OMCI实体信息
    typedef struct{
        INT16U wMeClass;               //实体类别
        OMCI_ATTR_INFO *pMeAttrInfo;   //实体所定义的属性信息指针
        INT8U  ucAttrNum;              //实体所定义的属性数目
        INT16U wTotalAttrLen;          //实体所有属性所占的总字节数,初始化为0,动态计算
        INT8U  *pszDbName;             //实体存库时的数据表名称,建议不要超过DB_NAME_LEN(32)
        INT16U wMaxRecNum;             //实体存库时支持的最大记录数目
        OmciChkFunc fnCheck;           //Omci校验函数指针
        BOOL   bDbCreated;             //实体数据表是否已创建
    }OMCI_ME_INFO_MAP;
    

    如上,在OmciChkFunc函数的实现源文件内包含T_MeInfoMap和T_OmciMsg所在头文件即可。 另举一例如下:

    typedef TBL_SET_MODE (*OperTypeFunc)(INT8U *pTblEntry);
    
    typedef INT8U (*CmpRecFunc)(VOID *pvCmpData, VOID *pvRecData); //为避免头文件交叉引用,与CompareRecFunc异名同构
    
    //表属性信息
    typedef struct{
      INT16U wMaxEntryNum;         //表属性最大表项数目(实体记录数目wMaxRecNum * wMaxEntryNum <= MAX_RECORD_NUM)
      OperTypeFunc fnGetOperType;  //操作类型函数指针。根据表项数据或外界需求(只读表)解析当前表项操作类型
      TBL_KEY_INFO tCmpKeyInfo;    //检索表属性子表记录时的匹配关键字信息(TBL_KEY_INFO)
      CmpRecFunc   fnCmpAddKey;    //增加表项时需要检测的关键字匹配函数指针
      CmpRecFunc   fnCmpDelKey;    //删除表项时需要检测的关键字匹配函数指针
      INT16U wTblEntrySize;        //表属性表项字节数,由外部动态赋值
    }TBL_ATTR_INFO;
    

    如上,CompareRecFunc函数原型由其他头文件提供,此处为避免头文件交叉引用定义其异名同构原型CmpRecFunc。

    在不会引起歧义的前提下,头文件内尽可能使用VOID指针代替非基本类型的值变量或指针,以避免再包含类型定义所在的头文件。但这将影响代码可读性并降低程序执行效率,应权衡利弊。

  • 避免包含重量级的平台头文件,如windows.h或d3d9.h等。若仅使用该头文件少量函数,可extern函数到源文件内。如下:
    	1 /**********************************************************************************************
    	2                       外部函数声明 (当外部接口未提供头文件或头文件过于复杂时) 
    	3 **********************************************************************************************/
    	4 //因声明所在头文件引用混乱,此处仅extern函数声明。
    	5 extern INT32S DBShmCliInit(VOID); //#include "db_shm_mgr.h"
    	6 extern INT32S cmLockInit(VOID);   //#include "common_cmapi.h"
    	```
    
  • 若还使用该头文件某些类型和宏定义,可创建适配性源文件。在该源文件内包含平台头文件,封装新的接口并将其声明在同名头文件内,其他源文件将通过适配头文件间接访问平台接口。如下:
    /*****************************************************************************************
    * 文件名称: Omci_Send_Msg.c
    * 内容摘要: OMCI消息转发接口
    * 其它说明: 该头文件封装SEND接口,以避免其他源文件包含支撑api和pid公共头文件导致引用混乱。
     *****************************************************************************************/
    
    
    #include "Omci_Common.h"
    #include "Omci_Send_Msg.h"
    #include "oss_api.h"
    /**********************************************************************************************	
    函数实现区
    **********************************************************************************************/
    
    //向自身进程发送异步消息
    INT32U OmciAsynSendSelf(INT16U wEvent, VOID *pvMsg, INT16U wMsgLen)
    {
        PID dwSelfPid = 0;
        SELF(&dwSelfPid);
        return ASEND(wEvent, pvMsg, wMsgLen, dwSelfPid);
    }
    
  • 对于函数库(包括标准库和自定义的公共宏及接口)的头文件,可将其加入到一个通用头文件中。需要控制该头文件的体积(主要是该头文件所包含的所有头文件内容大小),并确保所有源文件首先包含该通用头文件。示例如下:
    #ifndef  _OMCI_COMMON_H
    #define  _OMCI_COMMON_H
    
    
    /*******************************************************************************************
    * 说明:
    * 本文件仅应包含与具体通信协议无关的通用数据类型及宏定义。
    * 为简化头文件包含且不失可移植性,本文件内可包含少量C库通用头文件。
    * 因本文件内定义基本数据类型别名,故.c文件中应将本头文件置于包含列表顶端,
    * 否则编译时可能产生类型未定义错误。
    *******************************************************************************************/
    
    
    #include <stdio.h>
    #include <stdlib.h>
    #include <string.h>
    #include <sys/time.h>
    #include <limits.h>
    
    #include "Omci_Byte.h"
    
    
    //<Other Contents...>
    
    注意,示例头文件内包含C库文件虽能简化包含,但却与规则1冲突。也可另外增加包含库文件列表的通用头文件。
  • 若不确定类型、宏定义或函数声明所在头文件具体路径,可在源文件中再次定义或声明,编译器会以redefined警告或conflicting错误给出类型、宏定义或函数声明所在头文件路径。

4. 代码文件组织原则

建议C语言项目中代码文件组织遵循以下原则:

  1. 使用层次化和模块化的软件开发模型。每个模块只能使用所在层和下一层模块提供的接口。
  2. 每个模块的文件(可能多个)保存在一个独立文件夹中。
    模块文件较多时可采用子目录的方式,物理上隔离不同层次的文件。子目录下源文件和头文件应分开存放,如分别置入include和source目录。
  3. 用于模块裁减的条件编译宏保存在一个独立文件中,便于软件裁减。
  4. 硬件相关代码和操作系统相关代码与工程代码相对独立保存,以便于软件移植。
  5. 按相同功能或相关性组织源文件和头文件。同一文件内的聚合度要高,不同文件中的耦合度要低。
    在对既有工程做单元测试时,耦合度低的文件布局非常便于搭建环境。
  6. 声明和定义分开,使用头文件暴露模块需要提供给外部的类型、宏、变量和函数。尽量做到模块对外部透明,用户在使用模块功能时无需了解具体的实现。
  7. 作为对外接口的头文件一经发布,应保持稳定。修改时一定要慎重。
  8. 文件夹和文件命名要能够反映出模块的功能。
  9. 正式版本和测试版本使用统一文件,使用宏控制是否产生测试输出。
  10. 必要的注释不可缺少。

5. 注解

【注1】全局变量的使用原则

  • 若全局变量仅在单个源文件中访问,则可将该变量改为该文件内的静态全局变量;
  • 若全局变量仅由单个函数访问,则可将该变量改为该函数内的静态局部变量;
  • 尽量不要使用extern声明全局变量,最好提供函数访问这些变量。直接暴露全局变量是不安全的,外部用户未必完全理解这些变量的含义。
  • 设计和调用访问动态全局变量、静态全局变量、静态局部变量的函数时,需要考虑重入问题。

【注2】#pragma once的可移植性

  • #ifndef由C/C++语言标准支持,不受编译器任何限制;
  • #pragma once仅由编译器提供保证,存在可移植性等问题。
  • 某些gcc编译器版本(如3.2.3)会报告warning: #pragma once is obsolete的警告,而其他较老版本的编译器可能会报错。但随着gcc 3.4的发布,#pragma once中的一些问题(主要与符号链接和硬链接有关)得以解决,#pragma once命令也标记为“未废弃”。
  • 还有种写法同时使用#pragma once和header guard编写“可移植性”代码,以利用编译器可能支持的#pragma once优化。如下:
    #pragma once
    #ifndef    _PRJ_DIR_FILE_H
    #define   _PRJ_DIR_FILE_H
    
    //<头文件内容>
    
    #endif
    
  • 该法似乎兼有两者的优点。但既然使用#ifndef就有宏名重名的风险,也无法避免不支持#pragma once的编译器告警或报错,故混用两种方法似乎不能带来更多的好处,反倒让不熟悉的人感到困惑。
  • 注意,如果使用header guard,理论上可在代码任何地方判断当前是否已经包含某个头文件。但应避免通过该判断来改变后续代码的逻辑走向!这种做法将使程序依赖于头文件的包含顺序,极不可取。若需要实现“若当前包含HeaderA.h,才加入StructB结构”,可对StructB结构创建HeaderB.h头文件,在HeaderA.h中包含HeaderB.h。

【注3】extern “C”

  • C++语言在编译时为实现函数重载,会结合函数名、参数数目及类型信息而生成一个中间函数名。
    • 例如,C++中函数void foo(int x, float y)编译后在符号库中生成的名字为_foo_int_float(不同编译器可能生成不同函数名,但均采用相同机制,生成的新名字称为”mangled name”);
    • 而该函数被C编译器编译后在符号库中的名字为_foo。
  • C语言中不支持extern "C"声明,在.c文件中包含extern "C"时会出现编译语法错误。
  • 当然编译器也可以为其他语言提供链接说明。例如:extern “FORTRAN”、extern "Ada"等。

【注4】声明(declaration)与定义(definition)

全局变量或函数可(在多个编译单元中)有多处声明,但只允许定义一次。全局变量定义时分配空间并赋初始值(如果有);函数定义时提供函数体内容。

// 声明:
extern int iGlobal;
extern int func();int func();
//定义:
int iGlobal = 0; //或int iGlobal;
int func (){return 1;}

在多个源文件中共享变量或函数时,需确保定义和声明的一致性。通常在某个相关的源文件中定义,然后在头文件中进行外部声明。需要使用时包含相应的头文件即可。定义变量的源文件也应包含该头文件,以便编译器检查定义和声明的一致性。

该规则可提供高度的可移植性:它与ANSI/ISO C标准一致,同时也兼顾大多数ANSI前的编译器和链接器。(Unix编译器和链接器常使用允许多重定义的“通用模式”,只要保证最多对一处定义进行初始化即可。该方式被ANSI C标准称为一种“通用扩展”)。某些很老的系统可能要求显式初始化以区别定义和外部声明。

 通用扩展在《深入理解计算机系统》中解释为:多重定义的符号只允许最多一个强符号。函数和定义时已初始化的全局变量是强符号;未初始化的全局变量是弱符号。Unix链接器使用以下规则来处理多重定义的符号:
  • 规则一:不允许有多个强符号。在被多个源文件包含的头文件内定义的全局变量会被定义多次(预处理阶段会将头文件内容展开在源文件中),若在定义时显式地赋值(初始化),则会违反此规则。
  • 规则二:若存在一个强符号和多个弱符号,则选择强符号。
  • 规则三:若存在多个弱符号,则从这些弱符号中任选一个。

当不同文件内定义同名(即便类型和含义不同)的全局变量时,该变量共享同一块内存(地址相同)。若变量定义时均初始化,则会产生重定义(multiple definition)的链接错误;若某处变量定义时未初始化,则无链接错误,仅在因类型不同而大小不同时可能产生符号大小变化(size of symbol `XXX’ changed)的编译警告。在最坏情况下,编译链接正常,但不同文件对同名全局变量读写时相互影响,引发非常诡异的问题。这种风险在使用无法接触源码的第三方库时尤为突出。

因此,应尽量避免使用全局变量。若确有必要,应采用静态全局变量(无强弱之分,且不会和其他全局符号产生冲突),并封装访问函数供外部文件调用。


【注5】前向声明(forward declaration)

结构体类型S在声明之后定义之前是一个不完全类型(incomplete type),即已知S是一个类型,但不知道包含哪些成员。不完全类型只能用于定义指向该类型的指针,或声明使用该类型作为形参指针类型或返回指针类型的函数。指针类型对编译器而言大小固定(如32位机上为四字节),不会出现编译错误。

假设先后定义两个结构A和B,且两个结构需要互相引用。在定义A时B还没有定义,则要引用B就需要前向声明结构B(struct B;)。示例如下:

1 typedef BOOL (*func)(const DefStruct *ptStrt);
2 
3 typedef struct DefStruct_t{
4     int i;
5     func f;
6 }DefStruct;

如上在DefStruct中使用回调函数func声明,这样交叉引用必然编译报错。进行前向声明即可:

1 typedef struct DefStruct_t DefStruct;
2 typedef BOOL (*func)(const DefStruct *ptStrt);
3 
4 struct DefStruct_t{
5     int i;
6     func f;
7 };

注意,在前向声明和具体定义之间涉及标识符(变量、结构、函数等)实现细节的使用都是非法的。若函数被前向声明但未被调用,则编译和运行正常;若前向声明函数被调用但未被定义,则编译正常但链接报错(undefined reference)。将具体定义放在源文件中可部分避免该问题。

题外话

我直接搜到的网页是:

  • CSDN博客:C语言头文件组织与包含原则~,2020年
  • 知乎文章:发掘好文:C语言头文件组织与包含原则,2020年

以上两个文章均转载自博客园——C语言头文件组织与包含原则,2014年。

  • 博客园被搜索引擎收录的情况确实比不上知乎和CSDN啊,搜到的永远都是转载
  • 虽然很多都没看懂,但是以后应该可以看懂吧。
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

C语言头文件组织与包含原则 的相关文章

  • 研讨会 | 知识图谱前沿技术课程暨学术研讨会(武汉大学站)

    知识图谱作为大数据时代重要的知识表示方式之一 xff0c 已经成为人工智能领域的一个重要支撑 4月 28日 xff0c 武汉大学信息集成与应用实验室 与 复旦大学知识工场实验室 联合举办 知识图谱前沿技术课程暨学术研讨会 xff0c 将结合
  • jdbc中Statement和PreparedStatement有什么区别?哪个性能更好?

    Statement和PreparedStatement的功能主要是对sql语句的执行 区别 xff08 1 xff09 Statement每执行一条sql语句就需要生成一条执行计划 xff0c 执行100条就需要100条执行计划Prepar
  • redis的特性

    redis的特性 承接上文redis入门篇 xff0c 本文具体介绍一下redis的特性 xff0c 以及与另外一个nosql数据库memcached的对比 一 redis的优点 根据上文 xff0c 我们知道redis的如下特性成为了他的
  • Ubuntu22.04安装windows字体

    找到C Windows目录 xff0c 将其中的Fonts文件夹拷贝至ubuntu中 将该文件夹放至ubuntu的 usr share fonts目录下面 xff0c 可用下列命令 span class token function sud
  • 阿里巴巴笔试题选解

    阿里巴巴笔试题选解 9月22日 xff0c 阿里巴巴北邮站 小题 xff1a 1 有三个结点 xff0c 可以构成多少种二叉树形结构 xff1f 2 一副牌52 张 去掉大小王 xff0c 从中抽取两张牌 xff0c 一红一黑的概率是多少
  • ActiveMQ与Logback日志组件SLF4J冲突导致日志不输出

    ActiveMQ与Logback中的SLF4J日志组件冲突导致日志不输出 xff0c 控制台提示 Class path contains multiple SLF4J bindings 的解决方案 近期码的时候发现logback的组件日志都
  • 腾讯2014软件开发笔试题目

    腾讯2014软件开发笔试题目 9月21日 xff0c 腾讯2014软件开发校招 简答题 广州 简答题 xff1a 1 请设计一个排队系统 xff0c 能够让每个进入队伍的用户都能看到自己在 中所处的位置和变化 队伍可能随时有人加入和退出 x
  • C++ 进程间通信详解

    一 xff0c C 43 43 常用进程间通信 管道 Pipe xff1a 管道可用于具有亲缘关系进程间的通信 xff0c 允许一个进程和另一个与它有共同祖先的进程之间进行通信 命名管道 named pipe xff1a 命名管道克服了管道
  • OTA系统包的制作和测试方法

    OTA有两种制作方案 整包升级 xff0c 以及差分包升级 整包升级 完整的升级文件aosp XXXX XXX ota 01 20 010 00 00 zip 差异包 将第一个ota整包升级包和第二个ota整包升级包 xff0c 执行 bu
  • JAVA日记之javaJDK原生简单爬虫

    java原生爬虫 指定一个种子url放入到队列中 从队列中获取某个URL 使用HTTP协议发起网络请求 在发起网络请求的过程中 xff0c 需要将域名转化成IP地址 xff0c 也就是域名解析 得到服务器的响应 xff0c 此时是二进制的输
  • idea中修改自定义maven打包参数,打成一个大的jar

    解决idea打包只会把项目本身打成一个jar的问题 加上这条指令 clean package spring boot repackage Dmaven test skip 61 true f pom xml 在maven命令单这个里会生成新
  • 虚拟机桥接后更换外部网络无法访问

    另外一个网卡也改成这样 然后关闭再打开即可
  • kafka安装配置详解和安装成功验证

    kafka安装后测试 kafka配置项详解 默认 kafka server properties 配置如下 xff1a Server Basics 服务器基础知识 The id of the broker This must be set
  • jvm类加载器以及类加载原理学习

    宏观运行java com tuling jvm Math class命令后执行的流程 底层C 43 43 调用sun misc launcher getlauncher 方法 xff0c 获取launcher对象 xff0c jvm全局唯一
  • httpclient 开启ssl后加代理代理死锁

    记录一个遇到的问题 httpclient 开启ssl后加代理代理死锁 最近做一个爬虫使用搜狗识图功能 xff0c 由于ip问题需要增加代理 xff0c 使用httpclient4 5 3 开启了ssl 代码由异步线程执行 在查看日志时候发现
  • JAVA日记之mybatis-3一对一,一对多,多对多xml与注解配置 ----喝最烈的酒.

    1 Mybatis多表查询 1 1 一对一查询 1 1 1 一对一查询的模型 用户表和订单表的关系为 xff0c 一个用户有多个订单 xff0c 一个订单只从属于一个用户 一对一查询的需求 xff1a 查询一个订单 xff0c 与此同时查询
  • pandas 读取文件时的设置header

    用pandas 中的read table 函数时 xff0c 发现header设置值不一样 xff0c 所获得的结果也不一样 之前一直认为header 61 0 和header 61 None是一样的 xff0c 其实是不一样的 读取一个有
  • 阿里 arthas 使用介绍

    背景 xff1a 一次线上问题的综合排查排查 xff0c 两个相同的系统的某个模块 xff0c 数据量更少的系统查询更慢 先说下整体思路 xff1a 查看系统整理负载 xff0c 网络有100左右毫秒的延迟 xff0c 看起来影响不大查看正
  • Linux - 桌面的窗口的标题栏与边框缺失

    Linux 桌面的窗口的标题栏与边框缺失 原因 是Window Manager进程异常了 xff0c 需要重启 方法 以xfce桌面为例 xff0c 方法一 xff1a 执行xfwm4 replace 方法二 xff1a 执行killall

随机推荐