编写一个验证密码强度的程序,能够根据预先定义的规则,按各类排列组合来设定密码的强度。
最小要求,每一个密码都必须知足最小长度要求。
额外要求,能够添加其余规则限制,例如:c++
熟悉设计模式的读者确定知道,这里描述的问题是一个典型的装饰器(Decorator)模式问题。
装饰器模式在不改变现有对象结构的状况下,动态地给该对象增长一些行为,而不会影响相同类型的其余对象。git
采用装饰器模式扩展对象的功能比采用继承方式更加灵活。
能够设计出多个不一样的具体装饰类,创造出多个不一样行为的组合。设计模式
装饰器模式增长了许多子类,过分使用会使程序变得很复杂。ide
多个装饰器能够彼此叠加,每次添加新功能。 在这个案例中,验证给定密码是否符合特定组合的要求。spa
先看看整个validating passwords
模型设计
struct Validator { virtual ~Validator() {} virtual bool validate(std::string_view) = 0; };
基类Validator
定义了契约方法validate
,带有表示密码的字符串参数。c++11
struct LengthValidator final : Validator { LengthValidator(unsigned int length) : length_{length} {} bool validate(std::string_view password) override { return password.length() >= length_; } private: unsigned int length_; };
子类LengthValidator
实现了密码最小长度的强制性要求。code
struct ValidatorDecorator : Validator { explicit ValidatorDecorator(std::unique_ptr<Validator> validator) : impl_(std::move(validator)) {} virtual bool validate(std::string_view password) override { return impl_->validate(password); } private: std::unique_ptr<Validator> impl_; };
数字密码校验对象
struct DigitalPasswordValidator final : ValidatorDecorator { explicit DigitalPasswordValidator(std::unique_ptr<Validator> validator) : ValidatorDecorator(std::move(validator)) {} bool validate(std::string_view password) override { if (!ValidatorDecorator::validate(password)) return false; return password.find_first_of("0123456789") != std::string::npos; } };
大小写字母校验blog
struct CasePasswordValidator final : ValidatorDecorator { explicit CasePasswordValidator(std::unique_ptr<Validator> validator) : ValidatorDecorator(std::move(validator)) {} bool validate(std::string_view password) override { if (!ValidatorDecorator::validate(password)) return false; bool has_lower = false; bool has_upper = false; std::for_each(begin(password), end(password), [&](const auto ch) { if (islower(ch)) has_lower = true; else if (isupper(ch)) has_upper = true; }); return has_lower && has_upper; } };
特殊符号校验
struct SymbolPasswordValidator final : ValidatorDecorator { explicit SymbolPasswordValidator(std::unique_ptr<Validator> validator) : ValidatorDecorator(std::move(validator)) {} bool validate(std::string_view password) override { if (!ValidatorDecorator::validate(password)) return false; return password.find_first_of("`~!@#$%^&*-_=+(){}[]?<>,./") != std::string::npos; } };
最后举个简单的例子
int main() { auto validator = std::make_unique<LengthValidator>(8); auto num_validator = std::make_unique<DigitalPasswordValidator>(std::move(validator)); assert(num_validator->validate("abcd123,./")); assert(!num_validator->validate("abcdvdfs,./")); auto num_symbol_case_validator = std::make_unique<DigitalPasswordValidator>( std::make_unique<SymbolPasswordValidator>( std::make_unique<CasePasswordValidator>( std::make_unique<LengthValidator>(8)))); assert(num_symbol_case_validator->validate("Abc123!@#")); assert(!num_symbol_case_validator->validate("Abc12")); return 0; }