CSAPP-Lab5 CacheLab解析

news2024/12/23 16:44:05

Review

Cache Struct

  • A cache is a set of 2 s 2^s 2s cache sets
  • A cache set is a set of E cache lines
    • if E=1, it is called “direct-mapped”
  • Each cache line stores a block
  • Total Capacity = S * B * E

image-20221126152039796

由此,我们可以写出cache line和cache的结构:

typedef struct cache_line{
    unsigned valid;
    int tag;
    int lru_cnt;
}cache_line;

cache_line **cache; // cache_line *cache[];

cache[s][e]表示cache的第s组的第e行

Types of Cache Misses

  • Cold (compulsory) miss
    • 一个空的缓存对于任何数据的access都不会命中,此类不命中称为强制性不命中(compulsory miss)或冷命中(cold miss)
  • Conflict miss
    • 因为第k层的capacity一定是小于第k+1层的,根据cache所采用的placement policy可知,第k+1层的block映射到第k层的block时一定会存在重复映射的情况,这种造成的miss称为冲突不命中(conflict miss)
  • Capacity miss
    • 程序通常是按照一系列阶段来运行的,每个阶段访问缓存块的某个相对稳定不变的集合。例如,一个嵌套的循环会反复访问同一个数组的元素。这个块的集合会被称为这个阶段的工作集。当工作集的容量超过缓存的大小时,缓存会经历容量不命中(capacity miss)

Policy

  • Placement policy
    • 只要发生了不命中,第k层的cache就把必须执行某个放置策略(Placement policy),来决定把它从第k+1层取出来的block放置何处。
    • 通常cache是使用的是严格的Placement policy,即由地址中的tag和set index两部分的位模式来决定其放置的位置,set index决定其放于哪个组set,tag决定其放于该set中的哪一行line
  • Replacement policy
    • 如何第k层的缓存已经满了,此时又需要从第k+1层取新的块时,那么就需要决定该替换哪个块,这就又Replacement policy来决定
    • 常见的策略有FIFO、LRU、LFU和随机替换策略,FIFO就是先进来的先出去,LRU则会将最近一段时间未使用的数据evict,LFU将最近一段时间使用次数较少的数据evict,随机替换就会随机evict一个块

cache类型

  1. **直接映射高速缓存(direct-mapped):**E=1,S != 1

    根据[set index]确定数据所要存放的组别set,因为每个set只有一行,所以如果为空,则放,不为空则evict原数据

    该缓存tag位的意义不太,并不需要标记具体的行,但它可以用来记录confict miss

  2. **组相联高速缓存:**E != 1, S != 1

    根据[set index]确定数据所要存放的组别set,因为每个set有E行,如果有空的行,则放进任意空行;如果没有空行,则根据replacement policy来确定被evict的行,

  3. **全相联高速缓存:**E != 1 , S = 1

    该缓存因为S=1,所以对数据的位解释上就没有[set index],只有tag和block offset;因此就直接根据tag来进行match

cache set中的each line并不是按照顺序来存储,而是利用键值对(key,value)来进行match,key是由有效位+tag位组合的

PartA

完成一个模拟cache需要按照以下步骤来:

先贴出全局变量

int hit=0,miss=0,evict=0;
int s,E,b,t;
// 用于提取set tag block offset的掩膜
int cover=1;
int row,col;
  1. 读取s,E,b作为构建cache的三元组

    需要使用 getopt 函数来解析命令行参数,解析过程如下:

    // 所需的头文件
    #include <getopt.h>
    #include <stdlib.h>
    #include <unistd.h>
    
    while((opt = getopt(argc, argv, "s:E:b:t:")) != -1) {
        switch (opt)
        {
            case 's':
                s = atoi(optarg);
                break;
            case 'E':
                E = atoi(optarg);
                break;
            case 'b':
                b = atoi(optarg);
                break;
            case 't':
                pFile =fopen(optarg, "r");
                break;
            default:
                printf("unknown argument\n");
                exit(0);
                break;
        }
    }
    
  2. 构建cache并且初始化

    typedef struct cache_line{
        unsigned valid;
        int tag;
        int lru_cnt;
    }cache_line;
    
    cache_line** create_cache()
    {
        row = pow(2,s);
        col = E;
        cache_line** cache = (cache_line **)malloc(sizeof(cache_line*)*row);
        int i;
        for(i=0; i<row; i++) 
            cache[i] = (cache_line *)malloc(sizeof(cache_line)*col);
        return cache;
    }
    
    void initial(cache_line** cache)
    {
        t = 64-b-s;
        for(int i=0; i<(b+s-1); i++) {
            cover<<=1;
            cover+=1;
        } 
        for(int i=0; i<row; i++)
            for(int j=0; j<col; j++) {
                cache[i][j].valid = 0;
                cache[i][j].tag = 0;
                cache[i][j].lru_cnt = 0;
            }
    }
    
  3. 读取valgrind生成的trace文件,并且一条条执行命令,有三种命令

    • M(Modify):写分配+写回,先从cache中读,再写
    • L(Load):读取,从cache读数据,没有读到就直接写(这个store就一样了)
    • S(Store):存储,向cache中写数据

    因为并没有涉及到真正的数据写或存储,并不关心数据是什么,只关注与读或者写时的miss,hint和evict情况。

    我们写一个load函数,其作用就是从cache去读数据(Load),如果没有就写(Store),所以M命令对应2次load函数,L命令对应一次load函数,S命令也对应一次load函数

    char identifier;
    unsigned address;
    int size;
    int cnt=1; 	// 用cnt记录数据最后的访问时间,越小表示上次访问已经过去很久了
    while(fscanf(pFile, "%c %x,%d",&identifier, &address, &size) > 0)
    {
        switch (identifier) 
        {
            case 'L':
                load(cache,address,size,cnt);
                break;
            case 'S':
                load(cache,address,size,cnt);
                break;
            case 'M':
                load(cache,address,size,cnt);
                load(cache,address,size,cnt);
                break;
            default:
                break;
        }
        cnt++;
    }
    

    load函数的实现如下:

    void load(cache_line** cache, unsigned address, int size, int _time)
    {
        int tag = address>>(b+s);
        int set = (address&cover)>>b;
        for(int j=0; j<col; j++) {
            if(cache[set][j].valid && (cache[set][j].tag == tag)) {
                hit++;
                cache[set][j].lru_cnt = _time;
                return;
            }
            if(cache[set][j].valid == 0) {
                miss++;
                cache[set][j].valid = 1;
                cache[set][j].tag = tag;
                cache[set][j].lru_cnt = _time;
                return;
            }
        }
        // 没有找到并且set内的所有line都满了
        // 找到没有使用时间最长的数据 将它evict掉
        int index=0;
        for(int j=0; j<col; j++) {
            if(cache[set][j].lru_cnt < cache[set][index].lru_cnt)
                index = j;
        }
        cache[set][index].tag = tag;
        cache[set][index].lru_cnt = _time;
        evict++;
        miss++;
        return;
    }
    
  4. 输出记录的hint,miss和evict,并且释放cache和打开文件的的空间

printSummary(hit, miss, evict);
fclose(pFile);
free_cache(cache);

void free_cache(cache_line** cache)
{
    int row = pow(2,s);
    for(int i=0; i<row; i++)
        free(cache[i]);
}

完整的代码可以从我的github仓库获取

Part B

我们所写的转置代码在参考cache下运行(s = 5, E = 1, b =5)

实验讲义中允许我们可以针对特定形状的矩阵写出具有针对性的代码来降低miss率

我们先以矩阵乘法为例,讨论矩阵分块是如何对降低cache miss率的:

  1. **分块乘法的正确性:**对于矩阵C=矩阵A*矩阵B,三个矩阵都是32x32,我们可以通过将每个矩阵分成16x16的四个小矩阵:

    image-20221217234614336

    C0中的元素C[0][0]仍然等于A0中的蓝块*B0中的蓝块 + A1中的绿块 *B2中的绿块;这等价于A0与A1组成的一行乘以B0和B2组成的一列

    分块矩阵乘法的代码:

    image-20221217180728223

    这段代码是由2个三重循环组成的6重循环,外三层是整体矩阵乘法i,j索引矩阵C的每个分块,k索引矩阵A和矩阵B相对应的分块组,例如A0和B0,A1和B2;内三层循环是分块内部的矩阵乘法,即在这部分落实每个元素的运算,i1,j1索引矩阵C当前分块的单个元素,k1索引k索引的分块组中相对应的元素。

  2. **分块矩阵对命中率的提升:**我们假设矩阵的大小为n * n,cache block = 8。

    • 不分块:再假设cache的容量C远小于n,也就是说整个cache连矩阵的一行都装不下,每一次重新读一行必定miss

      对于矩阵A来说,是行读取,步数为1,每个block的第一个元素会miss,后面7个都会hint,即每8个元素miss一次,一行n个元素,miss n / 8 n/8 n/8 次;

      对于矩阵B来说,是列读取,步数为n,因为 C < < n C<<n C<<n,即使用完cache的每个组后,也不会读完所有行,然后新的行又会覆盖原来的block,所以矩阵B每次读都不会命中,一列n个元素,miss n n n 次;即对计算矩阵C的一个元素总共miss ( 9 n ) / 8 (9n)/8 (9n)/8,一共 n 2 n^2 n2个元素,所以一共miss ( 9 n 3 ) / 8 (9n^3)/8 (9n3)/8。(注意这里咱不考虑A和B的重复覆盖和读取C的miss,下面分块也是)

      image-20221217232243648

    • 分块:再假设分块的大小B * B(B<=8),cache的容量C远小于n,但是能容下3个分块大小, 3 B 2 < C 3B^2<C 3B2<C,可以放下A,B,C三个矩阵的方块

      对于矩阵A的方块来说,步数为1已经为最佳状态,还是每个block的第一个元素会miss,分块内一行B个元素,所以一行miss B / 8 B/8 B/8 次,一个分块B行,即一个分块miss B 2 / 8 B^2/8 B2/8次;

      对于矩阵B来说,因为cache可以放下完整的一个分块,所以不会存在读取新的一行时覆盖原来的block的这种浪费情况,也就是说矩阵分块解决了矩阵内部相互覆盖的问题,在读完第一列都不命中后,读取第二列,第三列…到第B列,每个元素都已经缓存在cache中,都会hint,即整个分块的miss次数为 B 2 / 8 B^2/8 B2/8次!矩阵B和矩阵A在分块后每块的miss率一致

      计算矩阵C的一个分块,需要读取 n / B n/B n/B个矩阵A分块和矩阵B分块,即计算矩阵C一个分块miss 2 n B ∗ B 2 8 = n B 4 \frac{2n}{B} * \frac{B^2}{8} = \frac{nB}{4} B2n8B2=4nB次。矩阵C共有 n B 2 \frac{n}{B}^2 Bn2个分块,即总miss次数为:

      n B 4 × n B 2 = 1 4 B n 2 \frac{nB}{4}\times \frac{n}{B}^2=\frac{1}{4B}n^2 4nB×Bn2=4B1n2

      image-20221217232321813

    经过上面的分析得知,分块对提升命中率的几点信息:

    1. 3 B 2 < C 3B^2 < C 3B2<C,即cache的容量需要放下3个分块的大小
    2. 分块对矩阵乘法命中率的提升是通过降低矩阵B的miss次数达到的

在介绍完矩阵分块乘法的前置知识后,回到本次lab的partB部分:

根据给出的cache参数,s=5,E=1,b=5,可以得知该cache是一个有 2 5 = 32 2^5=32 25=32个组,每组只有一行,每一行的block大小为 2 5 = 32 2^5=32 25=32个字节,可以存下8个int的直接映射高速缓存:

image-20221217233751351

partB不是提高矩阵乘法的命中率,而是提高矩阵转置的命中率,即只有两个矩阵A和B参与。
我们假设是将矩阵A进行转置,存储到矩阵B上:

  • 矩阵转置下不分块:和矩阵乘法部分一样,矩阵A一行不命中 n 8 \frac{n}{8} 8n次,矩阵A的一行对应于目标矩阵B的一列(这里我们考虑读取目标矩阵的miss),矩阵B一列全不命中,为 n n n次,所以转置矩阵A的一行不命中 9 n 8 \frac{9n}{8} 89n次,转置整个矩阵不命中 9 n 8 × n = 9 8 n 2 \frac{9n}{8} \times n = \frac{9}{8}n^2 89n×n=89n2次。
  • 矩阵转置分块:和矩阵乘法部分一样,矩阵A和矩阵B的不命中次数一样,都是 B 2 8 \frac{B^2}{8} 8B2,即转置一个方块的的不命中次数为 B 2 4 \frac{B^2}{4} 4B2,总共有 n B 2 \frac{n}{B}^2 Bn2个方块,即转置整个矩阵,总miss次数为 1 4 n 2 \frac{1}{4}n^2 41n2(这个结论我不敢保证其正确性,我只是需要算出一共理论最少的miss次数,供优化参考,如果有正确的值如果能告知我我将万分感激)。

trans.c中给出了暴力解法的代码,可以供我们作为对比进行优化:

void trans(int M, int N, int A[N][M], int B[M][N])
{
    int i, j, tmp;
    for (i = 0; i < N; i++) {
        for (j = 0; j < M; j++) {
            tmp = A[i][j];
            B[j][i] = tmp;
        }
    }    
}

下面是实验要求,Let’s go!

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-JCjzOfg2-1671357441384)(null)]

32 x 32

对于分块优化命中率,最重要的问题就是确定分块Block的大小,却决于3个size:

  1. cache block size(8个int)
  2. cache size(32x8个int)
  3. input cache size (32x32)

分块大小B设置的太大,如果超过了cache block的大小,那么对列模式的矩阵B来说miss,并且我们的缓存只够存矩阵的8行,如果分块超过8就意味着矩阵内的重叠;设置的太小,也会增加矩阵B的miss次数,缓存的资源被浪费。对于32x32,我们对其分成16个8x8的分块,转置代码如下:

void transpose_submit(int M, int N, int A[N][M], int B[N][M])
{
	for(int i=0; i<N; i+=8)
        for(int j=0; j<M; j+=8)
            for(int i1=i; i1<i+8;i1++)
                for(int j1=j; j1<j+8; j1++)
                    B[j1][i1] = A[i1][j1];
}

得到的结果如下:

image-20221218005026231

miss次数为343次,没有拿到满分,根据我们之前推到的公式来看,32x32矩阵,分块后的最少不命中次数应该是 1 4 × 3 2 2 = 256 \frac{1}{4} \times 32^2 = 256 41×322=256次,很明显还有我们没有考虑到的地方有多余的miss。

我们在计算公式时,对其进行了理想化处理,即不考虑矩阵A和矩阵B的重叠问题。但在实际情况中这是大量存在的,对于矩阵的大小是缓存大小的倍数时,两个相同大小的矩阵,在缓存进cache时相同位置是会产生冲突的。例如本例中,32x32矩阵是32x8大小的cache的4倍。

对于矩阵转置的情况下,除了对角线上的块,其他的块不会出现重叠的部分。

本例中,各个元素在缓存中的映射关系如下(图源):

img

矩阵转置下的冲突情况:

img

A中标红的块占用的是缓存的第 0,4,8,12,16,20,24,28组,而B中标红的块占用的是缓存的第2,6,10,14,18,16,30组,刚好不会冲突。

那么我们该如何解决对角线上的冲突?lab讲义中提到我们可以使用12个自定义变量,这个工具不能忘了,也就是说我们直接可以把A的每一行转置的元素转从到8个变量里,再存在B中,这就避免了对角线上的冲突。

代码如下:

for(int i=0; i<32; i+=8)
    for(int j=0; j<32; j+=8)
        for(int i1=i; i1<i+8; i1++)
        {
            int a1 = A[i1][j];
            int a2 = A[i1][j+1];
            int a3 = A[i1][j+2];
            int a4 = A[i1][j+3];
            int a5 = A[i1][j+4];
            int a6 = A[i1][j+5];
            int a7 = A[i1][j+6];
            int a8 = A[i1][j+7];

            B[j][i1] = a1;
            B[j+1][i1] = a2;
            B[j+2][i1] = a3;
            B[j+3][i1] = a4;
            B[j+4][i1] = a5;
            B[j+5][i1] = a6;
            B[j+6][i1] = a7;
            B[j+7][i1] = a8;
        }

结果如下:

image-20221218012357189

64x64

根据公式所得最小不命中次数为 1 4 × 6 4 2 = 1024 \frac{1}{4}\times 64^2 = 1024 41×642=1024次,我们讨论分块大小该怎么设置:变大后的矩阵,原来的缓存只够存下4行,如果分块超过4行,那么矩阵内部就会出现重叠。那我们先采取4x4的分块,并且进行对角线优化,代码如下:

for(int i=0; i<64; i+=4)
    for(int j=0; j<64; j+=4)
        for(int i1=i; i1<i+4; i1++)
        {
            int a1 = A[i1][j];
            int a2 = A[i1][j+1];
            int a3 = A[i1][j+2];
            int a4 = A[i1][j+3];


            B[j][i1] = a1;
            B[j+1][i1] = a2;
            B[j+2][i1] = a3;
            B[j+3][i1] = a4;
        }

得到的结果如下:

image-20221218014144790

1699次,还没有到达满分,但相比不分块的4723次已经有很大的提升了!

如果进一步缩小分块大小,显然会出现资源浪费导致的miss次数上升,而增大分块大小,也会因矩阵内重叠导致miss上升,但是放大比缩小有更多的操作空间,还可以分块内再分块。

我们考虑将分块大小增大到8x8,再在每个8x8的分块内分成4个4x4个分块,这样就能避免因为因为块增大而带来的矩阵内重叠。对于4x4分块带来的问题是缓存利用率不高,为此我们需要用点技巧来提高其利用率:

读取矩阵A时,会缓存A0、A1两个4x4的分块,如果只是单纯的4x4分块中,A1的缓存是会被浪费的,在这里为了充分利用缓存,我们把A1也存到矩阵B中相同位置上,但我们知道这个位置是错误的,所以用本地变量把此时矩阵B分块的右上角的4x4分块存起来。

image-20221218135845257

在写完矩阵B分块第一行后:

  1. 此时cache内缓存着矩阵B分块的第一行,我们利用缓存将矩阵B右上角写入本地变量中

  2. 此时我们读矩阵A的A2和A3(分块第二行)时,不会和矩阵B分块的第一行冲突覆盖。将A2块,复制到矩阵B分块的右上角,并且顺便将A3块存进本地变量。

  3. 在访问矩阵B左下角时,把之前存储的本地变量中的A1和A3复制进去。

    这样的顺序才能达到缓存的最大利用效果,示意图如下:

image-20221218155611293

转置代码如下:

int a1,a2,a3,a4,a5,a6,a7,a8,a9,a10,a11,a12;
// 遍历矩阵A的每个8x8的分块
for(int i=0; i<64; i+=8)
    for(int j=0; j<64; j+=8){
        // Step1:将A0 A1转给B0 B1
        for(int k=i; k<i+4; k++){
            a1 = A[k][j];
            a2 = A[k][j+1];
            a3 = A[k][j+2];
            a4 = A[k][j+3];
            a5 = A[k][j+4];
            a6 = A[k][j+5];
            a7 = A[k][j+6];
            a8 = A[k][j+7];

            B[j][k] = a1;
            B[j+1][k] = a2;
            B[j+2][k] = a3;
            B[j+3][k] = a4;
            B[j+0][k+4] = a5;
            B[j+1][k+4] = a6;
            B[j+2][k+4] = a7;
            B[j+3][k+4] = a8;
        }
        for(int k=j; k<j+4; k++){
            // 复制矩阵B右上角分块到local
            a1 = B[k][i+4];
            a2 = B[k][i+5];
            a3 = B[k][i+6];
            a4 = B[k][i+7];
            // 把矩阵A左下角分块A2复制到矩阵B右上角分块
            a5 = A[i+4][k];
            a6 = A[i+5][k];
            a7 = A[i+6][k];
            a8 = A[i+7][k];
			// 顺便将矩阵A右下角存进本地变量
            a9 = A[i+4][k+4];
            a10 = A[i+5][k+4];
            a11 = A[i+6][k+4];
            a12 = A[i+7][k+4];

            B[k][i+4] = a5;
            B[k][i+5] = a6;
            B[k][i+6] = a7;
            B[k][i+7] = a8;
            // 把local复制到矩阵B分块的第二行
            B[k+4][i] = a1;
            B[k+4][i+1] = a2;
            B[k+4][i+2] = a3;
            B[k+4][i+3] = a4;		
            B[k+4][i+4] = a9;
            B[k+4][i+5] = a10;
            B[k+4][i+6] = a11;
            B[k+4][i+7] = a12;
        }
    }

测试结果如下:

image-20221218160112559

1155次,成功!

61x67

对于该不规则的矩阵,需要留意在两层内层循环中多加上一个判断条件:

  • i1<N
  • j1<M

这样就能把整个矩阵每个元素都清理一遍,lab对这个矩阵的miss次数要求比较低,通过一个16x16分块就能直接通过,代码如下:

for(int i=0; i<N; i+=16)
    for(int j=0; j<M; j+=16)
        for(int i1=i; i1<i+16 && i1<N; i1++)
            for(int j1=j; j1<j+16 && j1<M; j1++)
                B[j1][i1] = A[i1][j1];

测试结果如下:

image-20221218165046505

1992次,成功!

Summary

到此,整个实验都已完成!

image-20221218170027441

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

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

相关文章

微服务框架 SpringCloud微服务架构 服务异步通讯 50 消息可靠性 50.4 失败重试机制 50.4.1 消费者失败重试

微服务框架 【SpringCloudRabbitMQDockerRedis搜索分布式&#xff0c;系统详解springcloud微服务技术栈课程|黑马程序员Java微服务】 服务异步通讯 文章目录微服务框架服务异步通讯50 消息可靠性50.4 失败重试机制50.4.1 消费者失败重试50 消息可靠性 50.4 失败重试机制 50…

可路由计算引擎实现前置数据库

❤️作者主页&#xff1a;小虚竹 ❤️作者简介&#xff1a;大家好,我是小虚竹。Java领域优质创作者&#x1f3c6;&#xff0c;CSDN博客专家&#x1f3c6;&#xff0c;华为云享专家&#x1f3c6;&#xff0c;掘金年度人气作者&#x1f3c6;&#xff0c;阿里云专家博主&#x1f3…

非零基础自学Golang 第9章 结构体 9.1 理解结构体 9.2 定义结构体 9.3 实例化结构体

非零基础自学Golang 文章目录非零基础自学Golang第9章 结构体9.1 理解结构体9.2 定义结构体9.3 实例化结构体9.3.1 标准实例化9.3.2 new函数实例化9.3.3 取地址实例化9.3.4 访问成员第9章 结构体 在Go语言官网的常见问题解答一栏中&#xff0c;有这样一个问题&#xff1a;“Is…

Fiddler(7):fiddler设置弱网测试

弱网测试 概念&#xff1a;弱网看字面意思就是网络比较弱&#xff0c;我们通称为信号差&#xff0c;网速慢。 意义&#xff1a;模拟在地铁、隧道、电梯和车库等场景下使用APP &#xff0c;网络会出现延时、中断和超时等情况。 Fiddler弱网测试流程&#xff1a; 一、限速操作…

基于云开发的微信小程序、QQ小程序详细教程(更新中)

云开发解决方案 小程序云开发解决方案 为企业和开发者提供一站式后端云服务&#xff0c;无需管理基础架构&#xff0c;一次开发多端运行&#xff0c;腾讯云和微信联合出品。 云开发&#xff08;Tencent CloudBase&#xff0c;TCB&#xff09;是腾讯云提供的云原生一体化开发环…

数据结构---大整数相加

大整数相加列竖式运算第一步第二步第三步第四步JAVA实现给出两个很大的整数&#xff0c;要求实现程序求出两个整数之和。 记得这个题目我大二ACM课程老师讲过&#xff0c;但是忘记了。。。。。。。。。。 列竖式运算 程序不可能通过一条指令计算出两个大整数之和&#xff0c;…

【学习打卡04】可解释机器学习笔记之Grad-CAM

可解释机器学习笔记之Grad-CAM 文章目录可解释机器学习笔记之Grad-CAMGrad-CAM介绍Grad-CAM是CAM的泛化形式Gard-CAM可视化结果Grad-CAM算法的优点Grad-CAM算法的缺点Grad-CAM变种Grad-CAMScore-CAMLayer-CAM思考与总结参考阅读首先非常感谢同济子豪兄拍摄的可解释机器学习公开…

SAP S4 FICO 固定资产模块后台配置详解

1. 概述 资产会计&#xff08;FI-AA&#xff09;作为总帐模块&#xff08;FI-GL&#xff09;的子分类帐&#xff0c;对企业的固定资产、无形资产、在 建工程、低值易耗品、长期待摊、从购置、资本化、折旧、调拨到出售/报废的整个生命周期 进行全过程的管理&#xff0c;并和…

java实现给微信群中定时推送消息

大家好&#xff0c;我是雄雄。 前言 上一篇&#xff0c;我们介绍了如何通过调用接口的方式&#xff0c;将每日新闻发送到自己的博客中。我们会发现&#xff0c;将新闻以文章的形式发布&#xff0c;并且相关内容按照markdown的形式进行格式调整&#xff0c;有需要的可以点击这里…

获取小程序生产、开发、体验等环境、版本信息、appid等信息

if (typeof __wxConfig "object"){const version __wxConfig.envVersion;console.log("当前环境:" version)if (version "develop"){cosole.log(测试 开发环境)}else if (version "trial"){cosole.log(测试 体验版)}else if (versio…

【电力系统】基于YALMIP+CPLEX求解带储能的微电网优化调度问题附matlab代码

✅作者简介&#xff1a;热爱科研的Matlab仿真开发者&#xff0c;修心和技术同步精进&#xff0c;matlab项目合作可私信。 &#x1f34e;个人主页&#xff1a;Matlab科研工作室 &#x1f34a;个人信条&#xff1a;格物致知。 更多Matlab仿真内容点击&#x1f447; 智能优化算法 …

【树莓派不吃灰】兄弟连篇② 文件编辑器vim与RPM软件包管理、yum

目录1、文件编辑器Vim1.1 工作模式1.2 插入命令1.3 定位命令1.4 删除命令1.5 复制和剪切命令1.6 替换和取消命令1.7 搜索和搜索替换命令1.8 保存和退出命令1.9 使用技巧2、yum包管理2.1 网络yum源2.2 yum命令2.3 rpm包与源码包安装位置❤️ 博客主页 单片机菜鸟哥&#xff0c;一…

联盛德W801开发板流水灯示例

联盛德W801开发板流水灯示例&#x1f4fa;&#x1f4fd;&#x1f39e;&#x1f3ac;本实验通过利用开发板中间的7个led&#xff0c;控制对应的IO实现流水灯效果。 &#x1f4dc;W801 32位WiFi蓝牙双模SoC开发板 &#x1f6e0;开发工具以及示例Demo工程 参考&#xff1a;《W801…

【数学模型】基于ARMR模型模拟风速附matlab完整代码

✅作者简介&#xff1a;热爱科研的Matlab仿真开发者&#xff0c;修心和技术同步精进&#xff0c;matlab项目合作可私信。 &#x1f34e;个人主页&#xff1a;Matlab科研工作室 &#x1f34a;个人信条&#xff1a;格物致知。 更多Matlab仿真内容点击&#x1f447; 智能优化算法 …

图计算的学习与思考

好的软件不是靠程序分析、查错查出来的&#xff0c;而是由正确的人构建出来的。图成为日益重要的运算对象&#xff0c;图结构是对群体关系的一种抽象&#xff0c;可以描述丰富的对象和关系。图计算的核心是如何将数据建模为图结构以及如何将问题的解法转化为图结构上的计算问题…

MySQL高级【存储引擎】

目录 1&#xff1a;Mysql体系结构图&#xff1a; 2&#xff1a;存储引擎简介 3&#xff1a;存储引擎特点 InnoDB存储引擎&#xff1a; MyISAM存储引擎&#xff1a; Memory存储引擎&#xff1a; 4&#xff1a;存储引擎选择 5&#xff1a;存储引擎小结 1&#xff1a;Mysq…

.net core 在arm linux 上运行

一、环境 开发板&#xff1a;OKMX6ULL 硬件&#xff1a;cortex A7 软件环境&#xff1a;Linux version 4.1.15-00041-g92e8c0f (zsdeveloper-RH2485-V2) (gcc version 5.3.0 (GCC) ) #1 SMP PREEMPT Mon May 24 17:20:23 CST 2021 .net core 版本&#xff1a; 二、问题 1…

网络攻防中的支持多种社交媒体的网络 go-fish 工具(包含智能探测、用户查找、权限提升等等),需谨慎使用,不然要被请去喝茶了

网络攻防中的支持多种社交媒体的网络 go-fish 工具(包含智能探测、用户查找、权限提升等等),需谨慎使用,不然要被请去喝茶了。 ShellPhish是一个可定制的网络 go-fish 工具,基于SocialFish。 支持以下社交平台: Instagram Facebook Twitter Snapchat Github Yahoo…

Linux学习笔记——Linux基础命令

02、Linux基础命令 2.1、Linux的目录结构 1、学习目标 掌握Linux系统的目录结构掌握Linux系统的路径表达式 2、Linux的目录结构是一个树型结构 Windows系统可以拥有多个盘符&#xff0c;如C盘、D盘、E盘 Linux没有盘符这个概念&#xff0c;只有一个根目录/&#xff0c;所有…

【大数据趋势】趋势在发生变化,稍安勿躁。

昔日(12月以前)行情核心源头 : 纳斯达克指数主连日线&#xff0c;趋势预测准确&#xff0c;继续筑底可能中 回顾一下上周大数据预测的趋势&#xff0c;指数用上引线方式神奇完成目标 【12.3趋势模拟情况】 预计&#xff1a;趋势不变情况下&#xff0c;预测反弹到黄金分割线…