CSAPP Lab07——Malloc Lab完成思路

等不到天黑

烟火不会太完美

回忆烧成灰

还是等不到结尾

——她说

完整代码见:CSAPP/malloclab-handout at main · SnowLegend-star/CSAPP (github.com)

 

Malloc Lab

按照惯例,我先是上来就把mm.c编译了一番,结果产生如下报错。搜索过后看样子应该是编译器的版本不匹配,得建立条软链接。

经过多番尝试,最后得到正确的链接形式是

“ln -s /usr/include/x86_64-linux-gnu/asm /usr/include/asm”

隐式空闲链表

这方法就是把书上的那几个函数搬过来就行,唯一需要自己动手的是realloc()函数。但是书上的那一大块宏定义属实是看得我两眼发昏。特别是NEXT_BLKP(bp)PREV_BLKP(bp)这两个定义,直接把我绕晕了。

后来仔细分析了一番,发现最根本的原因是我把隐式空闲链表的头/尾块和内存块的头/尾部弄混了,导致每次分析堆内块的4byte/尾部分的时候就会不自觉想到链表的头/尾部分。明白两者的不同之后,接下来分析宏定义等就显得直观明了了。

 在排查bug的时候,没有实现把指针ptr定位到堆块的头部再进行指针操作确实是害苦了我。刚才就是一个困扰我很久的问题。

说完指针定位到头部的问题后,再来谈谈void *mm_realloc(void *ptr, size_t size)的实现框架。

1、ptr=NULL,size≠0:调用malloc()

2、ptr≠NULL,size=0:调用free()

3、ptr≠NULL,size≠0:调整size为asize

       3.1、asize=blocksize_Cur:直接分配

       3.2、asize<blocksize_Cur:调用place()把当前块进行分割

       3.3、asize>blocksize_Cur:当前块的大小不足以分配

3.3.1、next block空闲,且两个块的大小和blocksize_Sum>aszie:对下一块使用place()

3.3.2、next block已分配,或者两块大小和blocksize_Sum<aszie:调用find_fit()来寻找一个全新的块进行分配,而不考虑当前块和next block

       3.3.2.1、find_fit()找到了合适的块:调用place()进行分配

3.3.2.2、find_fit()未找到合适的块:先用extend_heap()来申请新的堆空间,再调用place(),最后把原来块的内容拷贝到新块里面,释放原来块的空间。

 最后得到的测试结果如图。

//define function
static void *extend_heap(size_t words);
static void *coalesce(void *bp);
static void *find_fit(size_t asize);
static void place(void *bp,size_t asize);

/* single word (4) or double word (8) alignment */
#define ALIGNMENT 8

/* rounds up to the nearest multiple of ALIGNMENT */
#define ALIGN(size) (((size) + (ALIGNMENT-1)) & ~0x7)

#define SIZE_T_SIZE (ALIGN(sizeof(size_t)))

//basic constants and macros
#define WSIZE   4                         //字的大小和首部/脚部的大小
#define DSIZE   8                         //双字的大小
#define CHUNKSIZE (1<<12)              //扩展堆时的默认大小

#define MAX(x,y)    ((x)>(y)?(x):(y))

//pack a size and allocated bit into a word in header/footer 很绕啊
#define PACK(size ,alloc)   ((size)|(alloc))

//read and write a word at address p
#define GET(p)      (*(unsigned int *)(p))
#define PUT(p,val)  (*(unsigned int *)(p)=(val))

//read the size and allocated filds from address p
#define GET_SIZE(p)     (GET(p)&~0x7)
#define GET_ALLOC(p)    (GET(p)&0x1 )

//given block ptr bp,compute address of its header and footer
#define HDRP(bp)    ((char *)(bp)-WSIZE)
#define FTRP(bp)    ((char*)(bp)+GET_SIZE(HDRP(bp))-DSIZE)  //没把bp定位到头部坏大事

//given block ptr bp,computer address of next and previous blocks
#define NEXT_BLKP(bp)   ((char*)(bp)+GET_SIZE( (char*)(bp) - WSIZE) )
#define PREV_BLKP(bp)   ((char*)(bp)-GET_SIZE( (char*)(bp) - DSIZE) )

static void *heap_listp;

static void *extend_heap(size_t words){
    char *bp;
    size_t size;

    //分配偶数字或者进行填充
    size=(words%2)?(words+1)*WSIZE:words*WSIZE;
    if((long)(bp=mem_sbrk(size))==-1)
        return NULL;

    //初始化头部/脚部块和结束块
    PUT(HDRP(bp),PACK(size,0));
    PUT(FTRP(bp),PACK(size,0));
    PUT(HDRP(NEXT_BLKP(bp)),PACK(0,1)); //有点没看懂

    //如果前一个块空闲则合并
    return coalesce(bp);
}

static void *find_fit(size_t asize){
    //h第一次适应算法
    void *bp;

    for(bp=heap_listp;GET_SIZE(HDRP(bp))>0;bp=NEXT_BLKP(bp)){
        if(!GET_ALLOC(HDRP(bp))&&(asize<=GET_SIZE(HDRP(bp))))   //这个块没被分配且容量合适
            return bp;
    }

    return NULL;
}

static void place(void *bp,size_t asize){
    size_t cur_size=GET_SIZE(HDRP(bp));

    if((cur_size-asize)>=(2*WSIZE)){    //给16字节的头部、序言、结尾块腾位置
        PUT(HDRP(bp),PACK(asize,1));
        PUT(FTRP(bp),PACK(asize,1));
        bp=NEXT_BLKP(bp);               //移动到下一个块,就是分割完剩下的部分
        PUT(HDRP(bp),PACK(cur_size-asize,0));
        PUT(FTRP(bp),PACK(cur_size-asize,0));
    }
    else{                               //能用到place说明cur_size-asize>0  直接把这个给分配掉
        PUT(HDRP(bp),PACK(cur_size,1)); //因为剩下的空间也就0、1这两种,但是一个可用块最小为2WSIZE
        PUT(FTRP(bp),PACK(cur_size,1));
    }
}

static void *coalesce(void *bp){
    size_t prev_alloc=GET_ALLOC(FTRP(PREV_BLKP(bp)));
    size_t next_alloc=GET_ALLOC(HDRP(NEXT_BLKP(bp)));
    size_t size=GET_SIZE(HDRP(bp));

    if(prev_alloc&&next_alloc){
        return bp;
    }

    else if(prev_alloc&&!next_alloc){
        size+=GET_SIZE(HDRP(NEXT_BLKP(bp)));
        PUT(HDRP(bp),PACK(size,0));
        PUT(FTRP(bp),PACK(size,0));
    }

    else if(!prev_alloc&&next_alloc){
        size+=GET_SIZE(HDRP(PREV_BLKP(bp)));
        PUT(FTRP(bp),PACK(size,0));
        PUT(HDRP(PREV_BLKP(bp)),PACK(size,0));
        bp=PREV_BLKP(bp);
    }

    else{
        size+=GET_SIZE(HDRP(PREV_BLKP(bp)))+GET_SIZE(FTRP(NEXT_BLKP(bp)));
        PUT(HDRP(PREV_BLKP(bp)),PACK(size,0));
        PUT(FTRP(NEXT_BLKP(bp)),PACK(size,0));
    }

    return bp;
}

/* 
 * mm_init - initialize the malloc package.
 */
int mm_init(void)
{
    //开始创建初始的空堆,大小为4字
    if((heap_listp=mem_sbrk(4*WSIZE))==(void *) -1)
        return -1;
    // return 0;   //牛魔的,怎么这里有个return

    PUT(heap_listp,0);                              //alignment padding
    PUT(heap_listp+(1*WSIZE),PACK(DSIZE,1));        //prologue header
    PUT(heap_listp+(2*WSIZE),PACK(DSIZE,1));        //prologue footer
    PUT(heap_listp+(3*WSIZE),PACK(0,1));            //epologue block
    heap_listp+=(2*WSIZE);
    
    //增加堆的大小
    if(extend_heap(CHUNKSIZE/WSIZE)==NULL)
        return -1;
    return 0;
}



/* 
 * mm_malloc - Allocate a block by incrementing the brk pointer.
 *     Always allocate a block whose size is a multiple of the alignment.
 */
void *mm_malloc(size_t size)
{

    size_t asize;   //adjusted block size
    size_t extendsize;  //如果大小超过堆的大小应该增加的总数
    char* bp;

    if(size==0)
        return NULL;

    //双字对齐
    if(size<=DSIZE)
        asize=2*DSIZE;
    else
        asize=DSIZE*((size+(DSIZE)+(DSIZE-1))/DSIZE);   //加1向下舍入

    //从空闲链表里找合适的块进行分配
    if((bp=find_fit(asize))!=NULL){
        place(bp,asize);
        return bp;
    }

    //如果没有合适的空闲块,堆请求更大的空间
    extendsize=MAX(asize,CHUNKSIZE);
    if((bp=extend_heap(extendsize/WSIZE))==NULL)
        return NULL;

    place(bp,asize);
    return bp;

}

/*
 * mm_free - Freeing a block does nothing.
 */
void mm_free(void *ptr)
{
    size_t size=GET_SIZE(HDRP(ptr));

    PUT(HDRP(ptr),PACK(size,0));
    PUT(FTRP(ptr),PACK(size,0));
    coalesce(ptr);
}


/*
 * mm_realloc - Implemented simply in terms of mm_malloc and mm_free
 */
void *mm_realloc(void *ptr, size_t size)
{
    void *old_ptr=ptr,*next_ptr,*new_ptr;
    size_t asize;
    size_t extendsize;
    size_t blocksize_Cur,blocksize_Next,blocksize_Sum;    //当前块的大小,下一个块的大小

    if(ptr==NULL&&size!=0)
        return mm_malloc(size);
    if(ptr!=NULL&&size==0){
        mm_free(ptr); 
        return NULL;       
    }

    //接下来就是指针不为空,且分配大小非0的正常情况了
    //双字对齐
    if(size<=DSIZE)
        asize=2*DSIZE;
    else
        asize=DSIZE*((size+(DSIZE)+(DSIZE-1))/DSIZE);

    // blocksize_Cur=GET_SIZE(ptr);    //ptr得定位到头部☆☆☆
    blocksize_Cur=GET_SIZE(HDRP(ptr));
    if(asize==blocksize_Cur){

        return ptr;        
    }
    else if(asize<blocksize_Cur){       //当前块的大小>要求分配的空间大小

        place(ptr,asize);
        return ptr;
    }
    else{                               //当前块的大小<要求分配的空间大小

        next_ptr=NEXT_BLKP(ptr);
        blocksize_Next=GET_SIZE(HDRP(next_ptr));
        blocksize_Sum=blocksize_Cur+blocksize_Next;
        if(GET_ALLOC(HDRP(next_ptr))==0&&blocksize_Sum>=asize){  //当前块大小+下一块大小>asize
            PUT(HDRP(ptr),PACK(blocksize_Sum,0));           //把当前块和下一块合并
            place(ptr,asize);
            return ptr;
        }
        else{
            new_ptr=find_fit(asize);
            if(new_ptr==NULL){      //如果当前链表找不到合适的块,则申请额外的空间
                extendsize=MAX(CHUNKSIZE,asize);
                if((new_ptr=extend_heap(extendsize/WSIZE))==NULL)
                    return NULL;
            }
            place(new_ptr,asize);
            memcpy(new_ptr,old_ptr,blocksize_Cur);
            mm_free(old_ptr);
            return new_ptr;
        }
    }
}

显式空闲链表

按照书上思路来写就行

Tip:有char **p1和int **p2,那p1==p2吗?

/* single word (4) or double word (8) alignment */
#define ALIGNMENT 8                     //对齐8个字节(2个字)

/* rounds up to the nearest multiple of ALIGNMENT */
#define ALIGN(size) (((size) + (ALIGNMENT-1)) & ~0x7)
#define SIZE_T_SIZE (ALIGN(sizeof(size_t))) //头部、脚部、两指针、8字节数据

//basic constants and macros
#define WSIZE   4                         //字的大小和首部/脚部的大小
#define DSIZE   8                         //双字的大小
#define CHUNKSIZE   (1<<12)              //扩展堆时的默认大小
#define MINBLOCK    (DSIZE+2*WSIZE+2*WSIZE)

#define MAX(x,y)    ((x)>(y)?(x):(y))

//pack a size and allocated bit into a word in header/footer 很绕啊
#define PACK(size ,alloc)   ((size)|(alloc))

//read and write a word at address p
#define GET(p)      (*(unsigned int *)(p))
#define PUT(p,val)  (*(unsigned int *)(p)=(val))
#define GETADDR(p) (*(unsigned int **)(p)) //读地址p处的一个指针
#define PUTADDR(p,addr) (*(unsigned int **)(p)=(unsigned int *)(addr)) //在地址p处写的指针

//read the size and allocated filds from address p
#define GET_SIZE(p)     (GET(p)&~0x7)
#define GET_ALLOC(p)    (GET(p)&0x1 )

//given block ptr bp,compute address of its header and footer
#define HDRP(bp)    ((char *)(bp)-WSIZE)
#define FTRP(bp)    ((char*)(bp)+GET_SIZE(HDRP(bp))-DSIZE)  //没把bp定位到头部坏大事

//given block ptr bp,computer address of next and previous blocks
#define NEXT_BLKP(bp)   ((char*)(bp)+GET_SIZE( (char*)(bp) - WSIZE) )
#define PREV_BLKP(bp)   ((char*)(bp)-GET_SIZE( (char*)(bp) - DSIZE) )

//链表特有的指针
#define PRED_POINTER(bp)    (bp)    //指向父指针的指针
#define SUCC_POINTER(bp)    ((char*)(bp)+WSIZE) //指向后继的指针

static void *heap_listp;
static void *head_free;

//define function
static void *extend_heap(size_t words);
static void *coalesce(void *bp);
static void *find_fit(size_t asize);
static void place(void *bp,size_t asize);

//链表操作
static void insert_freelist(void *bp);
static void remove_freelist(void *bp);
static void place_freelist(void *bp);

static void *extend_heap(size_t words){
    char *bp;
    size_t size;

    //分配偶数字或者进行填充
    size=(words%2)?(words+1)*WSIZE:words*WSIZE;
    if((long)(bp=mem_sbrk(size))==-1)
        return NULL;

    //初始化头部/脚部块和结束块
    PUT(HDRP(bp),PACK(size,0));
    PUT(FTRP(bp),PACK(size,0));
    PUT(HDRP(NEXT_BLKP(bp)),PACK(0,1)); //有点没看懂

    //如果前一个块空闲则合并
    return coalesce(bp);
}

static void *find_fit(size_t asize){
    //h第一次适应算法
    void *bp;

    for(bp=GETADDR(head_free);bp!=NULL;bp=GETADDR(SUCC_POINTER(bp))){ //遍历空闲链表
        if((asize<=GET_SIZE(HDRP(bp))))   //这个块没被分配且容量合适
            return bp;
    }

    return NULL;
}

static void place(void *bp,size_t asize){
    size_t cur_size=GET_SIZE(HDRP(bp));
    void *next_bp;
    if((cur_size-asize)>=(MINBLOCK)){    //最小块的大小为24B,这里包括了有效载荷的部分
        PUT(HDRP(bp),PACK(asize,1));
        PUT(FTRP(bp),PACK(asize,1));
        next_bp=NEXT_BLKP(bp);               //移动到下一个块,就是分割完剩下的部分
        PUT(HDRP(next_bp),PACK(cur_size-asize,0));
        PUT(FTRP(next_bp),PACK(cur_size-asize,0));
        place_freelist(bp);
    }
    else{                               //能用到place说明cur_size-asize>0  直接把这个给分配掉
        PUT(HDRP(bp),PACK(cur_size,1)); //因为剩下的空间也就0、1这两种,但是一个可用块最小为2WSIZE
        PUT(FTRP(bp),PACK(cur_size,1));
        remove_freelist(bp);
    }
}

static void *coalesce(void *bp){//基本思路没变,加入对空闲链表的操作
    size_t prev_alloc=GET_ALLOC(FTRP(PREV_BLKP(bp)));
    size_t next_alloc=GET_ALLOC(HDRP(NEXT_BLKP(bp)));
    size_t size=GET_SIZE(HDRP(bp));
    char *pre_block,*next_block;

    if(prev_alloc&&next_alloc){
        insert_freelist(bp);
        return bp;
    }

    else if(prev_alloc&&!next_alloc){   //合并下一块
        size+=GET_SIZE(HDRP(NEXT_BLKP(bp)));
        next_block=NEXT_BLKP(bp);   
        remove_freelist(next_block);
        insert_freelist(bp);
    }

    else if(!prev_alloc&&next_alloc){   //合并前一块
        size+=GET_SIZE(HDRP(PREV_BLKP(bp)));
        bp=PREV_BLKP(bp);
        remove_freelist(bp);
        insert_freelist(bp);
    }

    else{   //前后块都合并
        size+=GET_SIZE(HDRP(PREV_BLKP(bp)))+GET_SIZE(FTRP(NEXT_BLKP(bp)));
        pre_block=PREV_BLKP(bp);
        next_block=NEXT_BLKP(bp);
        bp=PREV_BLKP(bp);
        remove_freelist(pre_block);
        remove_freelist(next_block);
        insert_freelist(bp);
    }
    PUT(HDRP(bp),PACK(size,0));
    PUT(FTRP(bp),PACK(size,0));
    return bp;
}


//使用头插法,将空闲块插入空闲链表中
static void insert_freelist(void *bp){  //LIFO,先进后出
    if(GETADDR(head_free)==NULL){
        PUTADDR(SUCC_POINTER(bp),NULL);
        PUTADDR(PRED_POINTER(bp),head_free);
        PUTADDR(head_free,bp);
    }
    else{
        void *tmp;
        tmp=GETADDR(head_free);
        PUTADDR(SUCC_POINTER(bp),tmp);
        PUTADDR(PRED_POINTER(bp),head_free);
        PUTADDR(head_free,bp);
        PUTADDR(PRED_POINTER(tmp),bp);
        tmp=NULL;
    }
}


//将bp指向的空闲块从空闲链表中移除
static void remove_freelist(void *bp){
    void *pred_ptr,*succ_ptr;
    pred_ptr=GETADDR(PRED_POINTER(bp));
    succ_ptr=GETADDR(SUCC_POINTER(bp));
    //处理前驱节点
    if(pred_ptr==head_free){
        PUTADDR(head_free,succ_ptr);
    }
    else{
        PUTADDR(SUCC_POINTER(pred_ptr),succ_ptr);
    }
    //处理后继节点
    if(succ_ptr!=NULL){
        PUTADDR(PRED_POINTER(succ_ptr),pred_ptr);
    }
}

//对空闲链表中的空闲块进行分割
static void place_freelist(void *bp){
    void *pred_ptr,*succ_ptr,*next_bp;
    //储存前后结点的地址
    pred_ptr=GETADDR(PRED_POINTER(bp));
    succ_ptr=GETADDR(SUCC_POINTER(bp));
    next_bp=NEXT_BLKP(bp);
    //处理新的bp,进行前后链接
    PUTADDR(PRED_POINTER(next_bp),pred_ptr);
    PUTADDR(SUCC_POINTER(next_bp),succ_ptr);
    //处理前序节点,针对head_free是前序节点的特殊处理
    if(pred_ptr==head_free){
        PUTADDR(head_free,next_bp);
    }
    else{
        PUTADDR(SUCC_POINTER(pred_ptr),next_bp);
    }
    //处理后序节点
    if(succ_ptr!=NULL){
        PUTADDR(PRED_POINTER(succ_ptr),next_bp);
    }
}
/* 
 * mm_init - initialize the malloc package.
 */
//设立序言块、结尾块,以及序言块前的对齐块(4B),总共需要4个4B的空间
int mm_init(void)
{
    //开始创建初始的空堆,大小为4字
    if((heap_listp=mem_sbrk(4*WSIZE))==(void *) -1)
        return -1;

    PUTADDR(heap_listp,NULL);                       //堆起始位置的对齐块,是bp对齐8字节
    // PUT(heap_listp,0);                              //alignment padding
    PUT(heap_listp+(1*WSIZE),PACK(DSIZE,1));        //prologue header
    PUT(heap_listp+(2*WSIZE),PACK(DSIZE,1));        //prologue footer
    PUT(heap_listp+(3*WSIZE),PACK(0,1));            //epologue block 存疑
    head_free=heap_listp;
    PUTADDR(head_free,NULL);
    heap_listp+=(2*WSIZE);
    
    //增加堆的大小
    if(extend_heap(CHUNKSIZE/WSIZE)==NULL)
        return -1;
    return 0;
}



/* 
 * mm_malloc - Allocate a block by incrementing the brk pointer.
 *     Always allocate a block whose size is a multiple of the alignment.
 */
void *mm_malloc(size_t size)
{

    size_t asize;   //adjusted block size
    size_t extendsize;  //如果大小超过堆的大小应该增加的总数
    char* bp;

    if(size==0)
        return NULL;

    //双字对齐
    if(size<=DSIZE)
        asize=2*DSIZE;
    else
        asize=DSIZE*((size+(DSIZE)+(DSIZE-1))/DSIZE);   //加1向下舍入

    //从空闲链表里找合适的块进行分配
    if((bp=find_fit(asize))!=NULL){
        place(bp,asize);
        return bp;
    }

    //如果没有合适的空闲块,堆请求更大的空间
    extendsize=MAX(asize,CHUNKSIZE);
    if((bp=extend_heap(extendsize/WSIZE))==NULL)
        return NULL;

    place(bp,asize);
    return bp;

}

/*
 * mm_free - Freeing a block does nothing.
 */
void mm_free(void *ptr)
{
    size_t size=GET_SIZE(HDRP(ptr));

    PUT(HDRP(ptr),PACK(size,0));
    PUT(FTRP(ptr),PACK(size,0));
    coalesce(ptr);
}


/*
 * mm_realloc - Implemented simply in terms of mm_malloc and mm_free
 */
void *mm_realloc(void *ptr, size_t size)
{
    void *old_ptr=ptr,*new_ptr;
    size_t asize;
    size_t extendsize;
    size_t blocksize_Cur;    //当前块的大小,下一个块的大小

    if(ptr==NULL&&size!=0)
        return mm_malloc(size);
    if(ptr!=NULL&&size==0){
        mm_free(ptr); 
        return NULL;       
    }

    //接下来就是指针不为空,且分配大小非0的正常情况了
    //双字对齐
    if(size<=DSIZE)
        asize=2*DSIZE;
    else
        asize=DSIZE*((size+(DSIZE)+(DSIZE-1))/DSIZE);

    // blocksize_Cur=GET_SIZE(ptr);    //ptr得定位到头部☆☆☆
    blocksize_Cur=GET_SIZE(HDRP(ptr));
    if(asize==blocksize_Cur){

        return ptr;        
    }
    else if(asize<blocksize_Cur){       //当前块的大小>要求分配的空间大小
        if(blocksize_Cur-asize>=MINBLOCK)
            place(ptr,asize);
        return ptr;
    }
    else{                               //当前块的大小<要求分配的空间大
            new_ptr=find_fit(asize);
            if(new_ptr==NULL){      //如果当前链表找不到合适的块,则申请额外的空间
                extendsize=MAX(CHUNKSIZE,asize);
                if((new_ptr=extend_heap(extendsize/WSIZE))==NULL)
                    return NULL;
            }
            place(new_ptr,asize);
            memcpy(new_ptr,old_ptr,blocksize_Cur-2*WSIZE);
            mm_free(old_ptr);
            return new_ptr;
    }
}

分离空闲链表

对于显式空闲链表,判断节点bp的前驱是否是头结点相当简单,如下

但是对于分离空闲链表来说,就显得比较繁琐了。按照上面的思路,我们得先用一个大循环来遍历24条链表的各个头结点,然后再用上述式子。不如把这个循环遍历做成一个单独的函数,以此来判断前驱是否头结点的问题。

if(isSegList(pred_ptr)){    //如果前驱是头结点

汗流浃背了,代码出了个bug。 

然后是调试环节。

我先是一直在那输入“gdb mm”,结果代码在gdb模式下run起来都有问题,让我心生怀疑可能是调试错了文件,最后才发现pdf上写着调试应该是“gdb mdriver”,白忙活一场。由于默认调试的执行文件是“short1-bal.rep”,想要调试其他文件就得改config.h,但是我发现无论怎么修改config.h,都会报下面的错误。这也是浪费最多时间的一部分。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Matlab进阶绘图第58期—带填充纹理的横向堆叠图

带填充纹理的横向堆叠图是通过在原始横向堆叠图的基础上添加不同的纹理得到的&#xff0c;可以很好地解决由于颜色区分不够而导致的对象识别困难问题。 由于Matlab中未收录提供填充纹理选项&#xff0c;因此需要大家自行设法解决。 本文使用hatchfill2工具&#xff08;Kesh I…

一些智能音箱类的软硬件方案

主要参考资料 Rabbit R1: https://www.rabbit.tech/rabbit-r1 mediatek-helio-p35: https://www.mediatek.com/products/smartphones-2/mediatek-helio-p35 NSdisplay: https://www.nsdisplay.com/ai-holobox-mini/ai-holobox-mini.html RK3566: https://www.rock-chips.com/a/…

I2C总线上拉电阻计算

I2C 总线上拉电阻计算 I2C接口的上拉电阻计算是一个常见问题。本文介绍如何使用简单的方程式进行计算。 1 介绍 I2C通信标准是当今电子系统中应用最广泛的芯片间通信标准。它是一种漏极开路/集电极开路通信标准&#xff0c;这意味着可以连接具有不同电源轨的集成电路 &#…

java jar包后台运行方式

在实际工作中&#xff0c;java开发的spring boot等通过jar包部署需要一直运行的程序部署到服务器上时&#xff0c;都希望后台运行&#xff0c;方便管理程序服务、防止被误操作关闭&#xff0c;本文结合自己工作经验讲解jar包后台运行的两种方式&#xff0c;分别是按操作系统支持…

LeetCode 两两交换链表中的节点

原题链接24. 两两交换链表中的节点 - 力扣&#xff08;LeetCode&#xff09; 思路&#xff0c;请看图片的过程模拟&#xff0c;这里添加了一个哨兵节点0&#xff0c;目的是为了方便操作&#xff0c;得到指向1节点的指针。 class Solution {public:ListNode* swapPairs(ListNod…

17、matlab实现均值滤波、中值滤波、Butterworth滤波和线性相位FIR滤波

1、创建信号 1&#xff09;创建正余弦信号、噪声信号和混合信号 原始正余弦信号公式&#xff1a;Signal1 sin(2*pi*20* t) sin(2*pi*40* t) sin(2*pi*60* t) 高斯分布的白噪声&#xff1a;NoiseGauss [randn(1,2000)] 均匀分布的白噪声&#xff1a;[rand(1,2000)] 正余弦…

AD软件底层丝印反转

快捷键VB&#xff0c;翻转后底部视图所有显示就正常了&#xff0c;当底层确认之后再按VB就回到正常状态。 否则你就看到一个镜像的丝印。 快捷键VB后 注意&#xff0c;经过VB反转BOTTOM后TOP层的丝印变镜像翻转了。 设计完毕后调整过来即可。

物质的量质量,它们可不是一个概念

物质的量&质量&#xff0c;它们可不是一个概念。 物质的量&质量 乍一听物质的量&#xff0c;还以为是和质量有什么关系&#xff0c;是不是&#xff1f;其实物质的量和质量没什么直接的联系。 物质的量是国际单位制中7个基本物理量之一&#xff0c;其符号为n&#xf…

【JAVA SE】抽象类和接口

✨✨欢迎大家来到Celia的博客✨✨ &#x1f389;&#x1f389;创作不易&#xff0c;请点赞关注&#xff0c;多多支持哦&#x1f389;&#x1f389; 所属专栏&#xff1a;JAVA 个人主页&#xff1a;Celias blog~ 目录 引言 一、抽象类 1.1 抽象类的定义 1.2 抽象方法 1.3 抽象…

QT等比例缩放图片

1、这里使用QLabe控件显示图片&#xff0c;如下&#xff1a; label->setPixmap(QPixmap::fromImage(image_2)); 以上是直接原始大小来显示QImage image_2图片。 label->setAlignment(Qt::AlignCenter); 以上代码&#xff0c;是将显示的图片居中展示&#xff0c;默认…

使用git将本地文件上传到仓库+git常用指令

个人主页&#xff1a;Yang-ai-cao 一个小小搬砖人~~~~~ 目录 个人主页&#xff1a;Yang-ai-cao 一个小小搬砖人~~~~~ 配置 基本操作 分支操作 远程仓库 标签 撤销操作 日常操作示例 进阶操作 &#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#…

【python】OpenCV—Merge Image

文章目录 np.hstack / np.vstackSlicecv2.addWeighted自定义渐变式叠加cv2.bitwise_not / cv2.bitwise_and / cv2.add np.hstack / np.vstack 利用 numpy 的 hstack 和 vstack&#xff0c;对图片进行拼接 import cv2 import numpy as nph, w 256,256 img1 cv2.resize(cv2.i…

企业内业务系统与Activiti流程引擎的结合(十一)

摘要:前文分享了企业内部系统集成Activiti的架构和API设计,这里再介绍下 bpmn 流程图的绘制以及与 流程图与bpm后台服务代码的结合点。 一、画流程图 以使用 eclipse 画流程图为例 1. 将 Activiti BPMN 插件安装到 eclipse 插件安装成本后的效果:新建向导中出现 Activiti…

干货分享 | TSMaster 中 Hex 文件编辑器使用详细教程

TSMaster 软件的 Hex 文件编辑器提供了文件处理的功能&#xff0c;这一特性让使用 TSMaster 软件的用户可以更便捷地对 Hex、bin、mot、s19 和 tsbinary 类型的文件进行处理。 本文重点讲述 TSMaster 中 Hex 文件编辑器的使用方法&#xff0c;该编辑器能实现将现有的 Hex、bin、…

Python 的 os 和 shutil 模块

大家好&#xff0c;在日常的编程工作中&#xff0c;处理文件和目录是一个非常常见的任务。无论是创建、复制、移动还是删除文件&#xff0c;这些操作都需要我们与文件系统进行交互。在 Python 中&#xff0c;有两个强大的模块可以帮助我们轻松地进行文件和目录操作&#xff0c;…

怎么判断护眼台灯真的有用吗?揭秘护眼台灯怎么选!

中国目前面临着严重的近视问题&#xff0c;各学段学生的近视率普遍偏高&#xff0c;且高度近视的占比也不容忽视。这不仅对学生的身体健康构成威胁&#xff0c;也对国家的经济社会可持续发展和国家安全构成潜在风险。随着时代的进步和教育的普及&#xff0c;儿童近视的问题已经…

私有仓库搭建

目前市面上比较常见的私有仓库搭建方法为&#xff1a; 通过 Sinopia 或 verdaccio 搭建&#xff08;Sinopia 已经停止维护&#xff0c;verdaccio 是 Fork 自 Sinopia&#xff0c;基本上大同小异&#xff09;&#xff0c;其优点是搭建简单&#xff0c;不需要其他服务。通过 cnp…

ESP32-C3模组上跑通OTA升级(11)

接前一篇文章&#xff1a;ESP32-C3模组上跑通OTA升级&#xff08;10&#xff09; 本文内容参考&#xff1a; 《ESP3-C3 物联网工程开发实战》 乐鑫科技 特此致谢&#xff01; 上一回说到解决了证书验证的问题&#xff0c;但紧接着又出现了一个新的问题&#xff1a;版本相同无…

分享一款开源可自行托管的 Heroku / Netlify / Vercel 替代方案工具,高达16.9K Star

之前分享了一篇文章【AI云部署】盘点开源项目免费云部署的几种技术方案 &#xff0c; 你可能会有这样的一个想法&#xff0c;要不我自建一个云平台自行托管。你看大公司都是自建的服务&#xff0c;托管在外面有隐私安全顾虑。 Coolify 是 Heroku / Netlify / Vercel 等的开源…

零基础入门学习Python第二阶04SQL详解03

MySQL 新特性 JSON类型 很多开发者在使用关系型数据库做数据持久化的时候&#xff0c;常常感到结构化的存储缺乏灵活性&#xff0c;因为必须事先设计好所有的列以及对应的数据类型。在业务发展和变化的过程中&#xff0c;如果需要修改表结构&#xff0c;这绝对是比较麻烦和难…