关于c语言的小问题

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了关于c语言的小问题相关的知识,希望对你有一定的参考价值。

157. main ( )
union int k; char i[2]; a, *s = &a ;
s -> i[0] = 0x39; s -> i[1] = 0x38; printf ( " %x \n ", s -> k ) ;

这个怎么看啊~输出值是什么?为什么?

107. int i;
main ( )
int i = 1, j = 2;
fun ( fun (i, &j ), &j );

fun ( int a, int *b )
static int m = 2;
i += m + a; m = ++( *b );
printf (" %d, %d ", i, m);
return ( m );

这个程序是怎么运算的啊,i是全局变量吗?那么在函数fun里面它的初值还是1吗?第二次m的值又是多少

44. 若有以下定义语句:int m[ ] = 5, 4, 3, 2, 1 ,i = 4;, 则下面对m数组元素的引用中错误的是:
A) m[ --i ] B) m [ 2*2 ] C) m [m[0] ] D) m [ m[i] ]
C和D的区别是什么啊?

不好意思,开始的时候看花眼了

157.该提输出值不确定,因为C++中规定,新声明的变量如果没有付初值则该变量的默认值将随机出现。如果在s->i[0]=0x39以前,先s->k=0,则将输出14393.
联合体的所占内存的大小等于其内所占内存最多的成员,因此,这个联合体的长度是4个字节。而k,i[0],i[1]共享这四个字节,其中i[0]为低八位,i[1]为次低八位,k占所有的四位。如果有s->k=0时内存空间中内容的二进制表示为32个0,而0x39=00111001,0x38=00111000,因此赋值后内存中的二进制表现是
00000000000000000011100100111000=十进制的14393
107.改程序考察三个知识点,第一个是,为函数传递参数的两种方式,传值和传地址的区别(还有一种是传引用,此处没考)。第二个是static修饰符的性质。第三个是++运算符的使用。
传值调用时函数在函数体中对形参的赋值不会改变实参的值,举个例子
fun(int i)

i=1;

main采用如下方式调用
main()

int n=0;
fun(n);

fun(n)运行后,main中n的值依旧为0
传地址调用会将地址传送给函数,对形参的改变会导致实参的值发生变化。比如:
fun(int *i)

*i=1;

main中采用上面相同的调用方式调用函数fun(n)后,n的值变为1。
函数中static修饰的变量声明语句只在第一次运行到该语句的时候有效,其他时间被忽略,函数结束后,对static修饰的变量所进行的修改值保留。
++运算符当放置在变量之前的时候表示先增值在引用,比如
int i=5;
int b=0;
b=++i;
则先i=i+1,然后b=i,即b=6
当++放置在变量后面时表示先引用在增值。比如
int i=5;
int b=0;
b=i++;
则先b=i,然后i=i+1,即b=5
当fun(fun(i,&j),&j)被调用的时候,首先括号内的fun(i,&j)先被运行,进入fun函数中后,形参a=1,b为j的地址,*b=2,static int m=2第一次运行,因此该语句有效,此时m=2。i+=m+a,即i=3,m=++(*b),首先*b=*b+1,然后m=*b,因此m=3,注意这里由于b的地址指向实参j的地址,因此实参j的值变为2,fun函数返回,返回值为3。回到main调用外层的fun函数又进入fun函数体,当执行到static int m=2时,由于前面已经调用过一次该语句,因此该语句被忽略,此时m为上一次运行后的结果,即m=3.形参a=3,b为j的地址,此时的值为2,运行i+=m+a,结果为6,m=++(*b)后结果为4,*b结果为4.
这里还隐藏着一个知识点,即++和*两个运算符的先后问题,例如:
int a=10;
int *b=a;
int c1=0,c2=0;
c1=++(*b);
c2=*(++b);
运行结果c1=11,那么c2等于多少呢?c2将为一个随机的数值,++b以后,b所指向的内存不再是a的地址,因此使用*求值会得到一个随机的值
44.语法上都没有错误。但C选项引用方式会报错,原因是m[0]=5,而数组m合法下标为0,1,2,3,4,因此m[m[0]]=m[5],下标溢出
参考技术A 晕,学了一学期c
竟还是看不懂你的题
union 公用体 老师说不考 s -> i[0] = 0x39 这个没看懂
fun ( fun (i, &j ), &j );函数前怎么没声明啊,还嵌套调用,i在fun函数中怎么没有初始值呢,i的作用域只在main函数。
参考技术B s -> i[0] = 0x39; s -> i[1] = 0x38; printf ( " %x \n ", s -> k ) ;
0x39是十六进制输出3*16+9=57

i是全局变量!i的初值在main中初始的在fun中也是1(最先)。

C中m[0]=5,则m[m[0]]即m[5],因为m的长度是5,所以m【】是从0开始的,所以m[]最大是4,
D中m[i]中i=4则m[i]=m[4]=1,所以m[m[i]]=m[1]=5是正确的引用
参考技术C 自己在电脑运行一下试试吧 一下子也说不明白的

c++编程时关于freopen的小问题

【问题描述】
平面上有N条直线,用方程Aix + Biy +Ci =0表示。这些直线没有三线共点的。现在要你计算出用这些直线可以构造出多少三角形?
【输入格式】
第1行:一个整数N(1 ≤ N≤ 300000)。
下面N行:每行3个整数:Ai, Bi 和Ci,表示对应直线方程的系数。不超过10^9.

【输出格式】
一行,一个整数。

input 1
6
0 1 0
-5 3 0
-5 -2 25
0 1 -3
0 1 -2
-4 -5 29
input 2
5
-5 3 0
-5 -3 -30
0 1 0
3 7 35
1 -2 -1
output 1
10
output 2
10

/////////////////////////////#include<cstdio>
#include<algorithm>
using namespace std;
struct edge

int a,b,c;
xian[300010];
bool cmp(edge x,edge y)

return 1LL*x.a*y.b>1LL*y.a*x.b;

long long ans=0;
int main()

// freopen("trokuti.in","r",stdin);
// freopen("trokuti.out","w",stdout);
int n;
scanf("%d",&n);
ans=1LL*(n)*(n-1)*(n-2)/6;
for(int i=1;i<=n;i++)

scanf("%d%d%d",&xian[i].a,&xian[i].b,&xian[i].c);


sort(xian+1,xian+n+1,cmp);
int l=1,r=1,len=1;
while(1)

if(1LL*xian[l].a*xian[r+1].b==1LL*xian[l].b*xian[r+1].a&&r<n)

len++;
r++;

if(r==n||1LL*xian[l].a*xian[r+1].b!=1LL*xian[l].b*xian[r+1].a)

if(len>=2)

ans-=1LL*len*(len-1)*(n-len)/2;

if(len>=3)

ans-=1LL*(len-1)*(len-2)*len/6;

len=0;
l=r+1;
r++;
if(l>n||r>n)

break;



printf("%lld",ans);

/////////////////////////////
代码如上,总之,去掉freopen两个样例是能过的,也能输出结果,问题加了这个freopen之后,程序会莫名的死机卡掉。以前都是加freopen编译运行后直接显示按任意键返回,这次直接显示等待输入,而且卡死无法输入,问题好像出现在while循环那里,注释掉while,freopen可以正常使用,但是没有freopen的时候程序明明是可以得出结果的。。这是为什么,从没遇到这种情况,求神犇不吝赐教!!!

参考技术A 我用的c++,试了试freopen,非常变扭,如果你要用文件输入输出,那就用fstream或者cstdio自带的FILE *追问

我学算法竞赛一直用这个啊。。。标程也用freopen。。今天突然出这个错我我也很绝望。。按理说不应该有这种事情..

追答

FILE和fstream都不允许使用吗?这两个文件输入还是很好用的,我的编译器不支持freopen,scanf,它说不安全,所以我也不知道怎么帮你

以上是关于关于c语言的小问题的主要内容,如果未能解决你的问题,请参考以下文章

c++编程时关于freopen的小问题

python-关于列表的小知识点

关于函数声明的小知识点

关于javascript的小问题

关于Go语言共享内存操作的小实例

请教一个C语言中的小问题,如图,那个L是啥意思呢?先写了哈