[从头学数学] 第137节 分式 小结与复习题

时间:2022-05-17 11:56:45
剧情提要:
[机器小伟]在[工程师阿伟]的陪同下进入了筑基中期的修炼,
这次要修炼的目标是[分式 小结与复习题]。

正剧开始:

星历2016年03月17日 11:46:50, 银河系厄尔斯星球*江南行省。
[工程师阿伟]正在和[机器小伟]一起研究[分式 小结与复习题]。

[从头学数学] 第137节 分式 小结与复习题

[从头学数学] 第137节 分式 小结与复习题

<span style="font-size:18px;">		var s = [
			'2s(s-2t)/(s+2t)',
			'1/(x^[2]-y^[2])',
			'2',
			'(u^[2]-4uv+4v^[2]-2)/(u^[2]-4v^[2])',
			'x^[-6]y^[-9]',
			'9x^[2]/(y^[6]z^[2])'
			
		
		
		];</span>

[从头学数学] 第137节 分式 小结与复习题


小伟想看一看这些分式有没有什么规律,于是随意画了几张图:

[从头学数学] 第137节 分式 小结与复习题

<span style="font-size:18px;">		//2(1)
		var result = 0;
		var array = [];
		var tmp = [];
		var colorArray = ['red', 'orange', '#0088FF', 'green', 'cyan', 'blue', '#FF00FF',
			'#888888', 'black'];
			
		var count = 0;
		for (var s = -10; s < 10; s+=2) {
			array = [];
			for (var t = -10; t < 10; t+=2) {
				result = (s-2*t)/(3*s)*(6*Math.pow(s, 2)/(s+2*t));
				
				array.push(shape.point3D(s, result/20, t));
			}
			tmp = [].concat(array);
			shape.multiLineDraw(tmp, colorArray[count%8], r);
			tmp = [].concat(array);
			shape.pointDraw(tmp, colorArray[count%8], r);
			count++;</span>


[从头学数学] 第137节 分式 小结与复习题

<span style="font-size:18px;">		//2(2)
		var result = 0;
		var array = [];
		var tmp = [];
		var colorArray = ['red', 'orange', '#0088FF', 'green', 'cyan', 'blue', '#FF00FF',
			'#888888', 'black'];
			
		var count = 0;
		for (var z = -10; z < 10; z+=2) {
			array = [];
			for (var x = -10; x < 10; x+=2) {
				//x-->x z-->y
				result = (x-z)/(x+z)+Math.pow(x-z, 2);
				
				array.push(shape.point3D(x, result/10-5, z));
			}
			tmp = [].concat(array);
			shape.multiLineDraw(tmp, colorArray[count%8], r);
			tmp = [].concat(array);
			shape.pointDraw(tmp, colorArray[count%8], r);
			count++;
			
				
				
		}</span>

[从头学数学] 第137节 分式 小结与复习题

<span style="font-size:18px;">		//2(4)
		var result = 0;
		var array = [];
		var tmp = [];
		var colorArray = ['red', 'orange', '#0088FF', 'green', 'cyan', 'blue', '#FF00FF',
			'#888888', 'black'];
			
		var count = 0;
		var u , v;
		for (var z = -10; z < 10; z+=2) {
			array = [];
			for (var x = -10; x < 10; x+=2) {
				//x-->u z-->v
				u = x;
				v = z;
				result = (u-2*v)/(u+2*v)-2/(u*u-4*v*v);
				
				array.push(shape.point3D(x, result, z));
			}
			tmp = [].concat(array);
			shape.multiLineDraw(tmp, colorArray[count%8], r);
			tmp = [].concat(array);
			shape.pointDraw(tmp, colorArray[count%8], r);
			count++;
			
				
				
		}</span>

这图可够乱的,小伟完全瞧不出什么规律来,算了,不管它了。


[从头学数学] 第137节 分式 小结与复习题

<span style="font-size:18px;">		var s = [
			'6',
			'b^[-1]',
			'(2x^[2]-12x+16)/(x^[2]-16)',
			'1/16p^[2]q^[3]r^[-1]+1/(2q)',
			'x/(1+x^[2])',
			'1/(a-b)',
			'-b/(a+b)',
			'(x+y)/(x-y)']</span>


[从头学数学] 第137节 分式 小结与复习题

小伟为了做这些非常烦的题,专门又给以前的多项式添加了一个计算两个多项式相乘的工具:

<span style="font-size:18px;">###
# @usage   两个多项式相乘
# @author  mw
# @date    2016年03月17日  星期四  14:47:14 
# @param   限制很多,但对于项数比较多的一层括号,可省不少麻烦
# @return
#
###
#多项式s1与s2的乘积,s1, s2都不能有括号
#它们的单项之间用+号或-号相连
#例如a+2b2c3, 2a+4b-c3-1等
#代数字母与次数之间不要有其它符号,不要有乘号和除号
def polymul(s1, s2):
    if (s1 == '' or s2 == ''):
        return;
    
    #只能有+或-号连接各项,不能有括号,分数要先化成小数
    s = s1.replace('-', '+-');
    if (s[0] == '+'):
        s = s[1:];
    #各项
    terms_1 = s.split('+');

    s = s2.replace('-', '+-');
    if (s[0] == '+'):
        s = s[1:];
    #各项
    terms_2 = s.split('+');
    
    len1 = len(terms_1);
    len2 = len(terms_2);

    resultStr = '';

    for i in range(len1):
        for j in range(len2):
            resultStr += monomul(terms_1[i], terms_2[j])+'+';

    lens = len(resultStr);
    if (resultStr[lens-1] == '+'):
        resultStr = resultStr[:lens-1];

    return resultStr;

def monomul(s1, s2):
    a1 = monoDecomposite(s1);
    a2 = monoDecomposite(s2);

    coeff = eval('('+a1[0]+')'+'*'+'('+a2[0]+')');
    return str(coeff)+a1[1]+a2[1];
    

#分解出单项式的系数和其余部分 
def monoDecomposite(s):
    if (s == ''):
        return ['0', ''];

    if (s[0].isalpha()):
        return ['1', s];

    lens = len(s);
    if (s[0] == '-'):       
        if (s[1].isalpha()):
            return ['-1', s[1:]];
        else:
            index = 1;
            while (index <= lens-1 and (not s[index].isalpha())):
                index+=1;
            return [s[:index], s[index:]];
    else:
        index = 0;
        while (index <= lens-1 and (not s[index].isalpha())):
            index+=1;
        return [s[:index], s[index:]];</span>

加上以前做的:

<span style="font-size:18px;">###
# @usage   单项式相关概念
# @author  mw
# @date    2016年02月26日  星期五  10:00:14 
# @param
# @return
#
###
#单项式
#可以有**, ^号,暂时只能处理代号为一个字母的式子,像x_1, x_n, ...这种还不能处理。
def monomial(s):
    #原始复本
    s0 = s;
    
    s = s.replace('**', '^');
    s = s.replace('*', '');

    if (s.find('+') != -1 or (s.find('-') != -1 and s.find('-')!=0)):
        print(s0, '不是单项式。');
        return [];
        
    try:
        #系数
        sign = 1;
        if s[0] == '-':
            #负号
            sign = -1;
            s = s[1:];
        coefficient = 1;

        #字符串长度
        length = len(s);
        index = 0;
        while (not s[index].isalpha()):
            index+=1;
            if index >= length:
                index = length;
                break;

        if (index > 0):
            coefficient = float(s[:index]);
            if abs(int(coefficient)-coefficient) < 0.001:
                coefficient = int(coefficient);        
            s = s[index:];
            
        coefficient = sign * coefficient;
        #系数为0的项,其实就是0
        if (coefficient == 0):
            return [];
        #print(coefficient, s);

        length = len(s);
        array = [];
        result = [];
        if (length > 0):
            index = 0;
            index2 = 0;
            name = '';
            degree = 0;
            while index < length:        
                if s[index].isalpha():
                    if (name != ''):
                        array.append([name, degree]);
                        name = '';
                        degree = 0;
                    name = s[index];
                    degree = 1;
                    index += 1;
                else:
                    index2 = index;
                    tmp = '';
                    while (not s[index2].isalpha()):
                        index2+=1;
                        if (index2 >= length):
                            index2 = length;
                            break;
                    tmp = s[index:index2];
                    tmp = tmp.replace('^', '');
                    degree = float(tmp);
                    if abs(int(degree)-degree) < 0.001:
                        degree = int(degree);

                    index = index2;
            if (name != ''):
                array.append([name, degree]);
                name = '';
                degree = 0;
                
            #print(array);

            #所有字母,去除重复的
            setA = set();
            size = len(array);
            #单项式的次数
            totalDegree = 0;
            for i in range(size):
                setA.add(array[i][0]);
                totalDegree += array[i][1];

            listA = list(setA);
            listA.sort();
            size2 = len(listA);
            result = [];
            for i in range(size2):
                #计算每个字母的次数(degree)
                tmp = 0;
                for j in range(size):
                    if listA[i] == array[j][0]:
                        tmp += array[j][1];

                result.append([listA[i], tmp]);

            print('单项式{0} 的系数是{1}, 次数是{2},详细是{3}。'.format(\
                s0, coefficient, totalDegree, result));
        else:
            totalDegree = 0;
            if (coefficient != 0):
                print('单项式{0} 的系数是{1}, 次数是{2},'.format(\
                    s0, coefficient, 0));
            else:
                print('这个数是0, 暂无规定。');
        #返回单项式的次数
        return [coefficient, totalDegree, result];
    except:
        print(s0, '有误,无法正确计算。');

###
# @usage   多项式相关概念
# @author  mw
# @date    2016年02月26日  星期五  10:30:37 
# @param   如果有括号,需要先自行去除,
# @return
#
###
def polynomial(s):
    if (s == ''):
        return;
    
    #预留复本
    s0 = s;

    #只能有+或-号连接各项,不能有括号,分数要先化成小数
    s = s.replace('-', '+-');
    if (s[0] == '+'):
        s = s[1:];
    #各项
    terms = s.split('+');
    

    print('多项式{0} 具有以下的项: {1}\n其中各单项分别是:'.format(s0, terms));

    try:
        size = len(terms);
        array = [];
        for i in range(size):
            #此处也可扩展单项的合法性检查。
            if (terms[i] == ''):
                pass;
            else:
                tmp = monomial(terms[i]);
                #对于返回[]的项,剔除掉
                if len(tmp)>0:
                    array.append(tmp);
        #print(array);

        size2 = len(array);

        for i in range(size2):
            #判断系数是否是0
            if array[i][0] == 0:
                continue;
            for j in range(i+1, size2):
                if array[j][0] == 0:
                    continue;
                else:
                    if (sameTerm(array[i], array[j])):
                        #合并同类项
                        array[i][0]+=array[j][0];
                        array[j][0] = 0;

        result = [];
        for i in range(size2):
            #判断系数是否是0
            if array[i][0] == 0:
                continue;
            else:
                #保留三位小数
                array[i][0] = round(array[i][0], 6);
                result.append(array[i]);

        print('合并同类项后详细情况是:', result);
        sResult = '';
        size3 = len(result);
        if (size3)>0:
            for i in range(size3):
                tmp1 = result[i][0];
                if (tmp1 >= 0):
                    if (tmp1 != 1):
                        sResult += '+'+str(tmp1);
                    else:
                        sResult += '+';
                if (tmp1 < 0):
                    if (tmp1!=-1):
                        sResult += str(tmp1);
                    else:
                        sResult += '-';

                tmp2 = result[i][2];
                length = len(tmp2);
                if (length == 0 and abs(tmp1)==1):
                    sResult += '1';
                else:
                    for j in range(length):
                        if tmp2[j][1] != 1:
                            sResult += tmp2[j][0]+str(tmp2[j][1]);
                        else:
                            sResult += tmp2[j][0];
            if sResult[0] == '+':
                sResult = sResult[1:];
        else:
            #所有项的系数刚好抵消,导致结果为0
            sResult = '0';
        print('合并同类项后是 => {0}\n\n'.format(sResult));
    except:
        print(s0, '有误,无法进行多项式操作。');
    return;

                

#同类项判断
def sameTerm(a, b):
    #由于a, b具有以下格式[1, 1, [['v', 1]]] [系数, 次数, 详细元素]
    if (a[1] == 0 and b[1] == 0):
        return True;
    
    if (a[1] != b[1]):
        #次数不同
        return False;
    if (len(a[2]) != len(b[2])):
        #元素个数不同
        return False;

    a1 = list(a[2]);
    b1 = list(b[2]);
    a1 = sorted(a1, key=lambda num:num[0]);
    b1 = sorted(b1, key=lambda num:num[0]);
    size = len(a1);
    for i in range(size):
        if a1[i][0] != b1[i][0] or a1[i][1] != b1[i][1]:
            return False;

    return True;

###
# @usage   代数式,可以对系数中含计算式的情况进行计算,得出简化后的多项式字符串
# @author  mw
# @date    2016年02月27日  星期六  09:23:44 
# @param
# @return
#
###
#代数式
def algExpr(s):
    s0 = s;

    #判断字符串中左右括号是否匹配
    if s.count('(') - s.count(')') != 0:
        print(s, '左右括号数目不匹配,表达式有误。');
        return '';
    #去除空格
    s = s.replace(' ', '');
    #存放需要计算的表达式子串
    sub = '';
    length = len(s);
    #新字符串
    sNew = '';
    #括号层次
    bracket = 0;

    result = [];

    #需要计算的部分,是从每一个单项的开始处,一般只需要计算系数
    need = 1;
    #精度
    precision = 6;
    
    #遍历字符串s
    i = 0;
    while i < length:
        if (s[i] == '('):
            bracket+=1;
            sub += s[i];
        elif (s[i] == ')'):
            bracket -=1;
            sub += s[i];                
        elif (s[i].isalpha()):
            if sub != '':
                if (sub[-1]=='*'):
                    sub=sub[:-1];
                sNew += str(round(eval(sub), precision))+ s[i];
                sub = '';
            else:
                sNew += s[i];
            #由于字母后面是次数,不需要计算
            #need = 0;            
        elif (s[i] == '+' or s[i] == '-'):
            if (bracket == 0):
                if sub != '':
                    if (sub[-1]=='*'):
                        sub=sub[:-1];
                    sNew += str(round(eval(sub), precision))+ s[i];
                    sub = '';
                else:
                    sNew += s[i];
                #need = 1;
            else:
                sub += s[i];
        else:
            if (need == 1):
                sub += s[i];
            else:
                sNew += s[i];

        i += 1;

    if sub != '':
        if (sub[-1]=='*'):
            sub=sub[:-1];
        sNew += str(round(eval(sub), precision));
        sub = '';

    #print(sNew);
    return sNew;</span>

这么多,反而越来越难以解答[人叫板老师]的题了,难道非要阿伟长出两只手来拿个草稿本慢慢算?


[从头学数学] 第137节 分式 小结与复习题

<span style="font-size:18px;">if __name__ == '__main__':
    s1 = polymul('5x+2', 'x+1');
    s2 = polymul('-3', 'x2+x');
    s = algExpr(s1+'+'+s2);
    print(polynomial(s));


多项式5xx+5x+2x+2+-3x2+-3x 具有以下的项: ['5xx', '5x', '2x', '2', '', '-3x2', '', '-3x']
其中各单项分别是:
单项式5xx 的系数是5, 次数是2,详细是[['x', 2]]。
单项式5x 的系数是5, 次数是1,详细是[['x', 1]]。
单项式2x 的系数是2, 次数是1,详细是[['x', 1]]。
单项式2 的系数是2, 次数是0,
单项式-3x2 的系数是-3, 次数是2,详细是[['x', 2]]。
单项式-3x 的系数是-3, 次数是1,详细是[['x', 1]]。
合并同类项后详细情况是: [[2, 2, [['x', 2]]], [4, 1, [['x', 1]]], [2, 0, []]]
合并同类项后是 => 2x2+4x+2

=> x = -1</span>

x = -1时分母为0了,所以这个方程无解!


<span style="font-size:18px;">if __name__ == '__main__':
    s1 = polymul('2x', '2x+5');
    s2 = polymul('-2', '2x-5');
    s3 = polymul('-2x+5', '2x+5');
    s = algExpr(s1+'+'+s2+'+'+s3);
    print(polynomial(s));

>>> 
多项式4xx+10x+-4x+10+-4xx+-10x+10x+25 具有以下的项: ['4xx', '10x', '', '-4x', '10', '', '-4xx', '', '-10x', '10x', '25']
其中各单项分别是:
单项式4xx 的系数是4, 次数是2,详细是[['x', 2]]。
单项式10x 的系数是10, 次数是1,详细是[['x', 1]]。
单项式-4x 的系数是-4, 次数是1,详细是[['x', 1]]。
单项式10 的系数是10, 次数是0,
单项式-4xx 的系数是-4, 次数是2,详细是[['x', 2]]。
单项式-10x 的系数是-10, 次数是1,详细是[['x', 1]]。
单项式10x 的系数是10, 次数是1,详细是[['x', 1]]。
单项式25 的系数是25, 次数是0,
合并同类项后详细情况是: [[6, 1, [['x', 1]]], [35, 0, []]]
合并同类项后是 => 6x+35


None
>>> 2*(-35/6)/(2*(-35/6)-5)-2/(2*(-35/6)+5)
1.0
</span>

验算通过,这个解是x = -35/6


[从头学数学] 第137节 分式 小结与复习题

[从头学数学] 第137节 分式 小结与复习题

[从头学数学] 第137节 分式 小结与复习题

[从头学数学] 第137节 分式 小结与复习题

[从头学数学] 第137节 分式 小结与复习题

[从头学数学] 第137节 分式 小结与复习题


好吧,这些烦琐的分式计算,还是留给需要计算它们的小朋友吧。


小伟在这里总结一下工具的使用:

<span style="font-size:18px;">if __name__ == '__main__':
    #可计算多项式的乘法,但不能有括号
    s1 = polymul('x+y+z+u+v', 'x+y+z-u-v');
    print(polynomial(s1));

    #指数上有负号时会出错,这个可以解决,但是会要求更复杂的写法
    s2 = '(5*2*3-8-6)a(-2)a2'
    print(polynomial(algExpr(s2)));

    #对于正指数会很方便
    s3 = '(5*2*3-8-6)a2abcdef2g+s(2*3)d3-28.66gdf-3';
    print(polynomial(algExpr(s3)));


</span>

<span style="font-size:18px;">>>> 
多项式1xx+1xy+1xz+-1xu+-1xv+1yx+1yy+1yz+-1yu+-1yv+1zx+1zy+1zz+-1zu+-1zv+1ux+1uy+1uz+-1uu+-1uv+1vx+1vy+1vz+-1vu+-1vv 具有以下的项: ['1xx', '1xy', '1xz', '', '-1xu', '', '-1xv', '1yx', '1yy', '1yz', '', '-1yu', '', '-1yv', '1zx', '1zy', '1zz', '', '-1zu', '', '-1zv', '1ux', '1uy', '1uz', '', '-1uu', '', '-1uv', '1vx', '1vy', '1vz', '', '-1vu', '', '-1vv']
其中各单项分别是:
单项式1xx 的系数是1, 次数是2,详细是[['x', 2]]。
单项式1xy 的系数是1, 次数是2,详细是[['x', 1], ['y', 1]]。
单项式1xz 的系数是1, 次数是2,详细是[['x', 1], ['z', 1]]。
单项式-1xu 的系数是-1, 次数是2,详细是[['u', 1], ['x', 1]]。
单项式-1xv 的系数是-1, 次数是2,详细是[['v', 1], ['x', 1]]。
单项式1yx 的系数是1, 次数是2,详细是[['x', 1], ['y', 1]]。
单项式1yy 的系数是1, 次数是2,详细是[['y', 2]]。
单项式1yz 的系数是1, 次数是2,详细是[['y', 1], ['z', 1]]。
单项式-1yu 的系数是-1, 次数是2,详细是[['u', 1], ['y', 1]]。
单项式-1yv 的系数是-1, 次数是2,详细是[['v', 1], ['y', 1]]。
单项式1zx 的系数是1, 次数是2,详细是[['x', 1], ['z', 1]]。
单项式1zy 的系数是1, 次数是2,详细是[['y', 1], ['z', 1]]。
单项式1zz 的系数是1, 次数是2,详细是[['z', 2]]。
单项式-1zu 的系数是-1, 次数是2,详细是[['u', 1], ['z', 1]]。
单项式-1zv 的系数是-1, 次数是2,详细是[['v', 1], ['z', 1]]。
单项式1ux 的系数是1, 次数是2,详细是[['u', 1], ['x', 1]]。
单项式1uy 的系数是1, 次数是2,详细是[['u', 1], ['y', 1]]。
单项式1uz 的系数是1, 次数是2,详细是[['u', 1], ['z', 1]]。
单项式-1uu 的系数是-1, 次数是2,详细是[['u', 2]]。
单项式-1uv 的系数是-1, 次数是2,详细是[['u', 1], ['v', 1]]。
单项式1vx 的系数是1, 次数是2,详细是[['v', 1], ['x', 1]]。
单项式1vy 的系数是1, 次数是2,详细是[['v', 1], ['y', 1]]。
单项式1vz 的系数是1, 次数是2,详细是[['v', 1], ['z', 1]]。
单项式-1vu 的系数是-1, 次数是2,详细是[['u', 1], ['v', 1]]。
单项式-1vv 的系数是-1, 次数是2,详细是[['v', 2]]。
合并同类项后详细情况是: [[1, 2, [['x', 2]]], [2, 2, [['x', 1], ['y', 1]]], [2, 2, [['x', 1], ['z', 1]]], [1, 2, [['y', 2]]], [2, 2, [['y', 1], ['z', 1]]], [1, 2, [['z', 2]]], [-1, 2, [['u', 2]]], [-2, 2, [['u', 1], ['v', 1]]], [-1, 2, [['v', 2]]]]
合并同类项后是 => x2+2xy+2xz+y2+2yz+z2-u2-2uv-v2


None
多项式16a-2a2 具有以下的项: ['16a', '-2a2']
其中各单项分别是:
单项式16a 的系数是16, 次数是1,详细是[['a', 1]]。
单项式-2a2 的系数是-2, 次数是2,详细是[['a', 2]]。
合并同类项后详细情况是: [[16, 1, [['a', 1]]], [-2, 2, [['a', 2]]]]
合并同类项后是 => 16a-2a2


None
多项式16a2abcdef2g+s6d3-28.66gdf-3 具有以下的项: ['16a2abcdef2g', 's6d3', '-28.66gdf', '-3']
其中各单项分别是:
单项式16a2abcdef2g 的系数是16, 次数是10,详细是[['a', 3], ['b', 1], ['c', 1], ['d', 1], ['e', 1], ['f', 2], ['g', 1]]。
单项式s6d3 的系数是1, 次数是9,详细是[['d', 3], ['s', 6]]。
单项式-28.66gdf 的系数是-28.66, 次数是3,详细是[['d', 1], ['f', 1], ['g', 1]]。
单项式-3 的系数是-3, 次数是0,
合并同类项后详细情况是: [[16, 10, [['a', 3], ['b', 1], ['c', 1], ['d', 1], ['e', 1], ['f', 2], ['g', 1]]], [1, 9, [['d', 3], ['s', 6]]], [-28.66, 3, [['d', 1], ['f', 1], ['g', 1]]], [-3, 0, []]]
合并同类项后是 => 16a3bcdef2g+d3s6-28.66dfg-3


None
>>> </span>

反正是烦琐透了。还不如用草稿本划同类项来得快呢。


本节到此结束,欲知后事如何,请看下回分解。