C语言进阶 ~ 一级指针与字符串

2023-05-16

目录

2.1 指针强化

2.2 一级指针(char*)易错地方

2.2.1 对空字符串和非法字符串的判断

2.2.2 越界

2.2.3 指针的叠加会不断改变指针的方向

2.2.4 局部变量不要外传

2.2.5 函数内使用辅助变量的重要性

2.3 字符串

2.3.1 字符串初始化

2.3.2 数组法和指针法操作字符串

2.3.3 字符串拷贝函数 strcpy

剖析片段程序

2.4 const

2.4.1 冒牌的 “const”

2.4.2 const 与 指针


2.1 指针强化

强化1:指针是一种数据类型

1)指针变量也是一种变量,占有内存空间,用来保存内存地址测试指针变量占有内存空间大小。

2)保证所指的内存块能修改

3)指针是一种数据类型,是指它指向的内存空间的数据类型

4 ) 不允许向NULL和未知非法地址拷贝内存。

	char *p3 = NULL;
	//给p3指向的内存区域拷贝内存
	strcpy(p3, "1111"); //err


	char *p3 = 0x0001;
	//给p3指向的内存区域拷贝内存
	strcpy(p3, "1111"); //err

强化2:间接赋值(*p)是指针存在的最大意义

强化3:理解指针必须和内存四区概念相结合

强化4:应用指针必须和函数调用相结合(指针做函数参数)

值得一看:

用1级指针形参,去间接修改了0级指针(实参)的值。

用2级指针形参,去间接修改了1级指针(实参)的值。

用3级指针形参,去间接修改了2级指针(实参)的值。

用n级指针形参,去间接修改了n-1级指针(实参)的值。

#include <stdio.h>

void fun(char **p , int *len)
{
	if (p == NULL)
	{
		return;
	}

	char *tmp = (char *)malloc(100);
	if (tmp == NULL)
	{
		return;
	}
	strcpy(tmp, "adlsgjldsk");

	//间接赋值
	*p = tmp;
	*len = strlen(tmp);
}

int main(void)
{

	char *p1 = NULL;
	int len1 = 0;
	fun(&p1, &len1);
	if (p1 != NULL)
	{
		printf("p1 = %s, len1 = %d\n", p1, len1);
	}

	printf("\n");
	system("pause");
	return 0;
}

① 程序运行至 char *tmp = (char *)malloc(100);

② 程序运行至 strcpy(tmp, "adlsgjldsk");

③ 程序运行至 *p = tmp;

④ 程序运行至 *len = strlen(tmp);

2.2 一级指针(char*)易错地方

2.2.1 对空字符串和非法字符串的判断

void copy_str(char *from, char *to)

{

    if (*from == '\0' || *to == '\0')  // 正确的为    if (from == '\0' || to == '\0')   地址是否为空

    {

        printf("func copy_str() err\n");

        return;

    }

    for (; *from!='\0'; from++, to++)

    {

        *to = *from;

    }

    *to = '\0';

}

2.2.2 越界

char buf[3] = "abc";

2.2.3 指针的叠加会不断改变指针的方向

char *getKeyByValue(char **keyvaluebuf, char *keybuf)

{

    int i = 0;

    char *a = (char *)malloc(50);

    for (; **keyvaluebuf != '\0'; i++)

    {

        *a++ = *(*keyvaluebuf)++;

    }          

    free(a);

}  

2.2.4 局部变量不要外传

char *my_stract(char *x, char* y)

{

    char str[80];

    char *z=str;           /*指针z指向数组str*/ 

    while(*z++=*x++);

    z--;                    /*去掉串尾结束标志*/

    while(*z++=*y++);

    z=str;                 /*str地址赋给指针变量z*/

    return(z);

}

2.2.5 函数内使用辅助变量的重要性

int getSubCount(char *str, char *substr, int *mycount)

{

    int ret = 0;

    char *p = str;   

    char *sub = substr;

    if (str==NULL || substr==NULL || mycount == NULL)

    {

        ret = -1;

        return ret;

    }

............

 

2.3 字符串

c语言没有字符串类型,通过字符数组模拟
c语言字符串,以字符‘\0’或数字0

2.3.1 字符串初始化

①  strlen:测字符串长度,不包含数字0,字符'\0'
      sizeof:测数组长度,包含数字0,字符'\0'

    char buf9[100] = "agjdslgjlsdjg";
    printf("strlen = %d, sizeof = %d\n", strlen(buf9), sizeof(buf9));

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

/*
  C语言没有字符串类型,通过字符数组模拟
  C语言字符串,以字符‘\0’, 数字0
*/
int main(void)
{

	//不指定长度, 没有0结束符,有多少个元素就有多长
	char buf[] = { 'a', 'b', 'c' };
	printf("buf = %s\n", buf);

	//指定长度,后面没有赋值的元素,自动补0
	char buf2[100] = { 'a', 'b', 'c' };
	printf("buf2 = %s\n", buf2);

	//所有元素赋值为0
	char buf3[100] = { 0 };
//

	//char buf4[2] = { '1', '2', '3' };  //数组越界

	char buf5[50] = { '1', 'a', 'b', '0', '7' };
	printf("buf5 = %s\n", buf5);

	char buf6[50] = { '1', 'a', 'b', 0, '7' };
	printf("buf6 = %s\n", buf6);

	char buf7[50] = { '1', 'a', 'b', '\0', '7' };
	printf("buf7 = %s\n", buf7);
//

	//使用字符串初始化,常用
	char buf8[] = "agjdslgjlsdjg";
//
	//strlen: 测字符串长度,不包含数字0,字符'\0'
	//sizeof:测数组长度,包含数字0,字符'\0'
	printf("strlen = %d, sizeof = %d\n", strlen(buf8), sizeof(buf8));
//
	char buf9[100] = "agjdslgjlsdjg";
	printf("strlen = %d, sizeof = %d\n", strlen(buf9), sizeof(buf9));

/

	printf("\n");
	system("pause");
	return 0;
}

2.3.2 数组法和指针法操作字符串

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

int main(void)
{
	char buf[] = "algjdlksajgldksjg";
	int i = 0;
	int n = strlen(buf);
	char *p = NULL;

	//[]方式
	for (i = 0; i < n; i++)
	{
		printf("%c", buf[i]);
	}
	printf("\n");

	//指针方法
	//数组名字,数组首元素地址
	p = buf;
//
	for (i = 0; i < n; i++)
	{
		printf("%c", p[i]);
	}
	printf("\n");
//
	for (i = 0; i < n; i++)
	{
		printf("%c", *(p+i) );
	}
	printf("\n");
//
	for (i = 0; i < n; i++)
	{
		printf("%c", *(buf+i) );
	}
	printf("\n");
//
	//buf和p完全等价吗?
	//p++;     正常
	//buf++;   此处出错
	//buf 只是一个常量,不能修改

//
	printf("\n");
	system("pause");
	return 0;
}

2.3.3 字符串拷贝函数 strcpy

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include <stdlib.h>
#include <string.h>


//成功为0,失败非0
//1、判断形参指针是否为NULL
//2、最好不要直接使用形参
int  my_strcpy(char *dst, char *src)
{
	if (dst == NULL || src == NULL)
	{
		return -1;
	}

	//辅助变量把形参接过来
	char *to = dst;
	char *from = src;

	//*to = *from
	//to++, from++
	//判断 *to是否为0, 为0跳出循环
	while (*to++ = *from++)
	{
		NULL;
	}

	printf("my_strcpy: dst = %s\n", dst);

	return 0;
}

int main(void)
{
	char src[] = "abcedfdgds";
	char dst[100] = { 0 };
	int ret = 0;


	ret = my_strcpy(dst, src);
	if (ret != 0)
	{
		printf("my_strcpy err:%d\n", ret);
		return ret;
	}
	printf("%s\n", dst);


	printf("\n");
	system("pause");
	return 0;
}

剖析片段程序


    while (*to++ = *from++)
    {
        NULL;
    }

① *to = *from

② to++, from++

③ 判断 *to是否为0, 为0跳出循环

2.4 const

2.4.1 冒牌的 “const”

const旨在用户层面不能修改,而一般在机器修改时候都是用的是地址进行改数。

    const int b = 10;
    //b = 100; //err
    int *q = &b;
    *q = 22;
    printf("%d, %d\n", b, *q);

2.4.2 const 与 指针

    指针变量, 指针指向的内存, 2个不同概念


	char buf[] = "aklgjdlsgjlkds";

    从左往右看,跳过类型,看修饰哪个字符

    const char *p = buf;
    // 等价于上面 char const *p1 = buf;

  •     如果是*, 说明指针指向的内存不能改变

    const char *p = buf;
    // 等价于上面 char const *p1 = buf;
    //p[1] = '2'; //err
    p = "agdlsjaglkdsajgl"; //ok

  •     如果是指针变量,说明指针的指向不能改变,指针的值不能修改

    char * const p2 = buf;
    p2[1] = '3';
    //p2 = "salkjgldsjaglk"; //err

  •    如果是指针变量和*, 指向不能变,指向的内存也不能变

//p3为只读,指向不能变,指向的内存也不能变
    const char * const p3 = buf;

 

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

C语言进阶 ~ 一级指针与字符串 的相关文章

  • Google浏览器视频倍速

    Google浏览器视频倍速 1 打开视频网页 xff1b 2 按下F12 xff1b 3 点击console xff1b 4 在框中粘贴代码 xff08 粘贴时 xff0c 鼠标放在箭头水平右侧或者用快捷键ctrl 43 v xff09 x
  • *** ERROR L127: UNRESOLVED EXTERNAL SYMBOL*** ERROR L128: REFERENCE MADE TO UNRESOLVED EXTERNAL

    把 Use extended linker instead of BL51 前面的 去掉就可以了 xff0c 就只有警告了 不知道为啥 xff0c 无意中试出来的 xff0c 编译生成的hex文件用了也没问题
  • 虚拟串口与串口调试助手的使用

    一 用到的软件 xff1a proteusVSPD VSPD是一款本地虚拟串口的软件 可以虚拟2个串口然后连接起来实现自发自收调试 xff0c 让你的程序读一个串口 xff0c 另外一个串口你就用来串口调试工具 二 串口通信步骤 打开VSP
  • keil5软件共用C51和ARM

    第一步 xff1a 创建文件夹keil mdk正常安装并破解MDK5 第二步 xff1a 创建另一个文件夹keil c51安装C51 将该文件夹里面的C51文件夹复制粘贴到keil mdk文件夹里与ARM文件夹保持同一目录 第三步 xff1
  • 【pytorch】Conv2d()里面的参数bias什么时候加,什么时候不加?

    代码中会发现有m 61 nn Conv2d 16 33 3 stride 61 2 bias 61 False bias是False xff0c 而默认的是True 因为一般为False的时候 xff0c nn Conv2d 后面通常接nn
  • Downloading https://ultralytics.com/assets/Arial.ttf to /data/..../.config/Ultralytics/Arial.ttf

    1 报错 xff1a 缺少字体Arial ttf 2 字体链接 xff1a https ultralytics com assets Arial ttf 3 方法 xff1a 下载该链接的字体 xff0c 然后放到 data config
  • 第四章 Opencv图像色彩空间与通道

    文章目录 1 色彩空间1 1 RGB BGR色彩空间1 2 GRAY色彩空间1 3 HSV色彩空间 2 通道2 1 拆分通道 xff1a 96 split 96 方法1 拆BGR色彩空间图像的通道2 拆HSV色彩空间图像的通道 2 2 合并
  • 第五章 Opencv图像的几何变换

    目录 1 缩放图像1 1 resize 方法 2 翻转图像2 1 flip 方法 3 仿射变换图像3 1 warpAffine 方法3 2 平移3 3 旋转3 4 倾斜 4 透视图像4 1 warpPerspective 方法 几何变换是指
  • pip、conda查看镜像源及更换镜像源

    1 查看已经安装过的镜像源 xff1a conda config show channels 查看配置项channels 2 删除镜像源 xff08 清华镜像源 xff09 xff1a conda config remove channel
  • 生成环境下的所有包

    pip freeze span class token operator gt span requirements span class token punctuation span txt 问题 xff1a 将虚拟环境的安装包导出 xff
  • java核心技术卷I

    第三章 xff1a java的基本程序设计结构 文章目录 第三章 xff1a java的基本程序设计结构3 2 注释3 3 数据类型3 4变量3 4 1初始化变量3 4 2常量 3 5运算符3 5 1数学函数与常量3 5 2数值类型之间的转
  • MOT学习笔记 — 行人检测及行人跟踪数据集总结

    1 行人红外数据集总结 xff08 1 xff09 OSU Thermal Pedestrian Database 下载链接 xff1a http vcipl okstate org pbvs bench Data 01 download
  • 使用k-近邻算法识别手写数字

    本文摘自 机器学习实战 案例 xff0c 对其进行了代码更新与注释 实战介绍 使用k 近邻分类器构造手写识别系统 xff0c 为了简单起见 xff0c 系统只识别0 9 xff0c 需要识别的数字已经使用图形处理软件 xff0c 处理成具有
  • ubuntu16.04下安装并使用小觅双目MYNT EYE 1.x SDK

    1 下载MYNT EYE 1 x SDK压缩包 首先 xff0c 点击进入github官网 xff0c 在右上角的搜索栏中输入mynt xff0c 进入如下界面 xff1a 点击第四个slightech MYNT EYE SDK进入 xff
  • UART通用异步收发传输器

    UART 全称Universal Asynchronous Receiver Transmitter xff0c 通用异步收发传输器 xff0c 是一种串行异步收发协议 又称为串口 xff09 功能是将并行的数据转变为串行的数据发送或者将接
  • C语言如何实现输入特定字符串(单词)作为终止符

    本文章以一个例题来进行讲解 xff08 新手第一次写 xff0c 目的仅是分享自己写代码中想到的一些方法和技巧 xff0c 仍存在很多不足 xff0c 希望能对大家有用 xff09 题目要求 xff1a 有一篇文章 xff0c 共有多行文字

随机推荐

  • kubernetes 教程 笔记

    K8s 安装kub ectl 下载kubectl curl LO 34 https dl k8s io release curl L s https dl k8s io release stable txt bin linux amd64
  • ros uwb2world坐标转换python示例

    ros uwb2world坐标转换python示例 span class token comment coding 61 utf 8 span span class token comment usr bin env python span
  • ARUCO marker的解释

    markers for ARUCO 一种汉明 海明 码的格子图 如图 百度百科解释汉明码规则概要 使用奇偶校验 具有一位纠错能力 校验位在2的次幂位置1 2 4 8 16 32 具体参看 https baike baidu com item
  • 使用ros_control ros_controllers 的牛刀真实驱动舵机手臂的源码

    现场 rqt graph 在一个陌生的框架下写代码 xff0c 免不了有很多疑问与槽点 不了解框架结构 xff0c 千头万续 xff0c 无从下手 xff0c 说不清 xff0c 理还乱 资料少没有文档 xff0c 要读懂程序猿的心 xff
  • 经典的pid公式,好脑子不如烂笔头。

    这个算法涉及昨天 xff0c 今天 xff0c 明天 思路就是以史为鉴 xff0c 预测明天 xff0c 改革当前
  • c++对8位灰度图进行二值化处理

    对灰度图进行位二值化 xff0c 输入图像像素部分的宽度和高度以及存储灰度像素值 得一维数组 xff0c 对灰度值进行直方图统计 xff0c 通过OSTU大律法公式 xff0c 确定自动灰度 图的阈值 xff0c 进而进行二值化处理 xff
  • vue 数组常用方法(总结)

    vue 数组常用方法 操作原数组push item pop shift unshift item n splice startIndex endIndex sort reverse 返回新数组slice startIndex endInde
  • 【亲测可用】kali linux 2020.1 设置为中文方法

    目录 0x00 提示0x01 更换更新源0x02 默认语言选择0x03 安装中文字体0x04 重启 xff0c 完成0x05 参考文章 kali 2020 1可用 进入我们的正题 xff0c 修改为中文的步骤 0x00 提示 由于kali
  • QT的TCP应用-传输图片

    1 server h span class token macro property span class token directive hash span span class token directive keyword ifnde
  • gazebo教程---使用roslaunch来启动gazebo,加载models

    1 使用roslaunch加载一个世界模型 roslaunch gazebo ros willowgarage world span class token punctuation span launch 运行效果如图 xff1a 下面看一
  • gazebo教程---ros_control

    一 ros control和Gazebo的数据流向 在Gazebo中模拟机器人的控制器是可以通过使用ros control和一个简单的Gazebo插件适配器来完成 下面是仿真 xff0c 硬件 xff0c 控制器和传动之间关系的概览 xff
  • CentOS Stream 安装 Docker

    版本LinuxCentOS Stream release 8 xff08 需要 CentOS 7 及以上 xff09 Docker20 10 17 卸载旧版本 旧版本的 Docker 被称为 docker 或 docker engine 如
  • CMakeLists.txt和.h头文件

    CMakeLists txt格式 xff08 随学习进度不断更新 xff09 声明要求的cmake最低版本 cmake minimum required VERSION 2 8 声明一个cmake工程 project HelloSLAM 添
  • 网络程序设计 面向TCP/IP编程总结

    第一章 网络编程基础知识 网络由节点和连线构成 现实用应用中的网络由硬件设备 xff08 路由器 交换机 网线 xff09 43 应用软件组成 计算机网路技术发展的第一个里程碑以报文或分组交换技术的出现为标志 数据交换的三种主要形式 xff
  • 训练时的Batchsize和Epoch之间的区别是什么?

    阅读这篇文章后 xff0c 你会知道 xff1a 随机梯度下降是一种迭代学习算法 xff0c 它使用训练数据集来更新模型 批量大小是梯度下降的超参数 xff0c 在模型的内部参数更新之前控制训练样本的数量 Epoch数是梯度下降的超参数 x
  • 如何在ROS下向ROS_PACKAGE_PATH中添加路径来解决找不到包的情况

    如果在创建ROS工作空间时不是严格按照 mkdir p catkin ws src 来创建的话可能后面会出现找不到包的情况 xff0c 这个时候你用命令 echo ROS PACKAGE PATH 会发现所找不到的包没有包含在这个路径里面
  • 移动平均法又称滑动平均法、滑动平均模型法(Moving average,MA)

    转自http jingji 100xuexi com view otdetail 20130625 230f09b0 6e36 473b 8830 7f2b873a5252 html 什么是移动平均法 移动平均法是用一组最近的实际数据值来预
  • C/C++ 数学库文件 (math.h)

    目录 1 三角函数 Trigonometric functions 1 1 cos 函数 1 2 sin 正弦函数 1 3 tan 正切函数 1 4 acos 反余弦函数 1 5 asin 反正弦函数 1 6 atan 反正切函数 1 7
  • C语言进阶 ~ 内存四区(栈、堆、全局、代码区)

    特别声明 xff1a 该部分是根据B站大佬 什么都想干好的视频学习而来 目录 1 1 数据类型本质分析 1 1 1 数据类型概念 1 1 2 数据类型的本质 1 1 3 数据类型的别名 1 1 4 数据类型之 void 1 2 变量的本质分
  • C语言进阶 ~ 一级指针与字符串

    目录 2 1 指针强化 2 2 一级指针 char 易错地方 2 2 1 对空字符串和非法字符串的判断 2 2 2 越界 2 2 3 指针的叠加会不断改变指针的方向 2 2 4 局部变量不要外传 2 2 5 函数内使用辅助变量的重要性 2