【c1】数据类型,运算符/循环,数组/指针,结构体,main参数,static/extern,typedef

news2024/9/21 16:28:29

文章目录

  • 1.数据类型:编译器(compiler)与解释器(interpreter),中文里的汉字和标点符号是两个字节,不能算一个字符(单引号)
  • 2.运算符/循环:sizeof/size_t
  • 3.数组:存数据类型相同的数据,数组下标越界导致段错误
  • 4.指针:指针数组:这个数组的所有元素都是指针类型。数组指针:这个指针存放着一个数组的首地址
    • 4.1 地址/值/变量名:p是地址,*p是指向的内容,01指0x01,&a:拿变量a的地址赋给
    • 4.2 函数调用:复制a/指向a
    • 4.3 函数返回:复制指针
    • 4.4 数组和指针转换:指针转不了数组,数组可转为指针,&取地址
    • 4.5 字符串str相关:putchar('w');putchar('w');
  • 5.结构体:存放一组不同数据类型的数据
  • 6.main函数的参数:main函数的参数是从命令提示符下执行程序时传入
  • 7.static/inline/回调/extern/堆内存:inline内联函数本身就是static(本文件私有)函数,inline函数在c中作用性不高,c中用习惯了宏定义
  • 8.typedef:define仅仅是简单的字符串的替换,而typedef则给这个类型新起了一个名字
    • 8.1 给已定义的变量类型起个别名:给struct __person起了个别名person_t
    • 8.2 定义函数指针类型:必须用typedef,方法就是在“定义函数指针变量”前加上typedef
    • 8.3 定义数组指针类型:先看如何定义数组指针变量
    • 8.4 定义数组类型:声明一个含5个int元素的一维数组:int a[5]


1.数据类型:编译器(compiler)与解释器(interpreter),中文里的汉字和标点符号是两个字节,不能算一个字符(单引号)

编译型语言:C/C++/golang/rust:并且都是AOT(ahead of time)预先编译,编译将源码编译成机器码生成二进制文件,可直接运行该文件,因为是机器码,所以运行效率很高。缺点:不够灵活,改代码要重新编译,此外平台依赖,linux平台编译出来的二进制文件无法在windows运行,跨平台还需借助交叉编译。

解释型语言:python/js/php:不需要生成二进制文件,灵活如线上php系统,改了代码功能直接生效,但运行效率低。

半解释半编译型语言:java:JIT将运行到的代码块在运行时编译成机器码,既可保证跨平台性,又能使热代码高效运行。

C库函数声明头文件(.h):存放在/usr/include目录中,如果 #include <facebook/cJSON.h>,则cJSON.h在/usr/include/facebook/目录里。如下github上软件包(devtool search),在image.bb中添加hiredis软件包查找hiredis.bb文件【内容有git网址】在yocto或common或meta-aspeed或meta-phosphor文件夹中,.h文件会在/usr/include/中,但rootfs中看不到。
在这里插入图片描述
C库函数定义(具体实现)文件(.c):gcc编译不用包含libgcc.a(缺省会包含),/usr/lib/gcc/x86_64…/4.4.4/libgcc.a(打包好的静态库)。

gcc main.c int_sum.c float_sum.c    -o main -Wall(显示所有警告)    -I../include(-I后面没空格,不加-I就在main.c中指定头文件的相对路径)
gcc main.c   -o main -Wall     -I../include      -L../lib     -lsum(有库文件/lib/libsum.so就只要main.c就行)

2个字节的-480在计算机中为0xfe20,ipmitool发指令:低位在前即0x20 0xfe。
在这里插入图片描述
先确定字节数,再将最高位1固定。错误显示:100…010。正确存储:11…110。计算用正确存储。
在这里插入图片描述

num=5 # 0000 0101
result=$((~num))  # 得到: 1111 1010
echo $result  # -6  # 负数在计算机中存储是取反加1,1000 0101 + 1 = 1000 0110(-6 错误显示)
# 所以~作用是不管符号位,加1作用: 0000 0101 + 1 = 6

num=-5 # 不是错误显示:1000 0101 ,是补码取反加1 :1111 1011
result=$((~num))  # 得到: 0000 0100
echo $result  # 4

在这里插入图片描述
浮点数不是2的0次方+2的1次方…,最小可识别精度和浮点数即0.1本身大小相关,而不是和float相关。
在这里插入图片描述
计算机中以字节为单位存储数据,1字节Byte=8bit,int=int32_t。
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
d:十进制。o:八进制。x:十六进制。
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

int main()
{
    int value = 1;
    char str[10] = "-q1"; // NA:0 , -1:-1 , 1:1 , q1:0
    value = atoi(str);    //字符串转换为整数(不是ascii码), strtol
    printf("%d\n",value); //0
}
#define PLATFORM_NAME_PATH      "./a.py"
void get_machine_ver(char * result)
{
    char buffer[32] = {0};
    char cmd[128] = {0};
    strcpy(cmd, "python3 ");
    strcat(cmd, PLATFORM_NAME_PATH);
    FILE* file = popen(cmd, "r");
    if (file == NULL) {
        return;
    }

    // memset(buffer, '\0', sizeof(buffer));
    fgets(buffer, sizeof(buffer), file);
    printf("111,%ld\n",sizeof(buffer));  // 32   // char buffer[32] = {0};
    printf("222,%ld\n",strlen(buffer));  // 14 连换行符

    strncat(result, buffer, strlen(buffer)-1); // 去除换行符  // memcpy(result, buffer, strlen(buffer)-1); result长度不知,请确保 result 缓冲区足够大,以容纳 buffer 中的数据,并且不会发生缓冲区溢出的情况。
    // while (fgets(buffer, 10, file) != NULL) {
    //     strcat(result, buffer);
    // }
    pclose(file);
    return;
}

int main(){
    char result[32] = {0};
    get_machine_ver(result);
    printf("333,%s",result); // 333, HP1-2C4F-0..
}

1.数据输出:在C语言中,有三个函数可以把数据输出到屏幕。
putchar 用于输出单个字符。
puts 输出字符串。

2.输出整数
输出的整数常量或整数变量用%d表示,在参数中列出待输出的整数常量或整数变量。
int age=18;
printf(“我年龄是%d岁。\n”,age);

3.输出字符
输出的字符常量或字符变量用%c表示,在参数中列出待输出的字符常量或字符变量。
char xb=‘x’;
printf(“我姓别是:%c。\n”,xb);

4.输出浮点数
输出的浮点型常量或浮点型变量用%lf表示,在参数中列出待输出的浮点型常量或浮点型变量。
double weight=62.5;
printf(“我体重是%lf公斤。\n”, weight);

5.输出字符串
输出的字符串常量或字符串型变量用%s表示,在参数中列出待输出的字符串常量或字符串变量。
char name[21];
memset(name,0,sizeof(name));
strcpy(name, “豫让”);
printf(“我的姓名是%s。\n”,name);

关键字:共32个,也就是说这些单词在C语言中有特别的含义,程序员不能把它用于变量或函数的命名。auto :声明自动变量。break:跳出当前循环。case:开关语句分支。char :声明字符型变量或函数返回值类型。const :声明只读变量。continue:结束当前循环,开始下一轮循环。default:开关语句中的“默认”分支。do :循环语句的循环体。double :声明双精度浮点型变量或函数返回值类型。else :条件语句否定分支(与 if 连用)。enum :声明枚举类型。extern:声明变量或函数是在其它文件或本文件的其他位置定义。float:声明浮点型变量或函数返回值类型。for:一种循环语句。goto:无条件跳转语句。if:条件语句。int: 声明整型变量或函数。long :声明长整型变量或函数返回值类型。register:声明寄存器变量。return :子程序返回语句(可以带参数,也可不带参数)。short :声明短整型变量或函数。signed:声明有符号类型变量或函数。sizeof:计算数据类型或变量长度(即所占字节数)。static:声明静态变量。struct:声明结构体类型。switch:用于开关语句。typedef:用以给数据类型取别名。unsigned:声明无符号类型变量或函数。union:声明共用体类型。void:声明函数无返回值或无参数,声明无类型指针。volatile:说明变量在程序执行中可被隐含地改变。while:循环语句的循环条件。

2.运算符/循环:sizeof/size_t

|| 左边的命令返回假(命令返回值 $? == 1),|| 右边的命令才会被执行,和c语言逻辑或相同。
在这里插入图片描述
如下ii++就是ii=ii+1。前先加1。
在这里插入图片描述
=是赋值,==才是判断两个数是否相等,C语言没有(之间)的关系运算符,如年龄在25-30岁之间:年龄大于等于25岁并且年龄小于等于30岁。

sizeof运算符(不是函数)计算某一个变量在当前系统的内存中所需占用的字节数:
1.用于数据类型:sizeof(int)) = 4,sizeof(指针)大小永远是8字节。
2.用于变量:sizeof使用形式:sizeof(var_name)或sizeof var_name。

sizeof结果类型是size_t:typedef unsigned int size_t为无符号整型,长度为4个字节(32位系统)。typedef unsigned long size_t为无符号长整型,长度为8个字节(64位系统)。ssize_t是有符号整型(在32位机器上等同int,在64位机器上等同long int)。

不要在if(判断条件)后面加分号。有;号就为空语句,下面都执行。c=(a>b)?a:b等同于两行:if(a>b) c=a;else c=b; if(0)即0假。if(a=b)是赋值,不是判断。当没有default时,如果所有case都匹配失败,则什么都不执行。char day=0,scanf(‘%c’',&day),case ‘0’。
在这里插入图片描述

#include<stdio.h>
int main()
{
  int a=1, b=2, re;
  char c;
  scanf("%c", &c);
  switch(c)
  {
    case '+':
      re = a + b;
      break;
    case '#':
    {
      int other = 3;   // switch语句里定义了新的变量 ,加{}
      re = a + b + other;
      break;
    }
    default:
      printf("Illegal input!\n");
      break;
  }
  printf("%d\n", re);
}

$gcc -o main *.c -lm
$main
Illegal input!
0

在这里插入图片描述
while(真)执行,continue跳到循环首部,break跳出循环。
在这里插入图片描述
在这里插入图片描述
如下do-while先执行一次循环。
在这里插入图片描述

for(i=0;i<10;i++)
{
    printf("%d\n",i); //没有下行0-9,有下行一直0
    i--;
}

3.数组:存数据类型相同的数据,数组下标越界导致段错误

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
如下如果用msg[1],虽然越界了,但是系统给结构体分配很多字节,所以不报错。可用msg[0]到msg[n-1]。
在这里插入图片描述

4.指针:指针数组:这个数组的所有元素都是指针类型。数组指针:这个指针存放着一个数组的首地址

4.1 地址/值/变量名:p是地址,*p是指向的内容,01指0x01,&a:拿变量a的地址赋给

在这里插入图片描述

#include<stdio.h>

char str[10];
char strbuf[10];
char *a(void)
{
    FILE *fp=0;
    if ((fp=fopen("a","rt")) ==0)  //a文件里写了 333aaa
    {
        printf("111");
    }
    if (fgets(strbuf, 10, fp) == NULL) 
    {
        printf("222");
        fclose(fp);
    }
    strcpy(str,"B");
    strcat(str,strbuf);
    fclose(fp);
    return str;
}

int main()
{
    char *b=a();
    printf("%s\n", b);  // B333aaa
}

int main()
{
    FILE *fp=0;
    if ((fp = fopen("/tmp/bbb","r")) == 0)
    {
        printf("file do not exist\n");
        fclose(fp); // Segmentation fault (core dumped),fp不存在不能close,上行没打印出是因为fclose报错早于printf
        return -1;
    }
    fclose(fp); // 走不到
    return 0;
}
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

int main()
{
    int len,i=0;
    char * pid_name_config_1[] = {"GPU111111110","CPU0_DIMM0","MOC2.5_CPU"};
    printf("000,%s\n",pid_name_config_1[0]);  // 000,GPU111111110
    printf("111,%ld\n",sizeof(pid_name_config_1[0])); // 111,8(指针大小)

    printf("222,%ld\n",sizeof(pid_name_config_1)); // 222,24(3*8)
    len = (sizeof(pid_name_config_1)/sizeof(pid_name_config_1[0]));
    printf("333,%d\n",len); // 333,3

    printf("444,%ld\n",sizeof("GPU0")); // 444,5(最后\0结束符)
    printf("555,%ld\n",strlen("GPU0")); // 555,4

    char pid_name[20]={0};  // char pid_name[20][20]={0}; 可以,每一行用来存字符串,不是字符
    for(i=0; i<len; i++){
        strcpy(pid_name[i], pid_name_config_1[i]); // 错误,strcpy参数expected ‘char * restrict’ but argument is of type ‘char
        memcpy(pid_name[i], pid_name_config_1[i], strlen(pid_name_config_1[i])); // 同上
        printf("666,%s\n", pid_name[i]);
    }
}

int *p 未赋值的指针称为野指针(危险),a=100,p=&a指向合法区域。或如下空指针的两种写法也可防止野指针。
在这里插入图片描述

4.2 函数调用:复制a/指向a

在这里插入图片描述
1.如下main中走到increament跳到这函数中,拷贝一份给increament中的a。main中的a和increament中的a各自独立一块内存,只是名字一样。
在这里插入图片描述
2.一定要通过increasement函数对a有修改怎么办?如下必须用到指针,increament运行结束后指针销毁。
在这里插入图片描述
在这里插入图片描述

4.3 函数返回:复制指针

如下当调用move_p时,p要往右移一位:move_p参数定义虽是指针,但调用时传入也是指针,复制一份。
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

4.4 数组和指针转换:指针转不了数组,数组可转为指针,&取地址

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

4.5 字符串str相关:putchar(‘w’);putchar(‘w’);

#include <stdio.h>
#include <string.h>
int main()
{
	  char *names = "PSU";
	  if(strcmp(names,"PSU")==0)
	  {
		  puts("aaaa");  // 打印出
	  }
}

int main() {   // error: expected ‘)’ before string constant    少了int main()
    char name_str[30];
    char name_str1[30];
    strcpy(name_str,"/sys/bus/i2c/devices/");
    strcpy(name_str1,"17-0064");
    strcat(name_str,name_str1);
    printf("is : %s\n", name_str);   // /sys/bus/i2c/devices/17-0064  
     // printf("[%s]__%4d__[%s]  %s \n", __FILE__, __LINE__, __FUNCTION__, n->name);
}
#include <stdio.h>
#include <string.h>
int main ()
{
   char str1[15];
   char str2[15];
   int ret;
   strcpy(str1, "abcdef");
   strcpy(str2, "ABCDEF");
   ret = strcmp(str1, str2);
   if(ret < 0)
   {
      printf("str1 小于 str2");
   }
   else if(ret > 0) 
   {
      printf("str1 大于 str2");
   }
   else 
   {
      printf("str1 等于 str2");
   }
   return(0);    //str1 大于 str2
}
#include <stdio.h>
#include <stdlib.h>
#include<string.h>
#include <pthread.h>
#include <unistd.h>

#if 0
int main() {
	// int a;
	 // char pNum[]="0x7f";
	 // a=strtoul(pNum,0,16);
	 // printf("%d\n",a);  //127 ,%x也是127 
	 // return 0;
   char str[30] = "2030300 This is test";
   char *ptr;
   long ret;
   ret = strtoul(str, &ptr, 10);
   printf("数字(无符号长整数)是 %lu\n", ret);  // 2030300
   printf("字符串部分是 |%s|\n", ptr);  // This is test
   return 0;
}
#endif
char temp_log_0[100] = {0};
sprintf(temp_log_0, " '%d' ", a);
char temp_log_1[100] = "echo ";
strcat(temp_log_1, temp_log_0);
char temp_log_2[100] = " >> /var/log/a.log";
strcat(temp_log_1, temp_log_2);
system(temp_log_1);  // C
if(-1 == std::system(temp_log_1)){}; // CPP
if(-1 == std::system("echo 'aaa' >> /var/log/a.log")){};
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
 
void cp (char *path_from, char *path_to)
{
	FILE *fp_read = NULL;
    FILE *fp_write = NULL;
    char ch = !EOF;
    fp_read = fopen(path_from, "r"); ///
    // if (fp_read == NULL)
    // {
    //     printf("您没有这个(%s)文件\n", path_from);
    // }
    fp_write = fopen(path_to, "w"); 
    
    while ((ch = fgetc(fp_read)) != EOF) // 读
    {
        fputc(ch,fp_write);   // 写
    }
    fclose(fp_read);
    fclose(fp_write);
}

int main(void)
{
    char path_from[50];
    char path_to[50];

    printf("输入文件名:"); // D:\1.txt
    scanf("%s", path_from);

	printf("\n输入文件名:"); // D:\2.txt,不存在会自动创建
    scanf("%s", path_to);

    cp(path_from,path_to); 
    return 0;
}
#include<stdio.h>
#include<assert.h>

char* my_strcpy(char* dest, char* src)  //所以我这可以用两个char*类型了指针来接收
{
	assert(dest && src);//这个的意思就是避免dest和src是空指针,(如果有了这个assert(断言),就可以使当它们其中之一有空指针的时候就会报错,避免程序运行不报错,但要注意引头文件)
	char* ret = dest;//写这步的好处和原因有两个 1.可以使我的dest发生改变的时候还有一个指针指向它,使我便与寻找 2.可以使我的返回类型变得更加完美,完美实现char*的返回值的目的
	while (*src)
	{
		*dest = *src;//这个的意思就是把源头的字符赋值给目的地
		dest++;//这两步一样就是使指针指向下一个字符,然后再循环
		src++;
	}
	*dest = *src;//这步的目的就是因为上面那个循环的条件是 src!='\0',所以当src为'\0'时,循环就会停止,导致*dest = *src这步在最后不能实现,所以'\0'就没有拷贝到dest中,所以我最后还要再进行一步赋值
	// while (*dest++ = *src++); // 这步的还是解引用后直接复制的意思,只是放在了循环之中(意思为当src为'\0'时,dest也为'\0',并且'\0'的ASCII码值为0,所以为假,所以此时循环依然停止), 这行可替代上面7行
	return ret;
}

char* my_strncpy(char* dest, char* src, size_t num)  //比上面多了一个n
{
	assert(dest && src);//同理
	char* ret = dest;
	while (num)
	{
		*dest = *src;  //复制
		src++;    //源地址往后+1
		dest++;
		num--; //我所需要拷贝的字符数随着我的循环一直减减
	}
	return ret;
}

int main()
{
	char arr1[] = "abcdef";
	char arr2[20] = "bcd";
	my_strcpy(arr2, arr1);//因为数组就是首元素的地址,所以这边传上去的其实就是两个地址
	printf("%s",arr2);
	return 0;
}

5.结构体:存放一组不同数据类型的数据

在这里插入图片描述
如下最后一行*pst就是queen结构体变量。
在这里插入图片描述
结构体复制:基本类型用=,字符串用strcpy,结构体memcpy。
在这里插入图片描述
结构体作为函数的参数:结构体成员较多,函数参数的初始化和赋值的开销很大,最好的办法就是传递结构体变量的地址。
在这里插入图片描述

6.main函数的参数:main函数的参数是从命令提示符下执行程序时传入

在这里插入图片描述
在这里插入图片描述

#include <stdio.h>
int main(int argc, char **argv) {
  char *stty, *dev;
  dev = argv[1];
  stty = argv[2];
  printf("%s\n%s\n",dev,stty);
}

在这里插入图片描述

// a.c
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <netdb.h>

extern int h_errno;
int main(int argc, char **argv)
{
	char   *ptr, **pptr;
	char    str[INET_ADDRSTRLEN];
	struct hostent *hptr;  //
	while (--argc> 0) 
	{
		ptr = *++argv;  //传入的域名
		if ( (hptr = gethostbyname (ptr) ) == NULL)   //完成域名解析
		{
			printf("gethostbyname error for host: %s: %s",ptr, hstrerror (h_errno) );
			continue;
		}
		printf ("official hostname: %s\n", hptr->h_name);

		for (pptr=hptr->h_aliases; *pptr!= NULL; pptr++)
			printf ("\talias: %s\n", *pptr);

		switch (hptr->h_addrtype) 
		{
			case AF_INET:
				pptr = hptr->h_addr_list;
				for ( ; *pptr != NULL; pptr++)
					printf ("\taddress: %s\n",inet_ntop (hptr->h_addrtype, *pptr, str, sizeof (str)));  //hptr->h_addrtype我们获取的IP地址
				break;
			
			default:
				printf("unknown address type");
				break;
		}
	}
	exit(0);
}

$ gcc a.c -o a
$ ./a www.baidu.com
official hostname: www.a.shifen.com
        alias: www.baidu.com
        address: 180.101.50.188  (浏览器输入都会跳转到百度)
        address: 180.101.50.242

7.static/inline/回调/extern/堆内存:inline内联函数本身就是static(本文件私有)函数,inline函数在c中作用性不高,c中用习惯了宏定义

什么函数被定义成内联函数呢?1.经常被调用(如果不经常被调用,节省一点时间也没意义),2.函数体里代码少(如果函数体里代码多,执行时间远大于跳转时间)。
在这里插入图片描述
如下左边的你有一个私人厨师,你将50块钱和厨师电话作为参数传给老好人函数,老好人做了一些买菜挑菜等杂活后调用你的厨师进行做饭,老好人不仅为你服务还为其他人服务。
在这里插入图片描述
如下方框是一个c文件,右边的c文件可以调到左边的var变量,每一个c文件不管里面写什么都可以编成o文件,右边的o文件var地址留空。最后链接成二进制文件时,链接就是编译,所以和1(外链)和2(内链)相关。
在这里插入图片描述
extern表示引用外部的变量,从外面来的。
在这里插入图片描述

8.typedef:define仅仅是简单的字符串的替换,而typedef则给这个类型新起了一个名字

8.1 给已定义的变量类型起个别名:给struct __person起了个别名person_t

struct __person
{
    char    name[20];
    uint8_t age;
}
typedef __person person_t;
//以上两段代码也可合并为一段:
typedef struct __person
{
	...
}person_t;

8.2 定义函数指针类型:必须用typedef,方法就是在“定义函数指针变量”前加上typedef

int (*pFunc)(char *frame, int len);  //定义了一个函数指针变量pFunc,它可以指向这样的函数:返回值为int,形参为char*、int
int *(*pFunc[5])(int len);  //定义了5个函数指针变量:pFunc[0]、pFunc[1]···,它们都可以指向这样的函数:返回值为int*,形参为int

//举例:
typedef  int (*pFunc_t)(char *frame, int len); //定义了一个类型pFunc_t 
int read_voltage(char *data, int len)
{
    int voltage = 0;
    return voltage;
}
int main(void)
{
    pFunc_t  pHandler = read_voltage;  //使用类型pFunc_t来定义函数指针变量
}

8.3 定义数组指针类型:先看如何定义数组指针变量

int(*pArr)[5]; //定义了一个数组指针变量pArr,pArr可以指向一个int[5]的一维数组
char(*pArr)[4][5]; //定义了一个数组指针变量pArr,pArr可以指向一个char[4][5]的二维数组

//举例:
int(*pArr)[5]; //pArr是一个指向含5个int元素(因为最前面是int)的一维数组的指针变量
int a[5] = {1,2,3,4,5};
int b[6] = {1,2,3,4,5,6};

pArr = &a; //完全合法,无警告
pArr = a; //发生编译警告,赋值时类型不匹配: a的类型为int(*) 相当于首地址,而pArr的类型为int(*)[5]
pArr = &a[0]; //发生编译警告,赋值时类型不匹配: a的类型为int(*),而pArr的类型为int(*)[5]

pArr = &b; //发生编译警告,赋值时类型不匹配:&b的类型为int(*)[6],而pArr的类型为int(*)[5]
pArr = (int(*)[5])&b; //类型强制转换为int(*)[5],完全合法,无警告,但复杂,简化如下:

typedef int(*pArr_t)[5];//定义一个指针类型,该类型的指针可以指向含5个int元素的一维数组
int main(void)
{
    int a[5] = {1,2,3,4,5};
    int b[6] = {1,2,3,4,5,6};
    pArr_t pA;//定义数组指针变量pA
    pA= &a;//完全合法,无警告    
    pA= (pArr_t)&b;//类型强制转换为pArr_t,完全合法,无警告
}

8.4 定义数组类型:声明一个含5个int元素的一维数组:int a[5]

// 声明多个含5个int元素的一维数组:int a1[5], a2[5], a3[5]···,或者 a[N][5]   复杂,这时应该把数组定义为一个类型:
typedef int arr_t[5];
int main(void)
{
    arr_t d;  //d是个数组,这一行等价于:  int d[5];
    arr_t b1, b2, b3; //b1, b2, b3都是数组
    d[0] = 1;
    d[5] = 253; //编译警告:下标越界
}

#define MaxNumbersOfName 10
#define MaxNumbersOfPhones 20
typedef char Elections_type[MaxNumbersOfNames]; // 10
Elections_type Phone[MAXNambersOfPhones]={"HUAWEI","XIAOMI","SAMSUNG","APPLE"}// char Phone[10][20]相同
typedef struct _jmp_buf
{ 
    int _jb[_JBLEN + 1]; 
} jmp_buf[1]; 
// jmp_buf(定义变量实体的同时,也获得了该变量的地址)是一个含一个元素的数组类型,数组的元素为struct  _jmp_buf类型

jmp_buf buf;      //这一行等价于:struct _jmp_buf buf[1]
buf->_jb[5] = 34; //这一行等价于:(&buf[0])->_jb[5] = 34
handle(buf);      //这一行等价于:handle(&buf[0])

#define char* pchar;
typedef pchar_type char*;
// 只有p1,p2,p3被成功定义字符指针变量
pchar_type p1,p2;
pchar p3,p4;  // 在编译的时候就会被编译器看作char *p3,p4;造成了p4是一个char类型的变量

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

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

相关文章

顶管机种类多样 国内产量不断增长

顶管机种类多样 国内产量不断增长 顶管机是一种用于非开挖管道铺设的机械设备&#xff0c;能够通过非开挖施工技术降低对地面活动的影响&#xff0c;具有工作效率高、安全性好、受地质条件限制小、环保性强等优点&#xff0c;在隧道修建、城市管网建设、地下管线敷设等场景中发…

《QT实用小工具·五十八》模仿VSCode的可任意拖拽的Tab标签组

1、概述 源码放在文章末尾 该项目实现了模仿VSCode的可任意拖拽的Tab标签组&#xff0c;包含如下功能&#xff1a; 拖拽标签页至新窗口 拖拽标签页合并控件 无限嵌套的横纵分割布局&#xff08;类似Qt Creator的编辑框&#xff09; 获取当前使用的标签组、标签页 自动向上合并…

测径仪视窗镜片的维护和保养步骤

关键字:测径仪镜片,测径仪保养,测径仪维护,视窗镜片维护,视窗镜片擦拭保养,视窗镜片的检查, 视窗镜片定期保养 视窗镜片是保护光学镜头免受污染和损伤的光学平镜片&#xff0c;它的污染和破损会直接影响光学系统的测量结果。 视窗镜片一般在受到轻微污染&#xff08;指镜片上…

项目管理-项目采购管理2/2

项目管理&#xff1a;每天进步一点点~ 活到老&#xff0c;学到老 ヾ(◍∇◍)&#xff89;&#xff9e; 何时学习都不晚&#xff0c;加油 本文承接 项目采购管理第二部分&#xff0c;详细讲解项目合同管理。 项目采购管理过程--重点&#xff1a; ①ITTO 输入&#xff0c;输出…

测试环境搭建:JDK+Tomcat+Mysql+Redis

基础的测试环境搭建&#xff1a; LAMPLinux(CentOS、ubuntu、redhat)ApacheMysqlPHP LTMJLinux(CentOS、ubuntu、redhat)TomcatMysql(Oracle)RedisJava 真实的测试环境搭建&#xff1a;&#xff08;企业真实的运维&#xff09; 基于SpringBoot&#xff08;SpringCloud分布式微…

分析:Palo Alto在从SASE向SASO演进中定位不佳

摘要 我们通过上一篇文章&#xff08;Fortinet的愿景——超越SASE&#xff09;中应用于Fortinet的相同框架来回顾Palo Alto Network在网络和网络安全方面的前景。 SASE涉及数据传输的第一英里。不过&#xff0c;随着SASE的发展&#xff0c;投资者还需要考虑中间和最后一英里。…

javaweb学习week7

javaweb学习 十四.Springboot 1.配置优先级 Springboot中支持三种格式的配置文件&#xff1a; 注意&#xff1a;虽然Springboot支持多种格式配置文件&#xff0c;但是在项目开发时&#xff0c;推荐使用一种格式的配置&#xff08;yml是主流&#xff09; Springboot除了支持…

【Osek网络管理测试】[TG3_TC3]tSleepRequestMin_L

&#x1f64b;‍♂️ 【Osek网络管理测试】系列&#x1f481;‍♂️点击跳转 文章目录 1.环境搭建2.测试目的3.测试步骤4.预期结果5.测试结果 1.环境搭建 硬件&#xff1a;VN1630 软件&#xff1a;CANoe 2.测试目的 验证DUT进入NMLimpHome状态后请求睡眠的最短时间是否正确…

周刊是聪明人筛选优质知识的聪明手段!

这是一个信息过载的时代&#xff0c;也是一个信息匮乏的时代。 这种矛盾的现象在 Python 编程语言上的表现非常明显。 它是常年高居编程语言排行榜的最流行语言之一&#xff0c;在国外发展得如火如荼&#xff0c;开发者、项目、文章、播客、会议活动等相关信息如海如潮。 但…

【LeetCode刷题记录】105. 从前序与中序遍历序列构造二叉树 106. 从中序与后序遍历序列构造二叉树

105 从前序与中序遍历序列构造二叉树 给定两个整数数组 preorder 和 inorder &#xff0c;其中 preorder 是二叉树的先序遍历&#xff0c; inorder 是同一棵树的中序遍历&#xff0c;请构造二叉树并返回其根节点。 示例 1: 输入: preorder [3,9,20,15,7], inorder [9,3,1…

近50亿元国资助阵,全球最大量子独角兽登场!

4月30日&#xff0c;澳大利亚与PsiQuantum公司宣布签订一项近10亿澳元&#xff08;约6.2亿美元、47.24亿人民币&#xff09;的协议&#xff0c;旨在建造世界上第一台商业上“有用”的量子计算机。 仅在一天前&#xff0c;澳大利亚还投资了1840万澳元&#xff0c;在悉尼大学成立…

【Osek网络管理测试】[TG3_TC5]等待总线睡眠状态_1

&#x1f64b;‍♂️ 【Osek网络管理测试】系列&#x1f481;‍♂️点击跳转 文章目录 1.环境搭建2.测试目的3.测试步骤4.预期结果5.测试结果 1.环境搭建 硬件&#xff1a;VN1630 软件&#xff1a;CANoe 2.测试目的 验证DUT在满足进入等待睡眠状态的条件时是否进入该状态 …

Linux学习(一)-- 简单的认识

目录 1. Linux的诞生 2.Linux发行版 拓展&#xff1a; &#xff08;1&#xff09;什么是Linux系统的内核&#xff1f; &#xff08;2&#xff09;什么是Linux系统发行版&#xff1f; 1. Linux的诞生 Linux创始人: 林纳斯 托瓦兹 Linux 诞生于1991年&#xff0c;作者上大学…

沃伦·巴菲特将AI比做原子弹:“瓶中精灵”使诈骗成为最快增长产业|TodayAI

在伯克希尔哈撒韦公司的年度股东大会上&#xff0c;投资大师沃伦巴菲特对人工智能的未来提出了严重警告。巴菲特对这项可以模拟现实并产生误导性内容的技术表示担忧&#xff0c;他认为这将成为史上增长最快的行业之一。 巴菲特在会上说&#xff1a;“当你思考诈骗人们的潜力时…

KernelSU 如何不通过模块,直接修改系统分区

刚刚看了术哥发的视频,发现kernelSU通过挂载OverlayFS实现无需模块,即可直接修改系统分区,很是方便,并且安全性也很高,于是便有了这篇文章。 下面的教程与原视频存在差异,建议观看原视频后再结合本文章进行操作。 在未进行修改前,我们打开/system/文件夹,并在里面创建…

【前端】输入时字符跳动动画实现

输入时字符跳动动画实现 在前端开发中&#xff0c;为了提升用户体验&#xff0c;我们经常需要为用户的交互行为提供即时的反馈。这不仅让用户知道他们的操作有了响应&#xff0c;还可以让整个界面看起来更加生动、有趣。本文将通过一个简单的例子讲解如何实现在用户输入字符时…

2分钟快速了解!全网最详细的性能测试教程之【Redis 简介和安装】

本篇文章主要介绍基于Redis的的简介和安装&#xff0c;其中参考了许多大佬写的文章&#xff0c;算是做一个Redis的基础教程吧。 Redis 简介 Redis 是完全开源的&#xff0c;遵守 BSD 协议&#xff0c;是一个高性能的 key-value 数据库。 Redis 与其他 key - value 缓存产品有…

SCI一区 | WOA-BiTCN-BiGRU-Attention多输入单输出回归预测(Matlab)

SCI一区 | WOA-BiTCN-BiGRU-Attention多输入单输出回归预测&#xff08;Matlab&#xff09; 目录 SCI一区 | WOA-BiTCN-BiGRU-Attention多输入单输出回归预测&#xff08;Matlab&#xff09;效果一览基本介绍程序设计参考资料 效果一览 基本介绍 1.Matlab实现WOA-BiTCN-BiGRU-A…

【C++】学习笔记——list

文章目录 八、list1. list的介绍2. list的使用3. list的模拟实现4. list模拟实现的代码整合1. list.h2. test.cpp 未完待续 八、list list链接 1. list的介绍 是的&#xff0c; list 就是带头双向循环链表。 2. list的使用 通过 string 和 vector 的学习&#xff0c;我们差…

Mybatis逆向工程笔记小结

&#x1f3f7;️个人主页&#xff1a;牵着猫散步的鼠鼠 &#x1f3f7;️系列专栏&#xff1a;Java全栈-专栏 &#x1f3f7;️个人学习笔记&#xff0c;若有缺误&#xff0c;欢迎评论区指正 目录 1.前言 2.实现方案 2.1. mybatis-generator生成 2.1.1. 环境说明 2.1.2. 数…