(c语言进阶)指针的进阶

news2024/10/1 2:19:06

一.字符指针

 1.一般应用

(1)%c的应用

(2)%s的应用

                                字符指针没有权限通过解引用去改变指针指向的值 

  

2.笔试题 

 题目:判断输出结果

int main()
{
    const char* p1 = "abcdef";
    const char* p2 = "abcdef";

    char arr1[] = "abcdef";
    char arr2[] = "abcdef";

    if (p1 == p2)
    {
        printf("p1==p2\n");
    }
    else
    {
        printf("p1!=p2\n");
    }

    if (arr1 == arr2)
    {
        printf("arr1==arr2\n");
    }
    else
    {
        printf("arr1!=arr2\n");
    }
    return 0;
}

解析: 

#include<stdio.h>
int main()
{
    const char* p1 = "abcdef";
    const char* p2 = "abcdef";
//常量字符串"zbcdef"被存储在只读区,指针p1、p2存储了这个字符串的首元素地址
    char arr1[] = "abcdef";
    char arr2[] = "abcdef";
//数组arr1、arr2各开辟了一块内存空间,将字符串"zbcdef"存储在开辟的内存中
    if (p1 == p2)
    {
        printf("p1==p2\n");
    }
    else
    {
        printf("p1!=p2\n");
    }
//因为p1和p2都是字符串"abcdef"的地址,是相同的
    if (arr1 == arr2)
    {
        printf("arr1==arr2\n");
    }
    else
    {
        printf("arr1!=arr2\n");
    }
//arr1和arr2为各自开辟的内存空间中,字符串"abcdef"首元素的地址
//两个内存空间不同,所以这两个地址是不同的

    return 0;
}

二.指针数组——元素为指针的数组

1.指针数组的类型 

2.指针数组的应用 

 

三.数组指针——指向数组的指针

1.数组指针的定义

  

2.数组指针与指针数组的区别 

#include<stdio.h>
int main()
{
    //数组的创建
    int arr1[5] = {1,2,3,4,5};
    int arr2[5] = {2,3,4,5,6};
    int arr3[5]= {3,4,5,6,7};
    //指针数组
    //parr是一个空间为3的数组,其中存储的元素为int*类型的
    //parr的类型为int*[3]

    int* parr[3] = {arr1,arr2,arr3};
    //数组指针
    //p为一个指针,其指向的元素为int[5]类型的
    //为了体现p是一个指针,故将*与p用()圈起来
    //p的类型为int(*)[5]

    int(*p)[5] = &arr1;
    int(*p)[5] = &arr2;
    int(*p)[5] = &arr3;
//——————————————————————————————————————————————————
    //练习
    int* (*pp)[3] = &parr;
    //pp为一个指针,其指向的元素为int*[3]类型的
    //为了体现pp是一个指针,故将*与pp用()圈起来
    //p的类型为int*(*)[3]

    return 0;
}

3.数组名的相关应用 

4.数组指针的应用 

例题:

#include<stdio.h>
void printf1(int arr[3][5],int h,int l)    //*(*(arr+H)+L)==arr[H][L],故用数组传参和指针传参本质相同
{
    int H,L;
    for (H = 0; H < h; H++)
    {
        for (L = 0; L < l; L++)
        {
            printf("%d ",arr[H][L]);
        }
        printf("\n");
    }
}
void printf2(int(*p)[5],int h,int l)    //二重指针首元素的地址为第一行元素
{
    int H,L;
    for (H = 0; H < h; H++)
    {
        for (L = 0; L < l; L++)
        {
            printf("%d ",*(*(p+H)+L));
          //printf("%d ",p[H][L]);      相同
        }
        printf("\n");
    }
}
//若设arr数组的第一行元素为一维数组brr,数组指针p解引用后为整个数组brr,brr为数组名,相当于首元素地址。
//所以数组指针p解引用后为brr的首个元素,故*(p+H)会依次获取每一行一维数组的首元素地址
//*(p+H)+L会依次获取每一行中的每一个元素的地址,故*(*(p+H)+L)会获取每一行中的每一个元素

int main()
{
    int arr[3][5] = {1,2,3,4,5,2,3,4,5,6,3,4,5,6,7};
    int h = 3;    //行
    int l = 5;    //列
    //数组传参输出
    printf1(arr,h,l);
    printf("\n");
    //指针传参输出
    printf2(arr,h,l);
    return 0;
}

输出结果: 

5.存放数组指针的数组 

 #include<stdio.h>
int main()
{
    int arr1[] = {1,2,3,4};
    int arr2[] = {2,3,4,5};
    int arr3[] = {3,4,5,6};
    int(*p1)[4] = &arr1;
    int(*p2)[4] = &arr2;
    int(*p3)[4] = &arr3;
    //数组指针不能没有元素个数,定义数组的时候可以没有
    int(*p[3])[4] = {&arr1,&arr2,&arr3};
    //存放数组指针的数组,其中元素指向的元素类型为int[4]
    //其类型为int(*[3])[4]

    return 0;
}

四.数组传参和指针传参

1.一维数组传参

#include<stdio.h>

void test1(int arr1[])
{}
void test1(int arr1[10])    //形参数组的元素可有可无
{}
void test1(int* arr1)    //接收首元素地址
{}

void test2(int* arr2[])
{}
void test2(int* arr2[20])    //形参数组的元素可有可无
{}
void test2(int** arr2)    //接收首元素地址
{}
int main()
{
    int arr1[10] = {0};
    int* arr2[20] = {0};

    test1(arr1);
    test2(arr2);
    return 0;
}

2.二维数组传参

#include<stdio.h>
void test(int arr[][5])    //二维数组的行可以省略,列不能省略
{}
void test(int arr[3][5])    
{}
void test(int(*arr)[5])    //接收首元素地址-->一个一维数组的地址
{}
int main()
{
    int arr[3][5] = {0};
    test(arr);
    return 0;

3、一级指针传参

#include<stdio.h>
void print(int* p)    //若实参的本质是一个一级整形指针就可以传参
{}
int main()
{
    int a = 10;
    int* ptr = &a;
    int arr[10];

    print(&a);
    print(ptr);
    print(arr);
    return 0;
}

4.二级指针传参

#include<stdio.h>
void test(int** p)    //若实参的本质是一个二级整形指针就可以传参
{}
int main()
{
    int* p1;
    int** p2;
    int* arr[10];  //指针数组

    test(&p1);
    test(p2);
    test(arr);
    return 0;
}

五.函数指针

1.函数指针的定义

#include<stdio.h>
int Add(int x, int y)
{
    return x + y;
}
int main()
{
    printf("%p\n",&Add);  //00007FF6991B13E8
    printf("%p\n",Add);  //00007FF6991B13E8
    //&函数名—>取出的是函数的地址
    //对于函数来说,&函数名和函数名都是函数的地址

    int (*p)(int, int) = &Add;
  //int p (int,int) = Add;
    int ret = (*p)(2, 3);
  //int ret = p (2,3);
    //对p解引用后得到函数Add,故(*p)(2, 3)==Add(2,3)

    printf("%d ",ret);
    return 0;

2.函数指针的应用 

 #include<stdio.h>
int Add(int x, int y)
{
    return x + y;
}
void calc(int(*p)(int, int))
{
    int a = 3;
    int b = 5;
    int ret = p(a, b);
    printf("%d\n",ret);
}
int main()
{
    calc(Add);    
    //将函数Add传递给函数calc,使其在函数内用指针调用Add函数
    return 0;
}

3.特殊的用法——来自《c的陷阱与缺陷》 

(1)例一 

int main()
{
    //求解如下代码的含义
    (*(void(*)())0)();
  //    void(*)()         返回值为void类型,无参的函数指针
  //  (             )0       将0强制类型转换为函数指针类型
  //(*                 )()    将函数指针类型的0解引用,调用指向地址为0处的函数

 //综上可知:这段代码的含义为一次函数调用
    return 0;
}

(2)例二 

 int main()
{
    //求解如下代码的含义
    void(*signal(int,void(*)(int)))(int);
//           signal(int,void(*)(int))        接收以整形和数组指针为参数的函数的函数指针
//  void(*                                 )(int)  解引用得到函数signal
// 最后得到:void signal (int)——为一次函数声明

    return 0;
}

4.函数指针的应用

 #include<stdio.h>    //简易计算器
void menu()
{
    //add—>加       sub—>减       mul—>乘        div—>除
    printf("*********************************************\n");
    printf("*********     1.add      2.sub    ***********\n");
    printf("*********     3.mul      4.div    ***********\n");
    printf("*********          0.exit         ***********\n");
    printf("*********************************************\n");
}
int add(int x, int y)
{
    return x + y;
}
int sub(int x, int y)
{
    return x - y;
}
int mul(int x, int y)
{
    return x * y;
}
int div(int x, int y)
{
    return x / y;
}
void DiaoYong(int (*p)(int,int))    //使用函数指针 ,减少代码冗余度
{
    int x = 0;
    int y = 0;
    printf("请输入两个整数\n");
    scanf("%d %d",&x,&y);
    int n=p(x,y);
    printf("%d\n",n);
}
int main()
{
    int input = 0;
    do
    {
        menu();
        printf("请输入选项\n");
        scanf("%d",&input);
        switch (input)
        {
        case 1:
            DiaoYong(add);
            break;
        case 2:
            DiaoYong(sub);
            break;
        case 3:
            DiaoYong(mul);
            break;
        case 4:
            DiaoYong(div);
            break;
        case 0:
            printf("退出程序\n");
            break;
        default:
            printf("数值错误,请重新输入\n");
                break;
        }
    } while (input);
    return 0;
}

5.typedef对函数指针的用法 

 #include<stdio.h>
typedef void(*pt)(int);
//将void(*)(int)改名为pt
int main()
{
    void(*signal(int,void(*)(int)))(int);
    void(*signal(int, pt))(int);
    return 0;
}

六.函数指针数组

1.函数指针的形式

 #include<stdio.h>   //输出两个整数加减乘除的结果
int add(int x, int y)
{
    return x + y;
}
int sub(int x, int y)
{
    return x - y;
}
int mul(int x, int y)
{
    return x * y;
}
int div(int x, int y)
{
    return x / y;
}
int main()
{
    int input = 0;
//    int(*p)(int, int) = add;   //函数指针
    int(*p[4])(int, int) = {add,sub,mul,div};   //函数指针数组
    for (int i=0;i<4;i++)
    {
        printf("%d\n",p[i](8,4));
    }
    return 0;
}

2.函数指针的应用 

#include<stdio.h>    //简易计算器
void menu()
{
    //add—>加       sub—>减       mul—>乘        div—>除
    printf("*********************************************\n");
    printf("*********     1.add      2.sub    ***********\n");
    printf("*********     3.mul      4.div    ***********\n");
    printf("*********          0.exit         ***********\n");
    printf("*********************************************\n");
}
int add(int x, int y)
{
    return x + y;
}
int sub(int x, int y)
{
    return x - y;
}
int mul(int x, int y)
{
    return x * y;
}
int div(int x, int y)
{
    return x / y;
}
int main()
{
    int input = 0;
    int x = 0;
    int y = 0;
    int ret = 0;
    int (*p[5])(int, int) = {0,add,sub,mul,div};
    do
    {
        menu();
        printf("请输入选项\n");
        scanf("%d", &input);
        if (input == 0)
        {
            printf("退出计算器\n");
        }
        else if(input >= 1 && input <= 4)
        {
            printf("请输入两个整数\n");
            scanf("%d %d", &x,&y);
            printf("%d\n", p[input](x,y));
        }
    }while (input);
    return 0;

七.指向函数指针数组的指针

 int main()
{
    //函数指针数组
    int (*p[5])(int, int) = {0,add,sub,mul,div};
    //指向[函数指针数组]的指针
    int(*(*pr)[5])(int, int) = &p;
    return 0;
}

八.回调函数

 1.回调函数的定义

2. 回调函数的应用

(1)例一:简易计算器

 #include<stdio.h>    //简易计算器
void menu()
{
    //add—>加       sub—>减       mul—>乘        div—>除
    printf("*********************************************\n");
    printf("*********     1.add      2.sub    ***********\n");
    printf("*********     3.mul      4.div    ***********\n");
    printf("*********          0.exit         ***********\n");
    printf("*********************************************\n");
}
int add(int x, int y)
{
    return x + y;
}
int sub(int x, int y)
{
    return x - y;
}
int mul(int x, int y)
{
    return x * y;
}
int div(int x, int y)
{
    return x / y;
}
void DiaoYong(int (*p)(int,int))    //使用函数指针 ,减少代码冗余度——回调函数
{
    int x = 0;
    int y = 0;
    printf("请输入两个整数\n");
    scanf("%d %d",&x,&y);
    int n=p(x,y);
    printf("%d\n",n);
}
int main()
{
    int input = 0;
    do
    {
        menu();
        printf("请输入选项\n");
        scanf("%d",&input);
        switch (input)
        {
        case 1:
            DiaoYong(add);
            break;
        case 2:
            DiaoYong(sub);
            break;
        case 3:
            DiaoYong(mul);
            break;
        case 4:
            DiaoYong(div);
            break;
        case 0:
            printf("退出程序\n");
            break;
        default:
            printf("数值错误,请重新输入\n");
                break;
        }
    } while (input);
    return 0;
}

(2)例二 :qsort()函数的应用——整数排序

#include<stdio.h>
#include<stdlib.h>
//void qsort          qsort()函数初始是递增排序
// (void* base,       要排序的初始位置
// size_t num,        待排序的元素个数
// size_t width,      待排序的数据元素的大小(字节)
// int(*cmp)(const void* e1,const void* e2))    函数指针—比较函数
//_cdecl——函数调用约定,约定函数由c语言的语法调用

//设置比较函数
int cmp1(const void* x1, const void* x2)
{
    return *(int*)x1 - *(int*)x2;
}
int cmp2(const void* x1, const void* x2)
{
    return *(int*)x2 - *(int*)x1;
}
//函数定义要求e1-e2>0时输出整数,e1-e2=0时输出1=0,e1-e2<0时输出负数
//qsort()函数初始是递增排序,若要变为递减排序,则要交换e1和e2

int main()
{
    //定义数组
    int i = 0;
    int arr[10] = {9,8,7,6,5,4,3,2,1,0};
    int brr[10] = {0,1,2,3,4,5,6,7,8,9};
    //求排序数组的大小
    int sz1 = sizeof(arr) / sizeof(arr[0]);
    int sz2 = sizeof(brr) / sizeof(brr[0]);
    //调用qsort()函数
    qsort(arr,sz1,sizeof(int),cmp1);
    qsort(brr,sz2,sizeof(int),cmp2);
    //输出排序后的数组
    for (i = 0; i < sz1; i++)
    {
        printf("%d ",arr[i]);
    }
    printf("\n");
    for (i = 0; i < sz2; i++)
    {
        printf("%d ", brr[i]);
    }
    return 0;
}

(3)qsort()函数的应用——结构体排序 

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
//void qsort          qsort()函数初始是递增排序
// (void* base,       要排序的初始位置
// size_t num,        待排序的元素个数
// size_t width,      待排序的数据元素的大小(字节)
// int(*cmp)(const void* e1,const void* e2))    函数指针—比较函数
//_cdecl——函数调用约定,约定函数由c语言的语法调用

//函数定义要求e1-e2>0时输出整数,e1-e2=0时输出1=0,e1-e2<0时输出负数
//qsort()函数初始是递增排序,若要变为递减排序,则要交换e1和e2
struct Stu
{
    char name[20];
    int age;
};
//设置比较函数
int cmp1(const void* x1, const void* x2)
{
    return strcmp(((struct Stu*)x1)->name,((struct Stu*)x2)->name);
}
int cmp2(const void* x1, const void* x2)
{
    return ((struct Stu*)x1)->age - ((struct Stu*)x2)->age;
}


void test1()        //排序结构体的name元素
{
    struct Stu s[3] = { {"zhangsan",15},{"lisi",30},{"wangwu",25} };
    int sz = sizeof(s) / sizeof(s[0]);
    qsort(s, sz, sizeof(s[0]),cmp1);
    for (int i = 0; i < 3; i++)
    {
        printf("%s ", s[i].name);
    }
}
void test2()    //排序结构体的age元素
{
    struct Stu s[3] = { {"zhangsan",15},{"lisi",30},{"wangwu",25} };
    int sz = sizeof(s) / sizeof(s[0]);
    qsort(s, sz, sizeof(s[0]), cmp2);
    for (int i = 0; i < 3; i++)
    {
        printf("%d ", s[i].age);
    }
}
int main()
{
    test1();
    printf("\n");
    test2();
    return 0;
}

 

(4)用冒泡排序模拟实现qsort()函数

 #include<stdio.h>    
int cmp(const void* x1, const void* x2)
{
    return (*(int*)x1 - *(int*)x2);
}
void Swap(char* x, char* y, int width)  //将两个数改为char*类型,每次只交换一个字节,直到将int*的四个字节全部交换一遍
{
    int i = 0;
    for (i = 0; i < width; i++)
    {
        char tmp = *x;
        *x = *y;
        *y = tmp;
        x++;
        y++;
    }
}
sqort_moni(int* arr,int sz,int width, int (*cmp)(const void*, const void*))
{
    int i,j;
    for (i = 0; i < sz - 1; i++)
    {
        int flag = 1;
        for (j = 0; j < sz - 1 - i; j++)
        {
            if (cmp((char*)arr + j * width, (char*)arr + (j + 1) * width )> 0) //返回值大于0,则说明x1>x2,需要顺序排列则要交换两个数
            {
                Swap((char*)arr + j * width, (char*)arr + (j + 1) * width, width);
                flag = 0;
            }
        }
        if (flag == 1)  //如果循环一整遍之后都符合条件,则直接跳出循环
        {
            break;
        }
    }
}
int main()
{
    int arr[10] = {9,8,7,6,5,4,3,2,1,0};
    int sz = sizeof(arr) / sizeof(arr[0]);
    sqort_moni(arr,sz,sizeof(arr[0]), cmp);
    for (int i = 0; i < sz; i++)
    {
        printf("%d ",arr[i]);
    }
    return 0;
}

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

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

相关文章

《ImageNet Classification with Deep Convolutional Neural Networks》阅读笔记

论文标题 《ImageNet Classification with Deep Convolutional Neural Networks》 ImageNet &#xff1a;经典的划时代的数据集Deep Convolutional&#xff1a;深度卷积在当时还处于比较少提及的地位&#xff0c;当时主导的是传统机器学习算法 作者 一作 Alex Krizhevsky 和…

点餐小程序实战教程07-点餐功能开发

目录 1 菜品列表功能2 给分类添加排序字段3 给数据列表设置筛选条件4 实现数量选择5 全局变量定义6 选购数量初始化总结 上一篇我们介绍了菜品分类功能&#xff0c;主要是使用侧边选项卡组件来实现。在点餐场景中&#xff0c;我们在选中分类的时候是希望看到分类下的菜品&#…

面试官:听说你很懂SpringMVC,那讲讲其内部对于请求的处理吧!

本系列文章皆在分析SpringMVC的核心组件和工作原理&#xff0c;让你从SpringMVC浩如烟海的代码中跳出来&#xff0c;以一种全局的视角来重新审视SpringMVC的工作原理. 作者&#xff1a;毅航 在过去的很长一段时间内&#xff0c;笔者对SpringMVC中的核心控制器DispatcherServle…

《鱿鱼游戏》杀入体验店 SandboxVR未能复制神话

曾经火爆全网的剧集《鱿鱼游戏》这回真的变成游戏了&#xff0c;而且还是能身临其境去玩的那种。 9月底&#xff0c;沉浸式游戏社交公司Sandbox VR与全球知名流媒体Netflix合作&#xff0c;将《鱿鱼游戏》搬入VR体验店&#xff0c;用线下场地VR的方式&#xff0c;重现123木头人…

jvm--执行引擎

文章目录 1. 执行引擎的工作流程2. 解释器、JIT及时编译器3. 热点代码及探测技术4. HotSpotVM 中 JIT 分类 执行引擎属于 JVM 的下层&#xff0c;里面包括解释器、及时编译器、垃圾回收器 JVM 的主要任务是负责 装载字节码到其内部&#xff0c;但字节码并不能够直接运行在操作…

vite+vue3+ts中使用require.context | 报错require is not defined | 获取文件夹中的文件名

vitevue3ts中使用require.context|报错require is not defined|获取文件夹中的文件名 目录 vitevue3ts中使用require.context|报错require is not defined|获取文件夹中的文件名一、问题背景二、报错原因三、解决方法 一、问题背景 如题在vitevue3ts中使用required.context时报…

JDK21 要来了,,你不会还在用 Java 8吧!

目前 Java 的最新稳定版是 JDK 20&#xff0c;但这是个过渡版&#xff0c;JDK21就是 LTS 版的了&#xff0c;也快要发布了&#xff0c;在今年9月份&#xff08;也就是2023年9月&#xff09;就要正式发布了。 但是&#xff0c;猜都不用猜&#xff0c;你肯定还在用 Java 8 吧&…

【密评】商用密码应用安全性评估从业人员考核题库(七)

商用密码应用安全性评估从业人员考核题库&#xff08;七&#xff09; 国密局给的参考题库5000道只是基础题&#xff0c;后续更新完5000还会继续更其他高质量题库&#xff0c;持续学习&#xff0c;共同进步。 1501 判断题 在公钥密码体制中&#xff0c;使用接收方的公钥加密的消…

MySQL通用查询日志 general query log 详解

通用查询日志&#xff08;general query log&#xff09;用来记录用户的所有操作&#xff0c;包括启动和关闭MySQL服务、所有用户的连接开始时间和截止时间、发送给MySQL数据库服务器的所有SQL指令等。当我们的数据发生异常时&#xff0c;查看通用查询日志&#xff0c;还原操作…

Oracle笔记-对ROWNUM的一次理解(简单分页)

此博文记录时间&#xff1a;2023-05-05&#xff0c;发到互联网上是2023-10-09 这个在分页里面用得比较多&#xff0c;在MySQL中&#xff0c;通常使用limit去操作&#xff0c;而去感觉比较简单&#xff0c;Oracle中无此关键字。 通过查阅资料后&#xff0c;要实现分页需要用到…

论文阅读笔记(Clover: 计算与存储被动分离的分布式键值存储系统)

关于Disaggregating Persistent Memory and Controlling Them Remotely: An Exploration of Passive Disaggregated Key-Value Stores这篇论文的笔记 原文链接 提出背景 传统的分布式存储系统中&#xff0c;每个节点都会包含计算和存储两个部分&#xff0c;一个节点既可以访…

分割等和子集

题目链接 分割等和子集 题目描述 注意点 数组 nums 非空数组 nums 只包含正整数 解答思路 最初想到的是根据回溯剪枝解决本题&#xff0c;如果数组大小小于2&#xff0c;则肯定不能找到分割等和子集&#xff0c;除此以外&#xff0c;如果数组之和sum不能被2整除&#xff0…

VMvare虚拟机安装国产麒麟V10桌面操作系统

一、系统下载 进入银河麒麟官网&#xff1a;https://www.kylinos.cn/ 选择桌面操作系统&#xff0c;然后进入操作系统版本选择页面&#xff0c;选择银河麒麟桌面操作系统V10 选择后&#xff0c;进入系统介绍页面&#xff0c;然后点击申请试用 点击后进入申请页面&#xf…

Apache Solr9.3 快速上手

Apache Solr 简介 Solr是Apache的顶级开源项目&#xff0c;使用java开发 &#xff0c;基于Lucene的全文检索服务器。 Solr比Lucene提供了更多的查询语句&#xff0c;而且它可扩展、可配置&#xff0c;同时它对Lucene的性能进行了优化。 安装 下载 : 下载地址解压 : tar -zxv…

uniapp apple 苹果登录 离线本地打包

官方文档 uni-app官网 文档写的不全&#xff0c;没有写离线打包流程 加lib 签名里带 sign in with apple hbuilder开关 代码 测试代码&#xff0c;获取app里所有的provider uni.getProvider({service: oauth,success: function (res) {console.log(res.provider)uni.showT…

ctfshow-web4(文件包含日志注入)

像web3那样使用php伪协议&#xff0c;回显error 看了看提示&#xff1a;日志注入 文件包含 使用Wappalyzer查看一下&#xff0c;使用的中间件是Ngnix 日志包含漏洞的成因还是服务器没有进行严格的过滤 &#xff0c;导致用户可以进行任意文件读取&#xff0c; 但是前提是服务器…

月报总结|Moonbeam 9月份大事一览

相信社区的小伙伴都过了一个圆满的中秋节&#xff0c;月圆人团圆&#xff0c;以月光代表生态的Moonbeam网络也如月亮一般&#xff0c;从始至终都在波卡。 9月&#xff0c;USDC上线波卡增加流动性、Moonbeam治理界面简化促进社区参与、Grant发放引爆Moonbeam Ignite再次回归、M…

QT基础入门——Qt事件(五)

前言&#xff1a; 事件&#xff08;event&#xff09;是由系统或者 Qt 本身在不同的时刻发出的。当用户按下鼠标、敲下键盘&#xff0c;或者是窗口需要重新绘制的时候&#xff0c;都会发出一个相应的事件。一些事件在对用户操作做出响应时发出&#xff0c;如键盘事件等&#x…

(一)Apache log4net™ 手册 - 介绍

0、相关概念 Log4j 几乎每个大型应用程序都包含自己的日志记录或跟踪 API。根据这一规则&#xff0c;E.U. SEMPER &#x1f339;项目决定编写自己的跟踪 API。那是在 1996 年初。经过无数次的增强、几个化身和大量的工作&#xff0c;API 已经发展成为 log4j —— 一个流行的 Ja…

开源音乐播放器!

导读音乐是生活的一部分。维基百科关于音乐发展历史的文章有这样一段不错的描述说&#xff1a;“全世界所有的人们&#xff0c;包括哪怕是最孤立、与世隔绝的部落&#xff0c;都会有自己的特色音乐……”好吧&#xff0c;我们开源人就构成了一个部落。我建议我们的“音乐形式”…