AngularJS之代码风格36条建议【一】(九)
前言
其实在新学一门知识时,我们应该注意下怎么书写代码更加规范,从开始就注意养成一个良好的习惯无论是对于bug的查找还是走人后别人熟悉代码都是非常好的,利人利己的事情何乐而不为呢,关于AngularJS中的代码风格分为几节来阐述。希望对打算学习AngularJS的新手或者已经在路上的老手有那么一丢丢的帮助也是可以的。
普遍规则
tips 01(定义一个组件脚本文件时,建议此文件的代码少于400行)
(1)有利于单元测试和模拟测试。
(2)增加可读性、可维护性、避免和团队在源代码控制上的冲突。
(3)当在文件中组合组件时,可能会共享变量、依赖不需要的耦合从而避免潜在的bugs。
避免如下这样做:
angular
.module('app', ['ngRoute'])
.controller('SomeController', SomeController)
.factory('someFactory', someFactory); function SomeController() { } function someFactory() { }
相同的组件应该分为各自的文件(推荐如下做):
// app.module.js
angular
.module('app', ['ngRoute']);
angular
.module('app')
.controller('SomeController', SomeController); function SomeController() { }
angular
.module('app')
.factory('someFactory', someFactory); function someFactory() { }
JavaScript作用域
tips 02 (包含Angular的组件应该作为匿名函数立即被调用)
(1)匿名函数移除了全局作用域中的变量,能够避免变量冲突以及变量长期存在于内存中。
(2)当代码经过捆绑和压缩到单个文件中,并将其文件部署到生产服务器中时会产生全局变量的冲突。
避免如下这样做:
angular
.module('app')
.factory('logger', logger); function logger() { } angular
.module('app')
.factory('storage', storage); function storage() { }
推荐如下做:
(function() {
'use strict'; angular
.module('app')
.factory('logger', logger); function logger() { }
})();
(function() {
'use strict'; angular
.module('app')
.factory('storage', storage); function storage() { }
})();
定义模块
tips 03 (声明模块时不要用变量来返回)
一个文件中的组件,很少使用需要引入一个变量的模块。
避免如下这样做:
var app = angular.module('app', [
'ngAnimate',
'ngRoute',
'app.shared',
'app.dashboard'
]);
推荐如下做:
angular
.module('app', [
'ngAnimate',
'ngRoute',
'app.shared',
'app.dashboard'
]);
使用模块
tips 04(使用模块时避免使用变量代替的应该是链式语法)
将使代码更加可读,避免变量的冲突和泄漏。
避免如下这样做:
var app = angular.module('app');
app.controller('SomeController', SomeController); function SomeController() { }
推荐如下这样做:
angular
.module('app')
.controller('SomeController', SomeController); function SomeController() { }
命名函数vs匿名函数
tips 05 (使用命名函数来作为函数的回调而非匿名函数)
使代码易读,易于调试且降低嵌套代码的回调量。
避免如下这样做:
angular
.module('app')
.controller('DashboardController', function() { })
.factory('logger', function() { });
推荐如下这样做:
angular
.module('app')
.controller('DashboardController', DashboardController); function DashboardController() { }
angular
.module('app')
.factory('logger', logger); function logger() { }
控制器
tips 06(使用controllerAs语法代替$scope语法)
避免如下:
<div ng-controller="CustomerController">
{{ name }}
</div>
推荐如下:
<div ng-controller="CustomerController as customer">
{{ customer.name }}
</div>
tips 07(使用控制器内部使用controllerAs语法代替$scope语法即再内部用this代替$scope)
避免如下:
function CustomerController($scope) {
$scope.name = {};
$scope.sendMessage = function() { };
}
推荐如下:
function CustomerController() {
this.name = {};
this.sendMessage = function() { };
}
tips 08(使用VM代替controllerAs语法即使用一个变量来捕获this,如VM,它代表ViewModel。)
this关键字代表上下文,在控制器内部使用函数时可能会改变它的上下文,用一个变量来捕获this能够避免面临这样的问题。
避免如下:
function CustomerController() {
this.name = {};
this.sendMessage = function() { };
}
推荐如下:
function CustomerController() {
var vm = this;
vm.name = {};
vm.sendMessage = function() { };
}
tips 09(在控制器的最顶部按照字母大小来排序,而非通过控制器代码来进行扩展)
(1)在顶部绑定成员易于阅读同时帮助我们识别可以在控制器中绑定的成员并在视图中使用。
(2)使用匿名函数虽然可能,但是一旦代码量超过一定数量则降低了代码的可阅读性。
避免如下:
function SessionsController() {
var vm = this; vm.gotoSession = function() {
/* ... */
};
vm.refresh = function() {
/* ... */
};
vm.search = function() {
/* ... */
};
vm.sessions = [];
vm.title = 'Sessions';
}
推荐如下:
function SessionsController() {
var vm = this; vm.gotoSession = gotoSession;
vm.refresh = refresh;
vm.search = search;
vm.sessions = [];
vm.title = 'Sessions'; //////////// function gotoSession() {
/* */
} function refresh() {
/* */
} function search() {
/* */
}
}
tips 10 (使用声明式函数来隐藏实现细节)
使用声明式函数来隐藏实现细节,并保持绑定的成员在顶部。当在控制器中需要绑定一个函数时,指向它到一个函数式声明紧接着在下面。即将成员绑定在顶部且使用声明式函数。
避免如下:
function AvengersController(avengersService, logger) {
var vm = this;
vm.avengers = [];
vm.title = 'Avengers'; var activate = function() {
return getAvengers().then(function() {
logger.info('Activated Avengers View');
});
} var getAvengers = function() {
return avengersService.getAvengers().then(function(data) {
vm.avengers = data;
return vm.avengers;
});
} vm.getAvengers = getAvengers; activate();
}
推荐如下:
function AvengersController(avengersService, logger) {
var vm = this;
vm.avengers = [];
vm.getAvengers = getAvengers;
vm.title = 'Avengers'; activate(); function activate() {
return getAvengers().then(function() {
logger.info('Activated Avengers View');
});
} function getAvengers() {
return avengersService.getAvengers().then(function(data) {
vm.avengers = data;
return vm.avengers;
});
}
}
tips 11(在控制器中通过服务和工厂将业务逻辑导入其中)
(1)业务逻辑可能在多个控制器中被重用,将服务通过函数进行暴露。
(2)在单元测试中,业务逻辑更容易被隔离,在控制器中进行调用时更容易被模拟。
(3)消除了依赖且在控制器中隐藏了实现的细节。
避免如下:
function OrderController($http, $q, config, userInfo) {
var vm = this;
vm.checkCredit = checkCredit;
vm.isCreditOk;
vm.total = 0; function checkCredit() {
var settings = {};
return $http.get(settings)
.then(function(data) {
vm.isCreditOk = vm.total <= maxRemainingAmount
})
.catch(function(error) {
});
};
}
推荐如下:
function OrderController(creditService) {
var vm = this;
vm.checkCredit = checkCredit;
vm.isCreditOk;
vm.total = 0; function checkCredit() {
return creditService.isOrderTotalOk(vm.total)
.then(function(isOk) { vm.isCreditOk = isOk; })
.catch(showError);
};
}
tips 12 (保持控制器关注)
对一个视图定义一个控制器,对于其他控制器不要重用控制器,代替的是将重用逻辑移到工厂以此来保持控制器简单,更多的是关注视图。
tips 13(分配控制器)
当控制器必须和一个视图配对并且组件会被其他控制器和视图重用时,通过路由来定义控制器。
避免如下:
angular
.module('app')
.config(config); function config($routeProvider) {
$routeProvider
.when('/avengers', {
templateUrl: 'avengers.html'
});
} <div ng-controller="AvengersController as vm">
</div>
推荐如下:
angular
.module('app')
.config(config); function config($routeProvider) {
$routeProvider
.when('/avengers', {
templateUrl: 'avengers.html',
controller: 'Avengers',
controllerAs: 'vm'
});
} <div>
</div>
服务
tips 14(单例)
服务被初始化通过new关键字,使用this关键字来修饰方法和变量,因为所有的服务是单例对象,所以对于每个injector的服务只有唯一的实例。
推荐如下:
// service
angular
.module('app')
.service('logger', logger); function logger() {
this.logError = function(msg) {
/* */
};
}
// factory
angular
.module('app')
.factory('logger', logger); function logger() {
return {
logError: function(msg) {
/* */
}
};
}
工厂
tips 15(将访问成员置顶)
(1)在顶部暴露要调用的服务的成员,加强可读性以及单元测试。
(2)当文件足够大时,可能需要滚动才能看到其暴露的函数。
(3)通过服务定义的接口在代码量超过100行时避免降低代码的可阅读性和造成更多的滚动。
避免如下:
function dataService() {
var someValue = '';
function save() {
/* */
};
function validate() {
/* */
}; return {
save: save,
someValue: someValue,
validate: validate
};
}
推荐如下:
function dataService() {
var someValue = '';
var service = {
save: save,
someValue: someValue,
validate: validate
};
return service; //////////// function save() {
/* */
}; function validate() {
/* */
};
}
服务
tips 16 (重构服务)
对于数据操作和将数据与工厂进行交互时重构逻辑,使数据服务负责ajax等或其他操作。
推荐如下:
angular
.module('app.core')
.factory('dataservice', dataservice); dataservice.$inject = ['$http', 'logger']; function dataservice($http, logger) {
return {
getAvengers: getAvengers
}; function getAvengers() {
return $http.get('/api/maa')
.then(getAvengersComplete)
.catch(getAvengersFailed); function getAvengersComplete(response) {
return response.data.results;
} function getAvengersFailed(error) {
logger.error('XHR Failed for getAvengers.' + error.data);
}
}
}
指令
tips 17(为每个指令定义一个文件,并以此指令命名)
(1)很容易将所有指令混合在一个文件中,但是很难对于共享跨应用程序或者共享模块等。
(2)每一个文件一个指令利于代码的可维护性。
避免如下:
angular
.module('app.widgets') .directive('orderCalendarRange', orderCalendarRange) .directive('salesCustomerInfo', salesCustomerInfo) function orderCalendarRange() { } function salesCustomerInfo() {
}
推荐如下:
angular
.module('sales.order')
.directive('acmeOrderCalendarRange', orderCalendarRange); function orderCalendarRange() {
}
angular
.module('sales.widgets')
.directive('acmeSalesCustomerInfo', salesCustomerInfo); function salesCustomerInfo() {
}
tips 18 (提供唯一的指令前缀)
提供一个短的、唯一的、描述性的指令前缀。例如cnblogsIngUserInfo,则在html中被声明为cnblogs-ing-user-info。
可以用唯一的指令前缀来标识指令的背景和来源,例如上述的cnblogsIngUserInfo,cnblogs代表博客园,而Ing代表闪存,User代表用户,info代表信息。
tips 19(对元素和特性进行约束)
在AngularJS 1.3+默认的是EA,在此之下需要用Restrict来进行限制。
避免如下:
<div class="my-calendar-range"></div>
推荐如下:
<my-calendar-range></my-calendar-range>
<div my-calendar-range></div>
tips 20 (在指令中使用controllerAs语法与控制器和视图中使用该语法要一致)
推荐如下:
<div my-example max="77"></div>
angular
.module('app')
.directive('myExample', myExample); function myExample() {
var directive = {
restrict: 'EA',
templateUrl: 'app/feature/example.directive.html',
scope: {
max: '='
},
controller: ExampleController,
controllerAs: 'vm'
}; return directive;
} function ExampleController() {
var vm = this;
vm.min = 3;
console.log('CTRL: vm.min = %s', vm.min);
console.log('CTRL: vm.max = %s', vm.max);
}
<!-- example.directive.html -->
<div>hello world</div>
<div>max={{vm.max}}<input ng-model="vm.max"/></div>
<div>min={{vm.min}}<input ng-model="vm.min"/></div>
tips 21(在指令添加属性bindToController = true)
当在指令中使用controllerAs语法时,若我们想绑定外部作用域到指令的控制器的作用域令bindToController = true。
如上述tips 20初始化文本值为vm.max为undifined,若设置bindToController = true,则vm.max = 77;
解析promise
tips 22(控制器激活promise)
在一个activate函数中来启动控制器的逻辑。
(1)在一致的地方放置启动逻辑有利于问题的定位以及测试,同时避免在跨控制器中传播激活逻辑。
(2)控制器激活可以更方便地重用刷新的控制器或者视图逻辑,保持逻辑在一起,使得更快加载视图。
避免如下:
function AvengersController(dataservice) {
var vm = this;
vm.avengers = [];
vm.title = 'Avengers'; dataservice.getAvengers().then(function(data) {
vm.avengers = data;
return vm.avengers;
});
}
推荐如下:
function AvengersController(dataservice) {
var vm = this;
vm.avengers = [];
vm.title = 'Avengers'; activate(); //////////// function activate() {
return dataservice.getAvengers().then(function(data) {
vm.avengers = data;
return vm.avengers;
})
}
}
tips 23(路由解析promise)
在控制器被激活之前,若控制器依赖于promise需要被解析时,在控制器逻辑执行之前通过$routerProvider来解析这些依赖。在控制器激活之前,如果需要依据条件来取消路由,通过路由解析来进行。
(1)在控制器加载之前之前它可能需要获取数据,数据可能来源于自定义的工厂或$http的promise。使用路由解析使得promise在控制器逻辑执行之前被解析,因此它可能根据在promise的数据来采取不同的动作。
(2)在路由和控制器的激活函数中的代码执行之后,视图开始被正确加载,当激活promise解析时,数据绑定开始进行。
避免如下:
angular
.module('app')
.controller('AvengersController', AvengersController); function AvengersController(movieService) {
var vm = this;
// unresolved
vm.movies;
// resolved asynchronously
movieService.getMovies().then(function(response) {
vm.movies = response.movies;
});
}
推荐如下:
// route-config.js
angular
.module('app')
.config(config); function config($routeProvider) {
$routeProvider
.when('/avengers', {
templateUrl: 'avengers.html',
controller: 'AvengersController',
controllerAs: 'vm',
resolve: {
moviesPrepService: function(movieService) {
return movieService.getMovies();
}
}
});
} // avengers.js
angular
.module('app')
.controller('AvengersController', AvengersController); AvengersController.$inject = ['moviesPrepService'];
function AvengersController(moviesPrepService) {
var vm = this;
vm.movies = moviesPrepService.movies;
}
或者推荐如下操作(更易于调试和处理依赖注入):
// route-config.js
angular
.module('app')
.config(config); function config($routeProvider) {
$routeProvider
.when('/avengers', {
templateUrl: 'avengers.html',
controller: 'AvengersController',
controllerAs: 'vm',
resolve: {
moviesPrepService: moviesPrepService
}
});
} function moviesPrepService(movieService) {
return movieService.getMovies();
} // avengers.js
angular
.module('app')
.controller('AvengersController', AvengersController); AvengersController.$inject = ['moviesPrepService'];
function AvengersController(moviesPrepService) {
var vm = this;
vm.movies = moviesPrepService.movies;
}
tips 24(用promise来处理异常)
一个promise的catch模块必须要返回一个reject的promise来在promise链中维护异常。
在服务或者工厂中一定要处理异常。
(1)如果一个catch模块没有返回一个reject的promise,那么此时这个promise的调用者不知道异常的出现,接着调用者的then然后被执行,用户根本不知道发生了什么。
(2)避免隐藏的错误以及误导用户。
避免如下:
function getCustomer(id) {
return $http.get('/api/customer/' + id)
.then(getCustomerComplete)
.catch(getCustomerFailed); function getCustomerComplete(data, status, headers, config) {
return data.data;
} function getCustomerFailed(e) {
var newMessage = 'XHR Failed for getCustomer'
if (e.data && e.data.description) {
newMessage = newMessage + '\n' + e.data.description;
}
e.data.description = newMessage;
logger.error(newMessage);
// ***
// Notice there is no return of the rejected promise
// ***
}
}
推荐如下:
function getCustomer(id) {
return $http.get('/api/customer/' + id)
.then(getCustomerComplete)
.catch(getCustomerFailed); function getCustomerComplete(data, status, headers, config) {
return data.data;
} function getCustomerFailed(e) {
var newMessage = 'XHR Failed for getCustomer'
if (e.data && e.data.description) {
newMessage = newMessage + '\n' + e.data.description;
}
e.data.description = newMessage;
logger.error(newMessage);
return $q.reject(e);
}
}
手动标注依赖注入
tips 25(手动识别依赖)
使用$inject来识别AngularJS组件中的依赖。
避免如下:
angular
.module('app')
.controller('DashboardController',
['$location', '$routeParams', 'common', 'dataservice',
function Dashboard($location, $routeParams, common, dataservice) {}
]);
避免如下:
angular
.module('app')
.controller('DashboardController',
['$location', '$routeParams', 'common', 'dataservice', Dashboard]); function Dashboard($location, $routeParams, common, dataservice) {
}
推荐如下:
angular
.module('app')
.controller('DashboardController', DashboardController); DashboardController.$inject = ['$location', '$routeParams', 'common', 'dataservice']; function DashboardController($location, $routeParams, common, dataservice) {
}
注意:当函数是如下一个返回语句,此时$inject可能无法访问(例如在指令中),此时解决这个问题的办法是将$inject移动到控制器的外面。
tips 26($inject无效的情况)
避免如下:
function outer() {
var ddo = {
controller: DashboardPanelController,
controllerAs: 'vm'
};
return ddo; DashboardPanelController.$inject = ['logger']; // Unreachable
function DashboardPanelController(logger) {
}
}
推荐如下:
function outer() {
var ddo = {
controller: DashboardPanelController,
controllerAs: 'vm'
};
return ddo;
} DashboardPanelController.$inject = ['logger'];
function DashboardPanelController(logger) {
}
tips 27(手动解析路由依赖)
使用$inject来手动识别Angular组件的路由解析依赖。
推荐如下:
function config($routeProvider) {
$routeProvider
.when('/avengers', {
templateUrl: 'avengers.html',
controller: 'AvengersController',
controllerAs: 'vm',
resolve: {
moviesPrepService: moviesPrepService
}
});
} moviesPrepService.$inject = ['movieService'];
function moviesPrepService(movieService) {
return movieService.getMovies();
}
异常处理
tips 28(用decorators来配置处理异常)
配置时使用$provider服务,当异常出现时在$exceptionHandler中使用decorator来处理异常。
提供一致的方式在运行时来处理未捕获的异常。
推荐如下:
angular
.module('blocks.exception')
.config(exceptionConfig); exceptionConfig.$inject = ['$provide']; function exceptionConfig($provide) {
$provide.decorator('$exceptionHandler', extendExceptionHandler);
} extendExceptionHandler.$inject = ['$delegate', 'toastr']; function extendExceptionHandler($delegate, toastr) {
return function(exception, cause) {
$delegate(exception, cause);
var errorData = {
exception: exception,
cause: cause
};
toastr.error(exception.msg, errorData);
};
}
tips 29(创建工厂并暴露其接口来捕获异常)
在代码执行过程中可能会抛出异常,我们需要提供统一的方式来捕获异常。
推荐如下:
angular
.module('blocks.exception')
.factory('exception', exception); exception.$inject = ['logger']; function exception(logger) {
var service = {
catcher: catcher
};
return service; function catcher(message) {
return function(reason) {
logger.error(message, reason);
};
}
}
tips 30(使用$document和$window代替document和window)
在AngularJS中存在$document和$window两个服务来代替document和window利于模拟和测试。
tips 31(使用$interval和$timeout代替interval和timeout)
在AngularJS中存在$interval和$timeout两个服务来代替interval和timeout利于测试和处理Angular的digest生命周期从而保持数据同步绑定。
命名
通过使用统一的命名方式来为所有组件命名,推荐的方式为feature.type.js。如下:
文件名:cnblogs.controller.js。
注册的组件名:CnblogsController。
tips 32(文件命名的特点)
避免如下:
// Controllers
avengers.js
avengers.controller.js
avengersController.js // Services/Factories
logger.js
logger.service.js
loggerService.js
推荐如下:
// controllers
avengers.controller.js
avengers.controller.spec.js // services/factories
logger.service.js
logger.service.spec.js // constants
constants.js // module definition
avengers.module.js // routes
avengers.routes.js
avengers.routes.spec.js // configuration
avengers.config.js // directives
avenger-profile.directive.js
avenger-profile.directive.spec.js
tips 33(控制器命名后缀为Controller)
控制器命名后缀是最常用且更明确、具体的描述。
推荐如下:
angular
.module
.controller('AvengersController', AvengersController); function AvengersController() { }
tips 34(工厂和服务命名)
根据其特征来统一为所有服务和工厂来命名,使用骆驼风格来命名。避免工厂和服务前缀使用$。
(1)提供一致的方式来快速识别和引用工厂。
(2)避免命名冲突。
(3)清除服务名称,如logger,不需要其后缀。
推荐如下:
// logger.service.js
angular
.module
.factory('logger', logger); function logger() { }
// credit.service.js
angular
.module
.factory('creditService', creditService); function creditService() { } // customer.service.js
angular
.module
.service('customerService', customerService); function customerService() { }
tips 36(指令组件命名)
通过使用骆驼风格来为指令组件统一命名,使用短前缀来描述这个区域信息(例如:前缀可为公司名称或者项目名称)。
提供统一的方式来快速识别和引用组件
推荐如下:
// cnblogs-profile.directive.js
angular
.module
.directive('xxCnblogsProfile', xxCnblogsrProfile); // usage is <xx-cnblogs-profile> </xx-cnblogs-profile> function xxCnblogsProfile() { }
总结
本节我们讲了在AngularJS中的代码风格,我们可以一定不需要这样做,但是我们推荐这样做。
AngularJS之代码风格36条建议【一】(九)的更多相关文章
- 优化PHP代码的40条建议(转载)
[size=5][color=Red](译文)优化PHP代码的40条建议[/color][/size] 40 Tips for optimizing your php Code 原文地址:http:/ ...
- C# 开发者审查代码的41条建议
1. 确保没有任何警告(warnings). 2.如果先执行Code Analysis(启用所有Microsoft Rules)再消除所有警告就更好了. 3. 去掉所有没有用到的usings.编码过程 ...
- 读读《编写高质量代码:改善Java程序的151条建议》
这本书可以作为平时写代码的一个参考书,这本书以我个人读的经验看来,最好是通过平时代码驱动的方式来读,这样吸收的快,也读的快. 这本书主要讲什么,我自己用了个思维导图概述: 根据这种导图可知,主要讲的就 ...
- Python 代码风格
1 原则 在开始讨论Python社区所采用的具体标准或是由其他人推荐的建议之前,考虑一些总体原则非常重要. 请记住可读性标准的目标是提升可读性.这些规则存在的目的就是为了帮助人读写代码,而不是相反. ...
- python代码风格-PEP8
转载自http://www.douban.com/note/134971609/ Python 的代码风格由 PEP 8 描述.这个文档描述了 Python 编程风格的方方面面.在遵守这个文档的条件下 ...
- ZZC语言代码风格
程序员之路--关于代码风格 优秀的代码风格如同一身得体的打扮,能够给人以良好的印象.初学程序设计,首先必须建立良好的编程习惯,这其中就包括代码风格.本文就代码风格中的几个重点问题进行了讨论,并在文后给 ...
- 【转】python代码风格-PEP8
转载自http://www.douban.com/note/134971609/ Python 的代码风格由 PEP 8 描述.这个文档描述了 Python 编程风格的方方面面.在遵守这个文档的条件下 ...
- PHP PSR 代码风格
FIG组织在制定跟PHP相关规范,简称PSR,PSR旨在通过讨论我们代码项目的共同点以找出一个协作编程的方法. 什么是psr0 强调自动加载的方式 下文描述了若要使用一个通用的自动加载器(autolo ...
- 点评阿里JAVA手册之编程规约(命名风格、常量定义、代码风格、控制语句、注释规约)
下载原版阿里JAVA开发手册 [阿里巴巴Java开发手册v1.2.0] 本文主要是对照阿里开发手册,注释自己在工作中运用情况. 本文难度系数为一星(★) 码出高效.码出质量. 代码的字里行间流淌的是 ...
随机推荐
- SqlServer windowss身份登陆和sa身份登陆
今天重新装了系统,但是计算机名变了,于是修改了计算机名,然后装了SQLSEVER,安装完成后登录,发现无论用WINDOWS身份还是SQLSERVER身份都登录不了 1.先说说sqlserver身份登录 ...
- 二分图&网络流&最小割等问题的总结
二分图基础: 最大匹配:匈牙利算法 最小点覆盖=最大匹配 最小边覆盖=总节点数-最大匹配 最大独立集=点数-最大匹配 网络流: 技巧: 1.拆点为边,即一个点有限制,可将其转化为边 BZOJ1066, ...
- 使用spring的@Scheduled注解执行定时任务,启动项目不输出警告
在applicationContext.xml中添加: xmlns:task="http://www.springframework.org/schema/task" xsi:sc ...
- STM32之通用定时器
广大的互联网的大家早上中午晚上..又好..没错了..我又来了..写博客不是定时的..为什么我要提写博客不是定时的呢??聪明的人又猜到我要说什么了吧.有前途.其实我还是第一次听到定时器有通用和高级之分的 ...
- Java反编译代码对齐
使用反编译的代码作为jar包源码进行调试时,经常会遇到的情况是反编译后的源码之在注释里包含行号,但是与代码所在行经常对应不上.这个时候,就有必要对代码进行对齐了. public class Reo ...
- python 函数之装饰器,迭代器,生成器
装饰器 了解一点:写代码要遵循开发封闭原则,虽然这个原则是面向对象开发,但也适用于函数式编程,简单的来说,就是已经实现的功能代码不允许被修改但 可以被扩展即: 封闭:已实现功能的代码块 开发:对扩张开 ...
- c++ 宏定义声明类,并在类中实现回调
#include <iostream> #include <windows.h> #include <string> using namespace std; ty ...
- linux下安装启动rpc服务
1.上传包 rocky:~ # ls Desktop dts.xml jdk1..0_41 oswbb rpc.rstatd- rpc.rstatd-.tar.gz rocky:~ # cd rpc. ...
- tomcat相关
一.下面这篇文章介绍了tomcat log相关内容 http://blog.csdn.net/cowmich/article/details/8173005
- mysql 命令行还原备份数据库
通常数据库还原备份可以通过navicat等数据库管理工具进行,只需要简单的导出导入就行了,但遇到有索引外键的数据库,数据库管理工具运行.sql文件会报错,这时候可以尝试命令行导入,亲测可以成功 MyS ...