(八)cmockery中的calculator和run_tests函数的注释代码

    所分析的calculator.c和calculator_test.c文件位于 工程中的 cmockery/src/example/ 目录下,是一个相对而言比较全面的样例程序,用到了cmockery项目中的大多数单元测试方法。基本上涵盖了之前所有的样例程序中的用法,还有两组测试是database操作的样例程序没有做分析也不准备去做了。在结束了这一个样例程序的分析之后,大概就可以将整个cmockery程序过了百分之70的样子,接下来就可以将整个项目代码过一遍了。
calculator.c
  1. #ifdef HAVE_CONFIG_H
  2. #include "config.h"
  3. #endif
  4. #include <assert.h>
  5. #ifdef HAVE_MALLOC_H
  6. #include <malloc.h>
  7. #endif
  8. #include <stdio.h>
  9. #include <stdlib.h>
  10. #include <string.h>
  11. #define UNIT_TESTING 1
  12. // If this is being built for a unit test.
  13. #if UNIT_TESTING
  14. /* Redirect printf to a function in the test application so it's possible to
  15. * test the standard output. */
  16. #ifdef printf
  17. #undef printf
  18. #endif // printf
  19. #define printf example_test_printf
  20. extern void print_message(const char *format, ...);
  21. extern int example_test_printf(const char *format, ...);
  22. /* Redirect fprintf to a function in the test application so it's possible to
  23. * test error messages. */
  24. #ifdef fprintf
  25. #undef fprintf
  26. #endif // fprintf
  27. #define fprintf example_test_fprintf
  28. extern int example_test_fprintf(FILE * const file, const char *format, ...);
  29. // Redirect assert to mock_assert() so assertions can be caught by cmockery.
  30. #ifdef assert
  31. #undef assert
  32. #endif // assert
  33. #define assert(expression) \
  34. mock_assert((int)(expression), #expression, __FILE__, __LINE__)
  35. void mock_assert(const int result, const char* expression, const char *file,
  36. const int line);
  37. /* Redirect calloc and free to test_calloc() and test_free() so cmockery can
  38. * check for memory leaks. */
  39. #ifdef calloc
  40. #undef calloc
  41. #endif // calloc
  42. #define calloc(num, size) _test_calloc(num, size, __FILE__, __LINE__)
  43. #ifdef free
  44. #undef free
  45. #endif // free
  46. #define free(ptr) _test_free(ptr, __FILE__, __LINE__)
  47. void* _test_calloc(const size_t number_of_elements, const size_t size,
  48. const char* file, const int line);
  49. void _test_free(void* const ptr, const char* file, const int line);
  50. /* main is defined in the unit test so redefine name of the the main function
  51. * here. */
  52. #define main example_main
  53. /* All functions in this object need to be exposed to the test application,
  54. * so redefine static to nothing. */
  55. #define static
  56. #endif // UNIT_TESTING
  57. // A binary arithmetic integer operation (add, subtract etc.)
  58. typedef int (*BinaryOperator)(int a, int b);
  59. // Structure which maps operator strings to functions.
  60. typedef struct OperatorFunction {
  61. const char* operator;
  62. BinaryOperator function;
  63. } OperatorFunction;
  64. int add(int a, int b);
  65. int subtract(int a, int b);
  66. int multiply(int a, int b);
  67. int divide(int a, int b);
  68. // Associate operator strings to functions.
  69. static OperatorFunction operator_function_map[] = {
  70. {"+", add},
  71. {"-", subtract},
  72. {"*", multiply},
  73. {"/", divide},
  74. };
  75. int add(int a, int b) {
  76. return a + b;
  77. }
  78. int subtract(int a, int b) {
  79. return a - b;
  80. }
  81. int multiply(int a, int b) {
  82. return a * b;
  83. }
  84. int divide(int a, int b) {
  85. assert(b); // Check for divide by zero.
  86. return a / b;
  87. }
  88. /* Searches the specified array of operator_functions for the function
  89. * associated with the specified operator_string. This function returns the
  90. * function associated with operator_string if successful, NULL otherwise.
  91. */
  92. BinaryOperator find_operator_function_by_string(
  93. const size_t number_of_operator_functions,
  94. const OperatorFunction * const operator_functions,
  95. const char* const operator_string) {
  96. size_t i;
  97. assert(!number_of_operator_functions || operator_functions);
  98. assert(operator_string);
  99. for (i = 0; i < number_of_operator_functions; i++) {
  100. const OperatorFunction *const operator_function =
  101. &operator_functions[i];
  102. if (strcmp(operator_function->operator, operator_string) == 0) {
  103. return operator_function->function;
  104. }
  105. }
  106. return NULL;
  107. }
  108. /* Perform a series of binary arithmetic integer operations with no operator
  109. * precedence.
  110. *
  111. * The input expression is specified by arguments which is an array of
  112. * containing number_of_arguments strings. Operators invoked by the expression
  113. * are specified by the array operator_functions containing
  114. * number_of_operator_functions, OperatorFunction structures. The value of
  115. * each binary operation is stored in a pointer returned to intermediate_values
  116. * which is allocated by malloc().
  117. *
  118. * If successful, this function returns the integer result of the operations.
  119. * If an error occurs while performing the operation error_occurred is set to
  120. * 1, the operation is aborted and 0 is returned.
  121. */
  122. int perform_operation(
  123. int number_of_arguments, char *arguments[],
  124. const size_t number_of_operator_functions,
  125. const OperatorFunction * const operator_functions,
  126. int * const number_of_intermediate_values,
  127. int ** const intermediate_values, int * const error_occurred) {
  128. char *end_of_integer;
  129. int value;
  130. unsigned int i;
  131. assert(!number_of_arguments || arguments);
  132. assert(!number_of_operator_functions || operator_functions);
  133. assert(error_occurred);
  134. assert(number_of_intermediate_values);
  135. assert(intermediate_values);
  136. *error_occurred = 0;
  137. *number_of_intermediate_values = 0;
  138. *intermediate_values = NULL;
  139. if (!number_of_arguments)
  140. return 0;
  141. // Parse the first value.
  142. value = (int)strtol(arguments[0], &end_of_integer, 10);
  143. if (end_of_integer == arguments[0]) {
  144. // If an error occurred while parsing the integer.
  145. fprintf(stderr, "Unable to parse integer from argument %s\n",
  146. arguments[0]);
  147. *error_occurred = 1;
  148. return 0;
  149. }
  150. // Allocate an array for the output values.
  151. *intermediate_values = calloc(((number_of_arguments - 1) / 2),
  152. sizeof(**intermediate_values));
  153. i = 1;
  154. while (i < number_of_arguments) {
  155. int other_value;
  156. const char* const operator_string = arguments[i];
  157. const BinaryOperator function = find_operator_function_by_string(
  158. number_of_operator_functions, operator_functions, operator_string);
  159. int * const intermediate_value =
  160. &((*intermediate_values)[*number_of_intermediate_values]);
  161. (*number_of_intermediate_values) ++;
  162. if (!function) {
  163. fprintf(stderr, "Unknown operator %s, argument %d\n",
  164. operator_string, i);
  165. *error_occurred = 1;
  166. break;
  167. }
  168. i ++;
  169. if (i == number_of_arguments) {
  170. fprintf(stderr, "Binary operator %s missing argument\n",
  171. operator_string);
  172. *error_occurred = 1;
  173. break;
  174. }
  175. other_value = (int)strtol(arguments[i], &end_of_integer, 10);
  176. if (end_of_integer == arguments[i]) {
  177. // If an error occurred while parsing the integer.
  178. fprintf(stderr, "Unable to parse integer %s of argument %d\n",
  179. arguments[i], i);
  180. *error_occurred = 1;
  181. break;
  182. }
  183. i ++;
  184. // Perform the operation and store the intermediate value.
  185. *intermediate_value = function(value, other_value);
  186. value = *intermediate_value;
  187. }
  188. if (*error_occurred) {
  189. free(*intermediate_values);
  190. *intermediate_values = NULL;
  191. *number_of_intermediate_values = 0;
  192. return 0;
  193. }
  194. return value;
  195. }
  196. int main(int argc, char *argv[]) {
  197. int return_value;
  198. int number_of_intermediate_values;
  199. int *intermediate_values;
  200. // Peform the operation.
  201. const int result = perform_operation(
  202. argc - 1, &argv[1],
  203. sizeof(operator_function_map) / sizeof(operator_function_map[0]),
  204. operator_function_map, &number_of_intermediate_values,
  205. &intermediate_values, &return_value);
  206. // If no errors occurred display the result.
  207. if (!return_value && argc > 1) {
  208. unsigned int i;
  209. unsigned int intermediate_value_index = 0;
  210. printf("%s\n", argv[1]);
  211. for (i = 2; i < argc; i += 2) {
  212. assert(intermediate_value_index < number_of_intermediate_values);
  213. printf(" %s %s = %d\n", argv[i], argv[i + 1],
  214. intermediate_values[intermediate_value_index++]);
  215. }
  216. printf("= %d\n", result);
  217. }
  218. if (intermediate_values) {
  219. free(intermediate_values);
  220. }
  221. return return_value;
  222. }
这个是calculator.c的源码,里面自带了一个main.c程序。这个文件可以直接编译:gcc calculator.c 然后生成 a.out之后  运行就是: ./a.out 1 + 2 - 3 + 4。我在做测试的时候发现不能很好的支持*号。
  1. // A binary arithmetic integer operation (add, subtract etc.)
  2. typedef int (*BinaryOperator)(int a, int b);
  3. // Structure which maps operator strings to functions.
  4. typedef struct OperatorFunction {
  5. const char* operator;
  6. BinaryOperator function;
  7. } OperatorFunction;
  8. int add(int a, int b);
  9. int subtract(int a, int b);
  10. int multiply(int a, int b);
  11. int divide(int a, int b);
  12. // Associate operator strings to functions.
  13. static OperatorFunction operator_function_map[] = {
  14. {"+", add},
  15. {"-", subtract},
  16. {"*", multiply},
  17. {"/", divide},
  18. };
  19. int add(int a, int b) {
  20. return a + b;
  21. }
  22. int subtract(int a, int b) {
  23. return a - b;
  24. }
  25. int multiply(int a, int b) {
  26. return a * b;
  27. }
  28. int divide(int a, int b) {
  29. assert(b); // Check for divide by zero.
  30. return a / b;
  31. }
这样就构造了一个结构体数组,用来保存运算符号和运算方法的一个映射关系,后续只要在operator_function_map数组里面去根据符号找到执行方法就可以了。

find_operator_function_by_string函数就是执行了上面说的这个功能,根据传入的操作符号找到对应的函数,并进行返回函数指针。

perform_operation函数中对于传入的参数进行解析,先解析第一个放到临时变量value中,然后再循环体重进行解析操作符获取计算方法的函数指针,然后再获取另一个操作数,进行执行计算函数,然后返回值会转储到value中方便下次继续使用。肯定有一些错误告警日志信息以及错误判断。最终如果参数无误,将计算的结果进行返回。还有就是在这个函数中会分配一块内存用来存储每一次计算得到的中间值信息。

最后的main函数就是调用测试的过程,后续应该把这个main.c该为example_main.c,否则会在calculator_test.c函数编译的过程中出错。

关于不能执行乘法的解释:
    在以./a.out 2 * 3  这个命令行进行执行程序的时候,会将其中*号解析为一个通配符,即当前目录下的所有内容名字的集合。  如果想正常调用乘法的话,可以将 * 号用单引号进行包起来就可以正常的调用了。还有就是这个方法只会从左向右进行计算,不会进行优先级的区分(有兴趣可以进一步完善哟)。
运行结果如下:

calculator_test.c函数中对于calculator.c函数中的方法进行了单元测试,需要在calculator.c的最开始添加一行  #define UNIT_TESTING 1  这样使能单元测试功能。然后执行make:
  1. test : cmockery.c calculator.c calculator_test.c
  2. gcc cmockery.c calculator.c calculator_test.c -g -O0 -o test
  3. clean:
  4. rm -f test

calculator_test.c中的main函数如下:
  1. int main(int argc, char* argv[]) {
  2. UnitTest tests[] = {
  3. unit_test(test_add),
  4. unit_test(test_subtract),
  5. unit_test(test_multiply),
  6. unit_test(test_divide),
  7. unit_test(test_divide_by_zero),
  8. unit_test(test_find_operator_function_by_string_null_functions),
  9. unit_test(test_find_operator_function_by_string_null_string),
  10. unit_test(test_find_operator_function_by_string_valid_null_functions),
  11. unit_test(test_find_operator_function_by_string_not_found),
  12. unit_test(test_find_operator_function_by_string_found),
  13. unit_test(test_perform_operation_null_args),
  14. unit_test(test_perform_operation_null_operator_functions),
  15. unit_test(test_perform_operation_null_number_of_intermediate_values),
  16. unit_test(test_perform_operation_null_intermediate_values),
  17. unit_test(test_perform_operation_no_arguments),
  18. unit_test(test_perform_operation_first_arg_not_integer),
  19. unit_test(test_perform_operation_unknown_operator),
  20. unit_test(test_perform_operation_missing_argument),
  21. unit_test(test_perform_operation_no_integer_after_operator),
  22. unit_test(test_perform_operation),
  23. unit_test(test_example_main_no_args),
  24. unit_test(test_example_main),
  25. };
  26. return run_tests(tests);
  27. }
其中的前四个单元测试就是简单的一个断言测试,取test_add为例,如下:
  1. // Ensure add() adds two integers correctly.
  2. void test_add(void **state) {
  3. assert_int_equal(add(3, 3), 6);
  4. assert_int_equal(add(3, -3), 0);
  5. }
test_divide_by_zero,test_find_operator_function_by_string_null_functions,test_find_operator_function_by_string_null_string测试,使用expect_assert_failure,确认100/0是非法的。否则会无法通过测试。

最核心的两个函数如下,重要的位置都加了注释。
  1. int _run_test(
  2. const char * const function_name, const UnitTestFunction Function,
  3. void ** const state, const UnitTestFunctionType function_type,
  4. const void* const heap_check_point) {
  5. const ListNode * const check_point = heap_check_point ?
  6. heap_check_point : check_point_allocated_blocks();
  7. void *current_state = NULL;
  8. int rc = 1;
  9. int handle_exceptions = 1;
  10. #ifdef _WIN32
  11. handle_exceptions = !IsDebuggerPresent();
  12. #endif // _WIN32
  13. #if UNIT_TESTING_DEBUG
  14. handle_exceptions = 0;
  15. #endif // UNIT_TESTING_DEBUG
  16. if (handle_exceptions) {
  17. #ifndef _WIN32
  18. unsigned int i;
  19. //注册新号处理函数,出现异常的时候,会跳到exception_handler这里去处理,
  20. //然后再exception_handler调用longjmp跳回到setjmp位置继续下一次单元测试。
  21. //会返回并保存默认的信号处理方发指针
  22. for (i = 0; i < ARRAY_LENGTH(exception_signals); i++) {
  23. default_signal_functions[i] = signal(
  24. exception_signals[i], exception_handler);
  25. }
  26. #else // _WIN32
  27. previous_exception_filter = SetUnhandledExceptionFilter(
  28. exception_filter);
  29. #endif // !_WIN32
  30. }
  31. if (function_type == UNIT_TEST_FUNCTION_TYPE_TEST) {
  32. print_message("%s: Starting test\n", function_name);
  33. }
  34. initialize_testing(function_name);
  35. global_running_test = 1;
  36. //在这里进行的setjmp,后续如果出现新号或者错误处理,后会返回这里继续else部分的执行。
  37. if (setjmp(global_run_test_env) == 0) {
  38. Function(state ? state : &current_state);
  39. fail_if_leftover_values(function_name);
  40. /* If this is a setup function then ignore any allocated blocks
  41. * only ensure they're deallocated on tear down. */
  42. if (function_type != UNIT_TEST_FUNCTION_TYPE_SETUP) {
  43. fail_if_blocks_allocated(check_point, function_name);
  44. }
  45. global_running_test = 0;
  46. if (function_type == UNIT_TEST_FUNCTION_TYPE_TEST) {
  47. print_message("%s: Test completed successfully.\n", function_name);
  48. }
  49. rc = 0;
  50. } else {
  51. global_running_test = 0;
  52. print_message("%s: Test failed.\n", function_name);
  53. }
  54. teardown_testing(function_name);
  55. if (handle_exceptions) {
  56. #ifndef _WIN32
  57. unsigned int i;
  58. //在这里回复默认的异常处理函数,每一个单元测试都要这么做一次。
  59. for (i = 0; i < ARRAY_LENGTH(exception_signals); i++) {
  60. signal(exception_signals[i], default_signal_functions[i]);
  61. }
  62. #else // _WIN32
  63. if (previous_exception_filter) {
  64. SetUnhandledExceptionFilter(previous_exception_filter);
  65. previous_exception_filter = NULL;
  66. }
  67. #endif // !_WIN32
  68. }
  69. return rc;
  70. }
  71. int _run_tests(const UnitTest * const tests, const size_t number_of_tests) {
  72. print_message("%d\n",sizeof(int));
  73. // 是否执行下一个测试.
  74. int run_next_test = 1;
  75. // 是否前一个测试执行失败了.
  76. int previous_test_failed = 0;
  77. //检查堆指针的状态
  78. const ListNode * const check_point = check_point_allocated_blocks();
  79. //当前的测试索引
  80. size_t current_test = 0;
  81. //已经执行过测试的个数
  82. size_t tests_executed = 0;
  83. //测试失败的数目
  84. size_t total_failed = 0;
  85. // setup 函数的个数
  86. size_t setups = 0;//(setup主要实现测试前的初始化工作,teardown主要实现测试完成后的垃圾回收工作)
  87. // teardown 函数的个数
  88. size_t teardowns = 0;
  89. /* 一个测试状态的栈. 一个状态是当测试的setup发生时入栈
  90. * 另个是当测试的teardown发生时出栈 */
  91. TestState* test_states = malloc(number_of_tests * sizeof(*test_states));
  92. size_t number_of_test_states = 0;
  93. //失败测试的名字指针数组,最多为所有的都失败
  94. const char** failed_names = malloc(number_of_tests *
  95. sizeof(*failed_names));
  96. void **current_state = NULL;
  97. //确保最大的整数类型要不小于一个指针类型
  98. assert_true(sizeof(LargestIntegralType) >= sizeof(void*));
  99. while (current_test < number_of_tests)
  100. {
  101. const ListNode *test_check_point = NULL;
  102. TestState *current_TestState;
  103. const UnitTest * const test = &tests[current_test++];
  104. if (!test->function)
  105. {//如果结构体中的函数指针为空,那么就continue
  106. continue;
  107. }
  108. //以run_next_test控制是否为test,区分setup必须首先被执行。
  109. switch (test->function_type)
  110. {
  111. case UNIT_TEST_FUNCTION_TYPE_TEST:
  112. run_next_test = 1;
  113. break;
  114. case UNIT_TEST_FUNCTION_TYPE_SETUP:
  115. {
  116. // Checkpoint the heap before the setup.
  117. current_TestState = &test_states[number_of_test_states++];
  118. current_TestState->check_point = check_point_allocated_blocks();
  119. test_check_point = current_TestState->check_point;
  120. current_state = &current_TestState->state;
  121. *current_state = NULL;
  122. run_next_test = 1;
  123. setups ++;
  124. break;
  125. }
  126. case UNIT_TEST_FUNCTION_TYPE_TEARDOWN:
  127. // Check the heap based on the last setup checkpoint.
  128. assert_true(number_of_test_states);
  129. current_TestState = &test_states[--number_of_test_states];
  130. test_check_point = current_TestState->check_point;
  131. current_state = &current_TestState->state;
  132. teardowns ++;
  133. break;
  134. default:
  135. print_error("Invalid unit test function type %d\n",
  136. test->function_type);
  137. exit_test(1);
  138. break;
  139. }
  140. if (run_next_test)
  141. {//进入测试单元进行运行函数
  142. int failed = _run_test(test->name, test->function, current_state,
  143. test->function_type, test_check_point);
  144. if (failed)
  145. {//记录错误函数的名字,后续会用来记录输出
  146. failed_names[total_failed] = test->name;
  147. }
  148. switch (test->function_type)
  149. {//统计信息
  150. case UNIT_TEST_FUNCTION_TYPE_TEST:
  151. previous_test_failed = failed;
  152. total_failed += failed;
  153. tests_executed ++;
  154. break;
  155. case UNIT_TEST_FUNCTION_TYPE_SETUP:
  156. if (failed)
  157. {
  158. total_failed ++;
  159. tests_executed ++;
  160. // Skip forward until the next test or setup function.
  161. run_next_test = 0;
  162. }
  163. previous_test_failed = 0;
  164. break;
  165. case UNIT_TEST_FUNCTION_TYPE_TEARDOWN:
  166. // If this test failed.
  167. if (failed && !previous_test_failed)
  168. {
  169. total_failed ++;
  170. }
  171. break;
  172. default:
  173. assert_false("BUG: shouldn't be here!");
  174. break;
  175. }
  176. }
  177. }
  178. //输出打印信息,并且释放申请的资源
  179. if (total_failed)
  180. {
  181. size_t i;
  182. print_error("%d out of %d tests failed!\n", total_failed,
  183. tests_executed);
  184. for (i = 0; i < total_failed; i++)
  185. {
  186. print_error(" %s\n", failed_names[i]);
  187. }
  188. }
  189. else
  190. {
  191. print_message("All %d tests passed\n", tests_executed);
  192. }
  193. if (number_of_test_states)
  194. {
  195. print_error("Mismatched number of setup %d and teardown %d "
  196. "functions\n", setups, teardowns);
  197. total_failed = -1;
  198. }
  199. free(test_states);
  200. free((void*)failed_names);
  201. //如果在这里还没有回复堆区域,那么就说明内存存在泄漏
  202. fail_if_blocks_allocated(check_point, "run_tests");
  203. return (int)total_failed;
  204. }
后续这个测试框架可以以静态库的方式存在于项目中,单独的简历一个测试文件夹,专门用于各个模块的单元测试代码。或者就编译成动态库安装到系统里,都是可以的。










转载于:https://www.cnblogs.com/cfzhang/p/a1d8bd91d72fbe49bd015471b049f2bb.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/371391.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

第一章魔兽窗口

开始显示第一个窗体 用户直接点登陆的话就会提示用户名不能为空密码不能为空 没有账号的话只能先注册&#xff0c;点击蓝色摁钮进入下一个窗体 这里有判断是否为空&#xff0c;注册成功后利用窗体传值&#xff0c;并且打开第一个窗口 把注册的用户名和密码写上去就可以的登陆到…

Apache Digester示例–轻松配置

解决问题–硬编码&#xff0c;需要为您的应用程序创建自定义配置&#xff0c;例如struts配置文件&#xff0c;以仅通过更改文件来改变应用程序行为。 Apache Digester可以轻松为您完成此任务。 使用Apache Digester相当容易将XML文档转换为相应的Java bean对象层次结构。 请参阅…

Java FlameGraph 火焰图

上周一个偶然的机会听同事提到了Java FlameGraph&#xff0c;刚实验了一下&#xff0c;效果非常好。 一、什么是FlameGraph 直接看图说话。FlameGraph 是 SVG格式&#xff0c;矢量图&#xff0c;可以随意扩大缩小&#xff0c;看不清的信息可以放大看。图中&#xff0c;各种红橙…

烟草局计算机笔试,2020年广西南宁烟草局什么时候笔试?

最近广西烟草局各地市社招通知频发&#xff0c;南宁烟草局报名截止至今都无任何消息&#xff0c;根据往年的考情&#xff0c;通知近期很大可能会发布&#xff0c;将于6月底完成笔面!你备考好了吗&#xff1f;今天广西中公国企小编来给大家说一下南宁烟草局社招的笔试内容及备考…

JAVA Swing 组件演示***

下面是Swing组件的演示&#xff1a; package a_swing;import java.awt.BorderLayout; import java.awt.Color; import java.awt.Container; import java.awt.Cursor; import java.awt.Dimension; import java.awt.GridLayout; import java.awt.event.ActionEvent; import java.…

centos 获取硬件序列号_如何在 Linux 上查找硬件规格

在 Linux 系统上有许多工具可用于查找硬件规格。-- Sk&#xff08;作者&#xff09;在 Linux 系统上有许多工具可用于查找硬件规格。在这里&#xff0c;我列出了四种最常用的工具&#xff0c;可以获取 Linux 系统的几乎所有硬件&#xff08;和软件&#xff09;细节。好在是这些…

Spring –持久层–编写实体并配置Hibernate

欢迎来到本教程的第二部分。 当您看到本文有多长时间时&#xff0c;请不要惊慌–我向您保证&#xff0c;这主要是简单的POJO和一些生成的代码。 在开始之前&#xff0c;我们需要更新我们的Maven依赖项&#xff0c;因为我们现在将使用Hibernate和Spring。 将以下依赖项添加到pom…

无线服务器主机名是,wifi默认服务器主机名

wifi默认服务器主机名 内容精选换一换以CentOS 7操作系统的弹性云服务器为例&#xff1a;登录Linux弹性云服务器&#xff0c;查看“cloud-init”的配置文件。检查“/etc/cloud/cloud.cfg”文件中“update_hostname”是否被注释或者删除。如果没有被注释或者删除&#xff0c;则需…

pygame里面物体闪烁运动_利用自闪烁发光二极管探究小车在倾斜轨道上的运动规律...

2020年11月23日&#xff0c;周一&#xff0c;24小时安全值班。利用当班中午的时间&#xff0c;微主在创客空间测试了自闪烁发光二极管在匀加速运动中的效果&#xff0c;结果还比较满意。将小车放置在倾斜的轨道上&#xff0c;将自闪烁发光二极管和纽扣电池构成频闪光源&#xf…

python网络爬虫与信息提取 学习笔记day3

Day3&#xff1a; 只需两行代码解析html或xml信息 具体代码实现:day3_1 注意BeautifulSoup的B和S需要大写&#xff0c;因为python大小写敏感 import requests r requests.get("http://python123.io/ws/demo.html") r.text demo r.text from bs4 import Beauti…

王者荣耀微信哪个服务器人最少,王者荣耀:微信区王者人数锐减,大神们都去哪了?这些原因很真实...

原标题&#xff1a;王者荣耀&#xff1a;微信区王者人数锐减&#xff0c;大神们都去哪了&#xff1f;这些原因很真实王者荣耀&#xff1a;微信区王者人数锐减&#xff0c;大神们都去哪了&#xff1f;这些原因很真实大家好&#xff01;王者荣耀S16赛季已经开启一月之余&#xff…

一个div压在另一个div上面_【CSS小分享】用CSS画一个新拟态风格键盘

什么是新拟态新拟态的英文名称是“Neumorphism”&#xff0c;也有人称为“Soft UI”。简单讲&#xff0c;新拟态是一种图形样式&#xff0c;其原理是通过模拟真实物体来为界面的UI元素赋予真实感。新拟态风格起源于dribbble&#xff0c;后面陆续被收录在2020设计趋势预测里面&a…

Hibernate中Hql查询

这篇随笔将会记录hql的常用的查询语句&#xff0c;为日后查看提供便利。 在这里通过定义了三个类&#xff0c;Special、Classroom、Student来做测试&#xff0c;Special与Classroom是一对多&#xff0c;Classroom与Student是一对多的关系&#xff0c;这里仅仅贴出这三个bean的属…

Java代码质量工具–概述

最近&#xff0c;我有机会在本地IT社区聚会上介绍了该主题。 这是基本演示&#xff1a; Java代码质量工具 以及更有意义的思维导图&#xff1a; 但是&#xff0c;我认为我需要更深入地探讨这一主题。 这篇博客文章应该像是在此方向上进行进一步调查的起点。 1. CodePro Anal…

cuda版本查看_ubuntu安装CUDA

0 写在前面安装环境&#xff1a;ubuntu18.04以及GTX1050Ti笔记本为什么要安装CUDA&#xff1f; 参考百科&#xff0c;CUDA是英伟达推出的集成技术&#xff0c;通过该技术可利用GeForce 8 以后的GPU或者较新的Quadro GPU进行计算。例如典型的tensorflow-GPU和pyCUDA安装之前都要…

idea新建scala文件_IDEA maven项目中新建.scala文件

本文首发于我的博客[IDEA maven项目中新建.scala文件]分为三步第一步、IDEA中安装scala插件1、搜索安装File-Sittings-Plugins-搜索安装scala2、安装完成重启安装完成之后点击重启idea第二步、下载、安装、配置Scala1、下载安装Scala SDK本体搜索引擎搜索Scala SDK或者点我去Sc…

Log4j线程死锁–案例研究

此案例研究描述了影响Weblogic Portal 10.0生产环境的Apache Log4j线程争用问题的完整根本原因分析和解决方案。 它还将说明在开发和支持Java EE应用程序时适当的Java类加载器知识的重要性。 本文也是您提高线程转储分析技能和了解线程竞争条件的另一个机会。 环境规格 Java …

堆栈跟踪从何而来?

我认为&#xff0c;阅读和理解堆栈跟踪是每个程序员都必须具备的一项基本技能&#xff0c;以便有效地解决每种JVM语言的问题&#xff08;另请参阅&#xff1a; 过滤日志中无关的堆栈跟踪行和首先记录引起异常的根本原因 &#xff09;。 那么我们可以从一个小测验开始吗&#xf…

@select 怎么写存储过程_MySQL4:存储过程和函数

什么是存储过程简单说&#xff0c;存储过程就是一条或多条SQL语句的集合&#xff0c;可视为批文件&#xff0c;但是起作用不仅限于批处理。本文主要讲解如何创建存储过程和存储函数以及变量的使用&#xff0c;如何调用、查看、修改、删除存储过程和存储函数等。使用的数据库和表…

跨域访问-预请求及跨域常见问题

预请求 参考&#xff1a;https://developer.mozilla.org/zh-CN/docs/Web/HTTP/Access_control_CORS#预请求 简而言之&#xff0c;在跨域并且尝试添加一些特殊头及自定义头的情况下&#xff0c;由于浏览器的安全机制&#xff0c;会加多一次OPTIONS预请求&#xff08;询问请求&am…