重庆分公司,新征程启航
为企业提供网站建设、域名注册、服务器等服务
/***4×4矩阵按键构架——火柴天堂作品-20110921***/
10年的思南网站建设经验,针对设计、前端、开发、售后、文案、推广等六对一服务,响应快,48小时及时工作处理。营销型网站建设的优势是能够根据用户设备显示端的尺寸不同,自动调整思南建站的显示方式,使网站能够适用不同显示终端,在浏览器中调整网站的宽度,无论在任何一种浏览器上浏览网站,都能展现优雅布局与设计,从而大程度地提升浏览体验。创新互联从事“思南网站设计”,“思南网站推广”以来,每个客户项目都认真落实执行。
/***源程序默认硬件环境:52单片机,12MHz晶振,P1口 4×4矩阵键盘,详细布局如下***/
/* 默认矩阵布局,按键扫描方式1使用
P0 P1 P2 P3
│ │ │ │
P4─┼──┼──┼──┤ S1 S2 S3 S4
│ │ │ │
P5─┼──┼──┼──┤ S5 S6 S7 S8
│ │ │ │
P6─┼──┼──┼──┤ S9 S10 S11 S12
│ │ │ │
P7─┴──┴──┴──┘ S13 S14 S15 S16
*/
#include"reg52.h" //包含52头文件
#define TRUE 1 //定义布尔量'1':真
#define FALSE 0 //定义布尔量'0':假
#define uchar unsigned char //定义 无符号字符型数据 简称
#define uint unsigned int //定义 无符号整型数据 简称
#define KeyPort P1
#define Key1Value 1
#define Key2Value 2
//uchar code KeyCodeList[16]={0xee,0xed,0xeb,0xe7,0xde,0xdd,0xeb,0xe7,0xbe,0xbd,0xbb,0xb7,0x7e,0x7d,0x7b,0x77};//按键代码列表,按键扫描方式2使用,可按需要随意修改顺序
uchar KeyScan() //按键扫描函数(方式1,需配合源程序矩阵布局,返回值0表示无按键,1-16为对应按键),缺点:无法扫描组合键(同时按2个按键以上)
{
uchar temp_h,temp_l,scan_value,i;
KeyPort=0xf0; //设置低4位为0(扫描线),准备读取高4位(返回线)状态
if(KeyPort==0xf0) return 0; //若高4位状态不变,表示无按键,返回无按键 键值0
temp_h=~KeyPort4; //若高4位状态改变,表示有按键,读取高4位,并将结果转成正逻辑(按键对应 行线 为 1 )存在temp_h低位上
KeyPort=0x0f; //设置高4位为0(扫描线),准备读取低4位(返回线)状态
temp_l=~(KeyPort|0xf0); //读取低4位,并将结果转成正逻辑(按键对应 列线 为 1 )存在temp_l低位上
while(i4) //将 按键行线号 转成数值
{
if((temp_hi)==0x01) break; //读取按键行线号
i++;
}
if(i==4) return 0; //若读取出错,返回无按键
temp_h=i; //将 按键行线数值 结果存于 temp_h
i=0;
while(i4) //将 按键列线号 转成数值
{
if((temp_li)==0x01) break; //读取按键列线号
i++;
}
if(i==4) return 0; //若读取出错,返回无按键
temp_l=i; //将 按键列线号 结果存于 temp_l
scan_value=(temp_h2)+temp_l+1; //合并行列线数值,并转成按键值,每条行线键值差为 4(temp_h2),按键值从1开始(+1,0为无按键 键值)
return scan_value; //返回 按键值
}
/*
uchar KeyScan() //按键扫描函数(方式2,需配合 按键代码列表 数组 进行 键值 查询,返回值0表示无按键,1-16为代码表顺序对应按键),缺点:无法扫描组合键
{
uchar temp,i;
KeyPort=0xf0; //设置低4位为0(扫描线),准备读取高4位(返回线)状态
if(KeyPort==0xf0) return 0; //若高4位状态不变,表示无按键,返回无按键 键值0
temp=KeyPort|0x0f; //若高4位状态改变,表示有按键,读取高4位,并将结果存于 temp 的高4位
KeyPort=0x0f; //设置高4位为0(扫描线),准备读取低4位(返回线)状态
temp=(KeyPort|0xf0); //读取低4位,并将结果存于 temp 的低4位
while(i16) //将按键行列线代码转换成键值
{
if(temp==KeyCodeList[i]) break; //将 行列线 代码与 按键代码表 进行对比,若一致则结束对比
i++; //进行下一个对比
}
if(i==16) return 0; //若查询出错,或 行列线代码 不在 按键代码表中,返回无按键 键值0
return i+1; //返回按键值1~16(按键代码列表中元素下标+1,0为无按键 键值)
}
*/
void KeyResp() //按键响应函数
{
static uchar KeyValue; //定义静态变量-按键值,
static bit KeyDownFlag,KeyReadyFlag; //定义静态标志位-按键按下标志,按键准备(响应)标志
uchar key_value=KeyScan(); //调用扫描函数,并将结果临时存放于key_value 中
if(key_value) //若扫描结果为真(即有按键)
{
KeyValue=key_value; //保存扫描结果
KeyDownFlag=TRUE; //按键按下标志 置位
KeyReadyFlag=TRUE; //按键准备(响应)标志 置位
}
else KeyDownFlag=FALSE; //若扫描结果为假(即无按键),则清空按键按下标志
if(KeyReadyFlag !KeyDownFlag) //若 按键已准备(响应),且无按键按下,(可知为 按下后又松手情况)
{
switch(KeyValue) //查找 按键值 对应的 按键处理
{
case Key1Value:break; //按键1处理
case Key2Value:break; //按键2处理
//case Key3Value:break;//......... //按键N处理
default:break; //无对应按键,或其它按键值处理
}
KeyValue=0; //清除按键值
KeyReadyFlag=FALSE; //清空 按键准备(响应)标志
}
}
void main() //主函数
{
while(1) //循环系统
{
KeyResp(); //调用 按键响应函数
}
}
#include stdlib.h
#define ROW 10
#define COL 10
int **matrix;
//动态申请空间
matrix = (int**) malloc(sizeof(int*) * ROW);
for(int i = 0; i COL; i++)
{
matrix[i] = (int*) malloc(sizeof(int) * COL);
}
//赋值
for(int i = 0; i ROW; i++)
{
for(int j = 0; j COL; i++)
{
*(*(matrix+i)+j) == 0;
}
}
方法1:直接定义10*10的数组,读取全部数据,再根据选择的行列数打印输出部分数据。
方法2:通过文件流指针的移动,跨过不需要的内容。(这里控制文件流指针可利用ftell和fseek函数来移动指定字节数)我是利用fscanf读取需要的内容,遇到需要跳行就用fgets来跳过。(注意常量参数根据自己需求修改)。
说明:边读取边打印输出,就不需要变量存储。我的代码是把读取出来的内容存放在三维数组中(文件内容你没指定,如是单纯数字,二维数组即可,我是作为字符串处理,故用三维数组。),数组是根据实际大小,动态申请,写成两个独立函数。
下面是代码:
#include stdio.h
#include string.h
#include malloc.h
#define MAXR 10//文件中矩阵最大行
#define MAXC 10//文件中矩阵最大列
#define MS 3//矩阵中元素字符串最大字符数(包含结束符号,不可小于实际内容)
#define MBUF 100//文件中一行最大字符数,可设置大一点
char ***getMem(int r,int c,int len);//动态申请一个三维数组
void freeMem(char ***datas,int r,int c);//释放数组空间
void showDatas(char ***datas,int r,int c);//打印输出结果
int main()
{
int r=MAXR+1,c=MAXC+1,i=0,j=0;
char ***datas=NULL,str[3],buf[MBUF];
FILE *fp=NULL;
printf("请输入要获取的行数和列数:\n");
while(r0 || rMAXR || c0 || cMAXC)scanf("%d%d",r,c);
datas=getMem(r,c,MS);
fp=fopen("C:\\test.data","r");
if(!fp) return 1;
while(fscanf(fp,"%s",str)!=-1)
{
if(ic) strcpy(datas[j][i],str),i++;
if(i==c)
{
if(!fgets(buf,sizeof(buf),fp)) break;
i=0,j++;
}
if(j==r) break;
}
showDatas(datas,r,c);
freeMem(datas,r,c);
return 0;
}
void showDatas(char ***datas,int r,int c)
{
int i,j;
for(i=0;ir;i++,printf("\n"))
for(j=0;jc;j++)
printf("%s ",datas[i][j]);
printf("\n");
}
void freeMem(char ***datas,int r,int c)
{
int i,j;
if(!datas)
{
for(i=0;ir;i++,free(datas[i]))
for(j=0;jc;j++)
free(datas[i][j]);
free(datas);
}
}
char ***getMem(int r,int c,int len)
{
int i,j;
char ***datas=NULL,**dr=NULL;
datas=(char ***)malloc(sizeof(char **)*r);
if(!datas) return NULL;
for(i=0;ir;i++)
{
dr=(char **)malloc(sizeof(char *)*c);
if(!dr) return NULL;
else
{
for(j=0;jc;j++)
{
dr[j]=(char *)malloc(sizeof(char)*len);
if(!dr[j]) return NULL;
}
datas[i]=dr;
}
}
return datas;
}
二维数组名不能直接传给二级指针,应该按以下方式使用:
int nChoose;
scanf("%d", nChoose); // 让用户输入二维数组的大小
int **a = (int **)malloc(nChoose * sizeof(int *));
for (int i = 0; i nChoose; i ++)
{
a[i] = (int *)malloc(nChoose * sizeof(int));
}
Scan(a, nChoose);
Calc(a, nChoose);
// 最后要释放数组,也要循环
//矩阵的赋值必须一个元素一个元素的赋值,如果不用循环,直接赋值,再没有规律也可以啊!至多代码长些。如:
int a[10][10];
a[0][0]=123;
a[0][1]=3567;
........
a[9][9]=76543;//这些赋值共100行。