C语言常见问题——数组初始化的四种方法

本文详细介绍了C语言中初始化数组的四种方法:初始化参数列表、使用memset函数、指定初始化器(GCC支持)和使用for循环。强调了初始化的重要性,分析了各种方法的优缺点,例如memset适合初始化字符数组,指定初始化器在某些编译器中不被支持,for循环初始化具有较好的移植性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

        有一回,小易同学对我说:“你学过编程吗”,我略略点一点头。他说,“学过编程,我便考一考你,在使用数组之前,应不应该初始化数组?

        我暗想,这么简单的问题,也配考我么?不只是数组,在使用任何变量之前,都应该对变量进行初始化,这是一个编程的好习惯,可以有效避免许多由于垃圾值导致的bug。于是我回过脸去,不再理会。

        小易同学等了许久也不见我回答,很诚恳的说道,“不知道罢?我教给你,记着!......数组的初始化有4种写法——”

目录

1.初始化参数列表

2.使用memset函数初始化(只建议用于字符数组)

3.指定初始化器(gcc支持)

4.使用for循环初始化

5.总结


1.初始化参数列表

全部初始化为0:

int a[10]={0};
char str[10]="\0";    //等价于char str[10]={0};

初始化为其他值:

int a[10]={0,1,2,3,4,5,6,7,8,9};
char str[10]="Hello";    //也可以写成char str[10]={'H','e','l','l','o','\0'}

如果初始化列表包含数组a的所有元素,可以省略数组的长度:

int a[]={0,1,2,3,4,5,6,7,8,9};

多维数组在计算机中也是线性存储的,因此下面两种写法等价:

int a[2][5]={{0,1,2,3,4},{5,6,7,8,9}};
//int a[2][5]={0,1,2,3,4,5,6,7,8,9};

        注意,如果你想把数组元素全部初始化为一个非零值(比如1),不能使用int a[10]={1}这种方式int a[10]={1}只会把a[0]初始化为1,而把其他元素初始化为0(C99标准下)。

2.使用memset函数初始化(只建议用于字符数组)

memset函数的定义包含在头文件string.h中,其函数原型如下:

void *memset(void *s, int c, unsigned long n);

//memset函数:将s指向的内存位置后面n个字节的存储空间替换为c

#include<stdio.h>
#include<string.h>
int main(){
	int i=0;
	char a[10];
	memset(a,'a',sizeof(a));//初始化

	for(i=0;i<10;i++)
		printf("%c ",a[i]);
	printf("\n");
	return 0;
}

 

注意memset通常只用于用于字符数组的初始化,因为memset替换内存空间的值时是以字节为单位的(char类型恰好只占1字节,而int类型一般占4字节)

如果你用memset来初始化int数组就会这样:

 因为(00000001 00000001 00000001 00000001)B=(16843009)D

 如果使用memset初始化int数组,初始值只能设为0或-1

#include<stdio.h>
#include<string.h>
int main(){
	int i=0;
	int a[10];

	memset(a,-1,sizeof(a));//全部替换为-1
	for(i=0;i<10;i++)
		printf("%d ",a[i]);
	printf("\n");

	memset(a,0,sizeof(a));//全部替换为0
	for(i=0;i<10;i++)
		printf("%d ",a[i]);
	printf("\n");
	return 0;
}

因为(00000000 00000000 00000000 00000000)B=(0)D,

(11111111 11111111 11111111 11111111)B=(-1)D

3.指定初始化器(gcc支持)

指定初始化器是C99标准新增的一个特性,形如int a[10]={[0 ... 9]=1}int a[10]={[1]=1,[2]=2,[9]=9},可以初始化指定的数组元素。

经笔者测试,只有gcc编译器(Codeblocks的.c文件就是用gcc编译的)支持该特性。vc++6.0以及g++编译器均不支持指定初始化器。

#include<stdio.h>
int main(){
	int i=0;
	int a[10]={[0 ... 9]=1};

	for(i=0;i<10;i++)
		printf("%d ",a[i]);
	printf("\n");
	return 0;
}


#include<stdio.h>
int main(){
	int i=0;
	int a[10]={[1]=1,[2]=2,[9]=9};

	for(i=0;i<10;i++)
		printf("%d ",a[i]);
	printf("\n");
	return 0;
}

4.使用for循环初始化

#include<stdio.h>
int main(){
	int i=0;
	int a[10];
	for(i=0;i<10;i++)
		a[i]=i;
	
	for(i=0;i<10;i++)
		printf("%d ",a[i]);
	printf("\n");
	return 0;
}

 

5.总结

        比较一下数组初始化的四种方法的优缺点:

  • 初始化参数列表:最常用的方法之一,尤其是初始化为0时非常方便。初始化速度较快;但是当需要把一个很长的数组初始化为非零值时比较麻烦。
  • 使用memset函数进行初始化:适合于char类型数组的初始化,速度较快;缺点是memset不适合初始化char类型以外的数组。
  • 指定初始化器:使用方便,初始化速度较快;缺点是很多编译器不支持该特性,并且C++的语法也不允许使用指定初始化器,可移植性较差。
  • for循环初始化:缺点是速度较慢;但是for循环的移植性较好,几乎适用于所有类型数组的初始化(可以考虑和初始化参数列表int a[10]={0}配合使用,并且release版本的for循环初始化经优化后速度并不慢)
### 大模型对齐微调DPO方法详解 #### DPO简介 直接偏好优化(Direct Preference Optimization, DPO)是一种用于改进大型语言模型行为的技术,该技术通过结合奖励模型训练和强化学习来提升训练效率与稳定性[^1]。 #### 实现机制 DPO的核心在于它能够依据人类反馈调整模型输出的概率分布。具体来说,当给定一对候选响应时,DPO试图使更受偏好的那个选项具有更高的生成概率。这种方法不仅简化了传统强化学习所需的复杂环境设置,而且显著增强了模型对于多样化指令的理解能力和执行精度[^2]。 #### PAI平台上的实践指南 为了便于开发者实施这一先进理念,在PAI-QuickStart框架下提供了详尽的操作手册。这份文档覆盖了从环境配置直至完成整个微调流程所需的一切细节,包括但不限于数据准备、参数设定以及性能评估等方面的内容。尤其值得注意的是,针对阿里云最新发布的开源LLM——Qwen2系列,文中给出了具体的实例说明,使得即使是初次接触此类工作的用户也能顺利上手。 ```python from transformers import AutoModelForCausalLM, Trainer, TrainingArguments model_name_or_path = "qwen-model-name" tokenizer_name = model_name_or_path training_args = TrainingArguments( output_dir="./results", per_device_train_batch_size=8, num_train_epochs=3, ) trainer = Trainer( model_init=lambda: AutoModelForCausalLM.from_pretrained(model_name_or_path), args=training_args, train_dataset=train_dataset, ) # 假设已经定义好了train_dataset trainer.train() ``` 这段代码片段展示了如何使用Hugging Face库加载预训练模型并对其进行微调的过程。虽然这里展示的例子并不完全对应于DPO的具体实现方式,但它提供了一个基础模板供进一步定制化开发之用[^3]。
评论 12
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

易水卷长空

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值