QT开发(三十)——计算器实例开发

QT开发(三十)——计算器实例开发

一、计算器界面制作

计算器界面需要QWidget组件作为顶层窗口,QLineEdit组件作为输入框,QPsuhButton作为按钮。

界面规划设计如下:

#include <QApplication>
#include <QWidget>
#include <QLineEdit>
#include <QPushButton>
 
int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    QWidget *widget = new QWidget(NULL, Qt::WindowCloseButtonHint);
    //构建输入框,设置属性
    QLineEdit *edit = new QLineEdit(widget);
    edit->move(10, 10);
    edit->resize(240, 30);
    edit->setReadOnly(true);
    //构造按钮,设置属性
    QPushButton *button[20] = {0};
    const char *buttontext[20] =
    {
        "7", "8", "9", "+", "(",
        "4", "5", "6", "-", ")",
        "1", "2", "3", "*", "<-",
        "0", ".", "=", "/", "C"
    };
    for(int i = 0; i < 4; i++)
    {
        for(int j = 0; j < 5; j++)
        {
            button[5*i + j] = new QPushButton(widget);
            button[5*i + j]->move(10 + (10 + 40)*j, 50 + (10 + 40)*i);
            button[5*i + j]->resize(40, 40);
            button[5*i + j]->setText(buttontext[5*i + j]);
        }
    }
    //设置窗口
    int ret = 0;
    widget->show();
    widget->setFixedSize(widget->width(), widget->height());
 
    ret = a.exec();
    delete widget;
    return ret;
}

二、项目代码重构

重构是以改善代码质量为目的的代码重写,使得软件的设计和架构更加合理,提高了软件的扩展性和维护性。

代码重构与代码实现的区别:

A、代码实现是按照设计编程实现,核心在于功能实现,不考虑架构的优劣

B、代码重构是以提高代码质量为目的的软件架构优化,核心在于优化架构,不考虑对已实现功能的修改。

代码重构在软件开发过程中的阶段:

代码重构的适用场合:

A、项目中重复代码越来越多

B、项目中代码功能越来越不清晰

C、项目中代码实现离设计越来越远

计算器界面代码重构:

由于需要申请堆空间资源,使用二阶构造模式。

CalculatorUI.h文件:

#ifndef CALCULATORUI_H
#define CALCULATORUI_H
 
#include <QWidget>
#include <QLineEdit>
#include <QPushButton>
 
class CalculatorUI : public QWidget
{
    Q_OBJECT
public:
    static CalculatorUI* newInstance();
    ~CalculatorUI();
    void show();
private:
    CalculatorUI();
    bool Construct();
 
private:
    QLineEdit *m_edit;
    QPushButton *m_buttons[20];
};
 
#endif // CALCULATORUI_H

CalculatorUI.cpp文件:

#include "CalculatorUI.h"
 
CalculatorUI::CalculatorUI() : QWidget(NULL, Qt::WindowCloseButtonHint)
{
}
 
CalculatorUI* CalculatorUI::newInstance()
{
    CalculatorUI *ret = new CalculatorUI();
    if(NULL == ret || !ret->Construct())
    {
        delete ret;
        ret = NULL;
    }
    return ret;
}
 
CalculatorUI::~CalculatorUI()
{
}
 
void CalculatorUI::show()
{
    QWidget::show();
    setFixedSize(width(), height());
}
 
bool CalculatorUI::Construct()
{
    bool ret = true;
    m_edit = new QLineEdit(this);
    if(m_edit != NULL)
    {
        m_edit->move(10, 10);
        m_edit->resize(240, 30);
        m_edit->setReadOnly(true);
    }
    else
    {
        ret = false;
    }
 
    const char *buttontext[20] =
    {
        "7", "8", "9", "+", "(",
        "4", "5", "6", "-", ")",
        "1", "2", "3", "*", "<-",
        "0", ".", "=", "/", "C"
    };
    for(int i = 0; (i < 4) && ret; i++)
    {
        for(int j = 0; (j < 5) && ret; j++)
        {
            m_buttons[5*i + j] = new QPushButton(this);
            if(m_buttons[5*i + j] != NULL)
            {
                m_buttons[5*i + j]->move(10 + (10 + 40)*j, 50 + (10 + 40)*i);
                m_buttons[5*i + j]->resize(40, 40);
                m_buttons[5*i + j]->setText(buttontext[5*i + j]);
            }
            else
            {
                ret = false;
            }
        }
    }
    return ret;
}

Main.cpp文件:

#include <QApplication>
#include <QWidget>
#include <QLineEdit>
#include <QPushButton>
#include "CalculatorUI.h"
 
int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    CalculatorUI *widget = CalculatorUI::newInstance();
    int ret = 0;
    if(widget != NULL)
    {
        widget->show();
        ret = a.exec();
        delete widget;
    }
 
    return ret;
}

三、计算器算法引擎

人类思维和阅读习惯的运算表达式为中缀表达式,但计算机的运算表达式为后缀表达式,因此需要将中缀表达式转换为后缀表达式。

算法引擎的解决方案如下:

A、将中缀表达式的数字和运算符分离

B、将中缀表达式转换为后缀表达式

C、使用后缀表达式计算运算表达式的结果

1、计算表达式的分离

中缀表达式通常包含如下元素:

A、数字与小数点[0-9 .]

B、符号位[+ -]

C、运算符[+ - * /]

D、括号[( )]

分离算法设计分析:

以符号作为标识对表达式中的字符进行逐个访问

A、定义累计变量

B、当前字符为数字或小数点

累计,num += exp[i]

C、当前字符exp[i]为为符号

num为运算数,分离并保存

如果exp[i]为正负号

累计符号位+、-:num += exp[i]

如果exp[i]为运算符

分离并保存

分离算法源码:

QQueue<QString> CalculatorDec::split(const QString& exp)
{
    QQueue<QString> ret;
    QString pre = "";
    QString num = "";
    for(int i=0; i<exp.length(); i++)
    {
        if(isDigitOrDot(exp[i]))
        {
            num += exp[i];
            pre = exp[i];
        }
        else if(isSymbol(exp[i]))
        {
            if(!num.isEmpty())
            {
                ret.enqueue(num);
                num.clear();
            }
            if(isSign(exp[i]) && ((pre == "") || (pre == "(") || isOperator(pre)))
            {
                num += exp[i];
            }
            else
            {
                ret.enqueue(exp[i]);
            }
            pre = exp[i];
        }
    }
 
    if(!num.isEmpty())
    {
        ret.enqueue(num);
    }
    return ret;
}

2、表达式的转换

中缀表达式转换为后缀表达式的注意事项:

A、运算表达式中的括号必须匹配

B、必须根据运算优先级进行转换

C、转换后的后缀表达式中不能有括号

D、转换后的后缀表达式可以计算出正确结果

在运算表达式中,括号匹配成对出现,左括号必须限于右括号出现。

匹配算法如下:

bool CalculatorDec::match(QQueue<QString>& exp)
{
    bool ret = true;
    int len = exp.length();
    QStack<QString> stack;
 
    for(int i=0; i<len; i++)
    {
        if( isLeft(exp[i]) )
        {
            stack.push(exp[i]);
        }
        else if( isRight(exp[i]) )
        {
            if( !stack.isEmpty() && isLeft(stack.top()) )
            {
                stack.pop();
            }
            else
            {
                ret = false;
                break;
            }
        }
    }
 
    return ret && stack.isEmpty();
}

转换过程:

A、当前元素e为数字,直接输出

B、当前元素e为运算符:

与栈顶运算符进行优先级比较,如果小于等于,将栈顶元素输出,转1;如果大于,将当前元素入栈。

C、当前元素e为左括号:入栈

D、当前元素e为右括号:

弹出栈顶元素并输出,直至栈顶元素为左括号,将栈顶的左括号从栈中弹出

转换算法如下:

bool CalculatorDec::transform(QQueue<QString>& exp, QQueue<QString>& output)
{
    bool ret = match(exp);
    QStack<QString> stack;
 
    output.clear();
    while( ret && !exp.isEmpty() )
    {
        QString e = exp.dequeue();
        if( isNumber(e) )
        {
            output.enqueue(e);
        }
        else if( isOperator(e) )
        {
            while( !stack.isEmpty() && (priority(e) <= priority(stack.top())) )
            {
                output.enqueue(stack.pop());
            }
 
            stack.push(e);
        }
        else if( isLeft(e) )
        {
            stack.push(e);
        }
        else if( isRight(e) )
        {
            while( !stack.isEmpty() && !isLeft(stack.top()) )
            {
                output.enqueue(stack.pop());
            }
            if( !stack.isEmpty() )
            {
                stack.pop();
            }
        }
        else
        {
            ret = false;
        }
    }
    while( !stack.isEmpty() )
    {
        output.enqueue(stack.pop());
    }
    if( !ret )
    {
        output.clear();
    }
 
    return ret;
}

3、结果计算

使用后缀表达式计算结果需要遍历后缀表达式中的数字和运算符。

A、当前元素为数字:进栈

B、当前元素为运算符:

从栈中弹出右操作数

从栈中弹出左操作数

根据符号进行运算

将运算结果压入栈中

遍历结束后,栈中的数字即为结果

QString CalculatorDec::calculate(QString l, QString op, QString r)
{
    QString ret = "Error";
 
    if( isNumber(l) && isNumber(r) )
    {
        double lp = l.toDouble();
        double rp = r.toDouble();
 
        if( op == "+" )
        {
            ret.sprintf("%f", lp + rp);
        }
        else if( op == "-" )
        {
            ret.sprintf("%f", lp - rp);
        }
        else if( op == "*" )
        {
            ret.sprintf("%f", lp * rp);
        }
        else if( op == "/" )
        {
            const double P = 0.000000000000001;
 
            if( (-P < rp) && (rp < P) )
            {
                ret = "Error";
            }
            else
            {
                ret.sprintf("%f", lp / rp);
            }
        }
        else
        {
            ret = "Error";
        }
    }
 
    return ret;
}
 
QString CalculatorDec::calculate(QQueue<QString>& exp)
{
    QString ret = "Error";
    QStack<QString> stack;
 
    while( !exp.isEmpty() )
    {
        QString e = exp.dequeue();
        if( isNumber(e) )
        {
            stack.push(e);
        }
        else if( isOperator(e) )
        {
            QString rp = !stack.isEmpty() ? stack.pop() : "";
            QString lp = !stack.isEmpty() ? stack.pop() : "";
            QString result = calculate(lp, e, rp);
 
            if( result != "Error" )
            {
                stack.push(result);
            }
            else
            {
                break;
            }
        }
        else
        {
            break;
        }
    }
    if( exp.isEmpty() && (stack.size() == 1) && isNumber(stack.top()) )
    {
        ret = stack.pop();
    }
 
    return ret;
}

四、用户界面与业务逻辑的分离

1、计算器架构设计

软件架构一般包括:

A、用户界面模块,接收用户输入,呈现数据

B、业务逻辑模块,根据用户需求处理数据

软件的各个模块之间必须遵循高内聚、低耦合的原则,每个模块应该只实现单一的功能,模块内部的子模块只作为整体的单一功能而存在,模块间通过约定好的接口进行交互。

模块间仅通过接口进行关联,因此必然存在使用接口的模块和实现接口的模块,模块间的关系必须是单向依赖的。

计算器应用程序的架构如下:

QCalculatorUI类实现了程序界面,QCalculatorDec类实现了计算器的核心算法。QCalculatorUI类通过引入ICalculator指针类型成员变量,增加了ICalculator属性;QCalculatorDec类继承自ICalculator,实现了ICalculator接口的计算功能。QCalculatorUI类与QCalculatorDec类间没有关系。

2、ICalculator类

#ifndef ICALCULATOR_H
#define ICALCULATOR_H
 
#include <QString>
 
class ICalculator
{
public:
    virtual bool expression(const QString & exp) = 0;
    virtual QString result() = 0;
};
 
#endif // ICALCULATOR_H

3、QCalculatorDec类

QCalculatorDec类继承自ICalculator,实现ICalculator类的计算功能

QCalculatorDec.h文件:

#ifndef CALCULATORDEC_H
#define CALCULATORDEC_H
 
#include <QString>
#include <QQueue>
#include <QStack>
#include "ICalculator.h"
 
class CalculatorDec : public ICalculator
{
public:
    CalculatorDec();
    ~CalculatorDec();
    bool expression(const QString & exp);
    QString expression();
    QString result();
private:
    bool isDigitOrDot(QChar c);
    bool isSymbol(QChar c);
    bool isSign(QChar c);
    bool isNumber(QString s);
    bool isOperator(QString s);
    bool isLeft(QString s);
    bool isRight(QString s);
    int priority(QString s);
    QQueue<QString> split(const QString& exp);
    bool match(QQueue<QString>& exp);
    bool transform(QQueue<QString>& exp, QQueue<QString>& output);
    QString calculate(QQueue<QString>& exp);
    QString calculate(QString l, QString op, QString r);
private:
    QString m_exp;
    QString m_result;
};
 
#endif // CALCULATORDEC_H

QCalculatorDec.cpp文件:

#include "CalculatorDec.h"
 
CalculatorDec::CalculatorDec()
{
    m_exp = "";
    m_result = "";
}
 
CalculatorDec::~CalculatorDec()
{
}
 
bool CalculatorDec::expression(const QString & exp)
{
    bool ret = false;
    QQueue<QString> spExp = split(exp);
    QQueue<QString> postExp;
 
    m_exp = exp;
    if( transform(spExp, postExp) )
    {
        m_result = calculate(postExp);
        ret = (m_result != "Error");
    }
    else
    {
        m_result = "Error";
    }
    return ret;
}
 
QString CalculatorDec::expression()
{
    QString ret;
 
    return ret;
}
 
QString CalculatorDec::result()
{
    return m_result;
}
 
bool CalculatorDec::isDigitOrDot(QChar c)
{
    return (‘0‘ <= c) && (c <= ‘9‘) || (c == ‘.‘);
}
 
bool CalculatorDec::isSymbol(QChar c)
{
    return isOperator(c) || (‘(‘ == c) || (‘)‘ == c);
}
 
bool CalculatorDec::isSign(QChar c)
{
    return (‘+‘ == c) || (‘-‘ == c);
}
 
bool CalculatorDec::isNumber(QString s)
{
    bool ret = false;
    s.toDouble(&ret);
    return ret;
}
 
bool CalculatorDec::isOperator(QString s)
{
    return ("+" == s) || ("-" == s) || ("*" == s) || ("/" == s);
}
 
bool CalculatorDec::isLeft(QString s)
{
    return ("(" == s);
}
 
bool CalculatorDec::isRight(QString s)
{
    return (")" == s);
}
 
int CalculatorDec::priority(QString s)
{
    int ret = -1;
    if(s == "+" || s == "-")
    {
        ret = 1;
    }
    if(s == "*" || s == "/")
    {
        ret = 2;
    }
    return ret;
}
 
QQueue<QString> CalculatorDec::split(const QString& exp)
{
    QQueue<QString> ret;
    QString pre = "";
    QString num = "";
    for(int i=0; i<exp.length(); i++)
    {
        if(isDigitOrDot(exp[i]))
        {
            num += exp[i];
            pre = exp[i];
        }
        else if(isSymbol(exp[i]))
        {
            if(!num.isEmpty())
            {
                ret.enqueue(num);
 
                num.clear();
            }
            if(isSign(exp[i]) && ((pre == "") || (pre == "(") || isOperator(pre)))
            {
                num += exp[i];
            }
            else
            {
                ret.enqueue(exp[i]);
            }
            pre = exp[i];
        }
    }
 
    if(!num.isEmpty())
    {
        ret.enqueue(num);
    }
    return ret;
}
 
bool CalculatorDec::match(QQueue<QString>& exp)
{
    bool ret = true;
    int len = exp.length();
    QStack<QString> stack;
 
    for(int i=0; i<len; i++)
    {
        if( isLeft(exp[i]) )
        {
            stack.push(exp[i]);
        }
        else if( isRight(exp[i]) )
        {
            if( !stack.isEmpty() && isLeft(stack.top()) )
            {
                stack.pop();
            }
            else
            {
                ret = false;
                break;
            }
        }
    }
 
    return ret && stack.isEmpty();
}
 
bool CalculatorDec::transform(QQueue<QString>& exp, QQueue<QString>& output)
{
    bool ret = match(exp);
    QStack<QString> stack;
 
    output.clear();
    while( ret && !exp.isEmpty() )
    {
        QString e = exp.dequeue();
        if( isNumber(e) )
        {
            output.enqueue(e);
        }
        else if( isOperator(e) )
        {
            while( !stack.isEmpty() && (priority(e) <= priority(stack.top())) )
            {
                output.enqueue(stack.pop());
            }
 
            stack.push(e);
        }
        else if( isLeft(e) )
        {
            stack.push(e);
        }
        else if( isRight(e) )
        {
            while( !stack.isEmpty() && !isLeft(stack.top()) )
            {
                output.enqueue(stack.pop());
            }
            if( !stack.isEmpty() )
            {
                stack.pop();
            }
        }
        else
        {
            ret = false;
        }
    }
    while( !stack.isEmpty() )
    {
        output.enqueue(stack.pop());
    }
    if( !ret )
    {
        output.clear();
    }
 
    return ret;
}
 
 
QString CalculatorDec::calculate(QString l, QString op, QString r)
{
    QString ret = "Error";
 
    if( isNumber(l) && isNumber(r) )
    {
        double lp = l.toDouble();
        double rp = r.toDouble();
 
        if( op == "+" )
        {
            ret.sprintf("%f", lp + rp);
        }
        else if( op == "-" )
        {
            ret.sprintf("%f", lp - rp);
        }
        else if( op == "*" )
        {
            ret.sprintf("%f", lp * rp);
        }
        else if( op == "/" )
        {
            const double P = 0.000000000000001;
 
            if( (-P < rp) && (rp < P) )
            {
                ret = "Error";
            }
            else
            {
                ret.sprintf("%f", lp / rp);
            }
        }
        else
        {
            ret = "Error";
        }
    }
 
    return ret;
}
 
QString CalculatorDec::calculate(QQueue<QString>& exp)
{
    QString ret = "Error";
    QStack<QString> stack;
 
    while( !exp.isEmpty() )
    {
        QString e = exp.dequeue();
        if( isNumber(e) )
        {
            stack.push(e);
        }
        else if( isOperator(e) )
        {
            QString rp = !stack.isEmpty() ? stack.pop() : "";
            QString lp = !stack.isEmpty() ? stack.pop() : "";
            QString result = calculate(lp, e, rp);
 
            if( result != "Error" )
            {
                stack.push(result);
            }
            else
            {
                break;
            }
        }
        else
        {
            break;
        }
    }
    if( exp.isEmpty() && (stack.size() == 1) && isNumber(stack.top()) )
    {
        ret = stack.pop();
    }
 
    return ret;
}

4、CalculatorUI类

CalculatorUI类使用ICalculator接口的计算功能。通过在CalculatorUI类中引入ICalculator指针成员变量,增加了ICalculator属性。将所有按钮发送的信号与onCalculate()槽函数连接。

CalculatorUI.h文件:

#ifndef CALCULATORUI_H
#define CALCULATORUI_H
 
#include <QWidget>
#include <QLineEdit>
#include <QPushButton>
#include "icalculator.h"
 
class CalculatorUI : public QWidget
{
    Q_OBJECT
public:
    static CalculatorUI* newInstance();
    ~CalculatorUI();
    void show();
    void setCalculator(ICalculator* cal);
    ICalculator* getCalculator();
private:
    CalculatorUI();
    bool Construct();
 
private slots:
    void onCalculate();
private:
    QLineEdit *m_edit;
    QPushButton *m_buttons[20];
    ICalculator *m_cal;
};
 
#endif // CALCULATORUI_H

CalculatorUI.cpp文件:

#include "CalculatorUI.h"
 
CalculatorUI::CalculatorUI() : QWidget(NULL, Qt::WindowCloseButtonHint)
{
    m_cal = NULL;
}
 
CalculatorUI* CalculatorUI::newInstance()
{
    CalculatorUI *ret = new CalculatorUI();
    if(NULL == ret || !ret->Construct())
    {
        delete ret;
        ret = NULL;
    }
    return ret;
}
 
CalculatorUI::~CalculatorUI()
{
}
 
void CalculatorUI::show()
{
    QWidget::show();
    setFixedSize(width(), height());
}
 
bool CalculatorUI::Construct()
{
    bool ret = true;
    m_edit = new QLineEdit(this);
    if(m_edit != NULL)
    {
        m_edit->move(10, 10);
        m_edit->resize(240, 30);
        m_edit->setReadOnly(true);
    }
    else
    {
        ret = false;
    }
 
    const char *buttontext[20] =
    {
        "7", "8", "9", "+", "(",
        "4", "5", "6", "-", ")",
        "1", "2", "3", "*", "<-",
        "0", ".", "=", "/", "C"
    };
    for(int i = 0; (i < 4) && ret; i++)
    {
        for(int j = 0; (j < 5) && ret; j++)
        {
            m_buttons[5*i + j] = new QPushButton(this);
            if(m_buttons[5*i + j] != NULL)
            {
                m_buttons[5*i + j]->move(10 + (10 + 40)*j, 50 + (10 + 40)*i);
                m_buttons[5*i + j]->resize(40, 40);
                m_buttons[5*i + j]->setText(buttontext[5*i + j]);
                connect(m_buttons[5*i + j], SIGNAL(clicked()), this, SLOT(onCalculate()));
            }
            else
            {
                ret = false;
            }
        }
    }
    return ret;
}
 
void CalculatorUI::setCalculator(ICalculator* cal)
{
    m_cal = cal;
}
 
ICalculator* CalculatorUI::getCalculator()
{
    return m_cal;
}
 
void CalculatorUI::onCalculate()
{
    QPushButton* button = dynamic_cast<QPushButton*>(sender());
 
    if( button != NULL )
    {
        QString buttontext = button->text();
 
        if( buttontext == "<-" )
        {
            QString text = m_edit->text();
 
            if( text.length() > 0 )
            {
                text.remove(text.length()-1, 1);
                m_edit->setText(text);
            }
        }
        else if( buttontext == "C" )
        {
            m_edit->setText("");
        }
        else if( buttontext == "=" )
        {
            if( m_cal != NULL )
            {
                m_cal->expression(m_edit->text());
                m_edit->setText(m_cal->result());
            }
        }
        else
        {
            m_edit->setText(m_edit->text() + buttontext);
        }
    }
}

5、Calculator

计算器Calculator类包含计算器程序界面CalculatorUI和核心算法CalculatorDec两部分。在Calculator构造过程中需要指定CalculatorUI对应的核心算法CalculatorDec。

Calculator.h文件:

#ifndef CALCULATOR_H
#define CALCULATOR_H
 
#include "CalculatorUI.h"
#include "CalculatorDec.h"
 
class Calculator
{
private:
    CalculatorUI *m_ui;
    CalculatorDec m_cal;
 
    Calculator();
    bool construct();
public:
    static Calculator* newInstance();
    void show();
    ~Calculator();
 
};
 
#endif // CALCULATOR_H

Calculator.cpp文件:

#include "Calculator.h"
 
Calculator::Calculator()
{
}
 
bool Calculator::construct()
{
    m_ui = CalculatorUI::newInstance();
    if(m_ui != NULL)
    {
        m_ui->setCalculator(&m_cal);
    }
    return (m_ui != NULL);
}
 
Calculator* Calculator::newInstance()
{
    Calculator *ret = new Calculator();
    if((ret == NULL) || (!ret->construct()))
    {
        delete ret;
        ret = NULL;
    }
    return ret;
}
 
void Calculator::show()
{
    m_ui->show();
}
 
Calculator::~Calculator()
{
    delete m_ui;
}

6、使用实例

Main.cpp文件:

#include <QApplication>
#include "Calculator.h"
 
int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    Calculator *cal = Calculator::newInstance();
    int ret = 0;
    if(cal != NULL)
    {
        cal->show();
        ret = a.exec();
        delete cal;
    }
    return ret;
}
时间: 2024-10-12 23:56:18

QT开发(三十)——计算器实例开发的相关文章

Android Studio第三十三期 - 蓝牙开发初识~

遇见到坑爹的队友只有接受现实并且解决问题~ 首先介绍一下网上几乎所有的能搜到的方法: 1.首先,要操作蓝牙,先要在AndroidManifest.xml里加入权限 <uses-permissionandroid:name="android.permission.BLUETOOTH_ADMIN" /> <uses-permissionandroid:name="android.permission.BLUETOOTH" /> 2.在androi

python运维开发(二十五)---cmdb开发

内容目录: 浅谈ITIL CMDB介绍 Django自定义用户认证 Restful 规范 资产管理功能开发 浅谈ITIL TIL即IT基础架构库(Information Technology Infrastructure Library, ITIL,信息技术基础架构库)由英国政府部门CCTA(Central Computing and Telecommunications Agency)在20世纪80年代末制订,现由英国商务部OGC(Office of Government Commerce)负

Android 开发第一项目——计算器的开发记录

2017.4.1 今天布局界面基本完成,现在写了一点事件绑定.计划是多用动态绑定,随时用随时改.关于布局方面,昨天弄到很晚,原因是Layout使用错误,用的自带的,没仔细看,预览的时候没有问题但是真机调试就出错.是兼容性问题,很多的安卓系统不支持constrainlayout(6.0不支持不要问我为什么知道).改用LinearLayout,简单嵌套了一下,主体还是用了几个LinearLayout分别写了每一行,其中用了几个LinearLayout用来调整按钮间隙.然后稍微调了字号和对齐方式.

三得公益系统开发详解

三得公益系统开发(苏念.188.1414.7927)三得公益模式开发 三得公益app开发 三得公益平台开发 三得公益公益模式开发 三得公益商业模式开发,亲,当你打开手机浏览微信朋友圈或微博的时候,是否会时不时地看到一两条公益信息?救助早产儿.帮助失独家庭.关注孤残群体--当下,互联网+公益的热潮正兴,做公益无需再东奔西走,只需动动手指就能搞定. 好事还能这么做! 我国现在约有1731万盲人,而且这一数字每分钟都在递增.一边是对盲人的看护和帮助需求快速攀升,另一边又是看护成本高所带来的看护人员的减

QT开发(十六)——QT绘图实例-钟表

QT开发(十六)--QT绘图实例-钟表 一.钟表实现原理 钟表的实现需要设置定时器,定时器每隔一秒发送timeout()信号到QWidget::update()槽函数,update()槽函数将会重绘一次窗口,重写重绘事件函数paintEvent(QPaintEvent *event),根据获取的当前系统时间的时钟.分钟.秒钟重绘钟表的时针.分针.秒针. QTimer *timer = new QTimer(this); timer->start(1000);//一秒钟 connect(timer

QT开发(三十八)——Model/View框架编程

QT开发(三十八)--Model/View框架编程 一.自定义模型 1.自定义只读模型 QAbstractItemModel为自定义模型提供了一个足够灵活的接口,能够支持数据源的层次结构,能够对数据进行增删改操作,还能够支持拖放.QT提供了 QAbstarctListModel和QAbstractTableModel两个类来简化非层次数据模型的开发,适合于结合列表和表格使用. 自定义模型需要考虑模型管理的的数据结构适合的视图的显示方式.如果模型的数据仅仅用于列表或表格的显示,那么可以使用QAbs

QT开发(三十四)——QT多线程编程

QT开发(三十四)--QT多线程编程 一.QT多线程简介 QT通过三种形式提供了对线程的支持,分别是平台无关的线程类.线程安全的事件投递.跨线程的信号-槽连接. QT中线程类包含如下: QThread 提供了开始一个新线程的方法    QThreadStorage 提供逐线程数据存储    QMutex 提供相互排斥的锁,或互斥量    QMutexLocker 是一个辅助类,自动对 QMutex 加锁与解锁    QReadWriterLock 提供了一个可以同时读操作的锁    QReadL

QT开发(三十五)——QT进程间通信

QT开发(三十五)--QT进程间通信 Qt 是一个跨平台的应用框架,其进程间通信机制当然可以使用所在平台的进程间通信机制,如在Windows平台上的Message机制.共享内存.文件映射.管道.Socket等.其中,Qt对一些许多平台共有的IPC机制进行了封装. 一.TCP/IP 其实就是通过网络模块实现的IPC.不过Qt对其进行了封装,并提供了两个层次的API,包括应用程序级的QNetworkAccessManager, QFtp等和底层的QTcpSocket, QTcpServer, QSs

QT开发(三十六)——Model/View框架

QT开发(三十六)--Model/View框架 一.Model/View框架简介 1.Model/View框架核心思想 Model/View框架的核心思想是模型(数据)与视图(显示)相分离,模型对外提供标准接口存取数据,不关心数据如何显示,视图自定义数据的显示方式,不关心数据如何组织存储. Model/View框架中数据与显示的分离,可以允许使用不同界面显示同一数据,也能够在不改变数据的情况下添加新的显示界面.为了处理用户输入,引入了委托(delegate).引入委托的好处是可以自定义数据项的渲