STL标准中没有区分基本算法或复杂算法,单SGI把常用的一些算法定义在<stl_algobase.h>只中,其他算法定义在<stl_algo.h>中。

stl_algobase.h中的算法,比较值得学习的是copy(),它“无所不用其极”的改善效率。copy的目的是复制一段元素到指定区间,复制操作最容易想到赋值操作符=,但是有的赋值操作符=是trivial的,可以直接拷贝。关于赋值操作符=是不是trivial的,可以参考“Memberwise copy(深拷贝)与Bitwise copy(浅拷贝)的区别”。

Memberwise copy: 在初始化一个对象期间,基类的构造函数被调用,成员变量被调用,如果它们有构造函数的时候,它们的构造函数被调用,这个过程是一个递归的过程.

Bitwise copy: 原内存拷贝.例子,给定一个对象object,它的类型是class Base.对象object占用10字节的内存,地址从0x0到0x9.如果还有一个对象objectTwo,类型也是class Base.那么执行objectTwo = object;如果使用Bitwise拷贝语义,那么将会拷贝从0x0到0x9的数据到objectTwo的内存地址,.也就是说Bitwise是字节到字节的拷贝.

对于默认的拷贝构造函数不会使用深拷贝,它只是使用浅拷贝.这意味着类的所有的成员是一层深度的拷贝而已。如果你的类或结构体成员中只是包含基本的数据类型例如int, float, char,那么Memberwise copy与Bitwise copy基本是相同的。但如果类中有指针存在,那么你可能会遇到问题。

例如下面的例子:

class A
{int m1;double d1;char* pString;
};

如果你创建两个这样的类对象,class A  a, b;并且你给a赋值,

a.mi = 6;
a.d1 = 10.123;
a.pString = new char[10];
astrcpy(a.pString, "test");//这里是浅拷贝

如果执行b = a;那么会把对象a的每一个成员的值赋值给b的每个成员。

b.m1 = a.m1;
b.d1 = a.d1;
b.pString = a.pString;//现在对象a和b的成员pString都执向相同的内存,删除任一个内存都会析放另一个对象的内存。

所以你需要深拷贝,它不是拷贝的内存地址而是拷贝内存地址的内容。一个默认的拷贝构造函数经常执行浅拷贝,只有拥有
自己的拷贝函数才可以实现深拷贝。

补充:

在BitWise Copy Sematics中,因为是按位拷贝的(内存复制),所以那些整数、数组等都会拷贝,新得到的类和原来的类完全一样。但是需要注意一点,如果有指针时,例如

Class Test{
……;
char * p;
};
Test B=A;//A是Test类的对象

这时候如果是BitWise Copy,那么A和B中的指针p就会指向同一内存,如果内存在构造函数中释放,那么另一个类的指针将失效。

在一下4中情况,不要BitWise Copy

1、当Class内的成员变量是一个类,这各类声明了Copy Constructor(不是编译器默认合成)时。

2、当Class继承自一个Base Class时,而Base Class存在Copy Constructor时(不是编译器默认合成)。

3、当Class中声明有virtual function时

4、当Class继承自一个继承串链,其中一个或多个virtual Base Class时。

下面是copy调用过程的讲解。

// Filename:    stl_algobase.h// 这个文件中定义的都是一些最常用的算法, 我仅仅给出一个思路,
// 不进行详尽讲解, 具体算法请参考算法书籍, 推荐《算法导论》
// 另外, 对于基础薄弱的, 推荐《大话数据结构》, 此书我读了一下
// 试读章节, 适合初学者学习/*** Copyright (c) 1994* Hewlett-Packard Company** Permission to use, copy, modify, distribute and sell this software* and its documentation for any purpose is hereby granted without fee,* provided that the above copyright notice appear in all copies and* that both that copyright notice and this permission notice appear* in supporting documentation.  Hewlett-Packard Company makes no* representations about the suitability of this software for any* purpose.  It is provided "as is" without express or implied warranty.*** Copyright (c) 1996* Silicon Graphics Computer Systems, Inc.** Permission to use, copy, modify, distribute and sell this software* and its documentation for any purpose is hereby granted without fee,* provided that the above copyright notice appear in all copies and* that both that copyright notice and this permission notice appear* in supporting documentation.  Silicon Graphics makes no* representations about the suitability of this software for any* purpose.  It is provided "as is" without express or implied warranty.*//* NOTE: This is an internal header file, included by other STL headers.*   You should not attempt to use it directly.*/#ifndef __SGI_STL_INTERNAL_ALGOBASE_H
#define __SGI_STL_INTERNAL_ALGOBASE_H#ifndef __STL_CONFIG_H
#include <stl_config.h>
#endif
#ifndef __SGI_STL_INTERNAL_RELOPS
#include <stl_relops.h>
#endif
#ifndef __SGI_STL_INTERNAL_PAIR_H
#include <stl_pair.h>
#endif
#ifndef __TYPE_TRAITS_H_
#include <type_traits.h>
#endif#include <string.h>
#include <limits.h>
#include <stdlib.h>
#include <stddef.h>
#include <new.h>
#include <iostream.h>#ifndef __SGI_STL_INTERNAL_ITERATOR_H
#include <stl_iterator.h>
#endif__STL_BEGIN_NAMESPACE// 第三个参数为什么为指针参见<stl_iterator.h>
template <class ForwardIterator1, class ForwardIterator2, class T>
inline void __iter_swap(ForwardIterator1 a, ForwardIterator2 b, T*)
{// 这里交换的其实是内部对象T tmp = *a;*a = *b;*b = tmp;
}template <class ForwardIterator1, class ForwardIterator2>
inline void iter_swap(ForwardIterator1 a, ForwardIterator2 b)
{// 型别以第一个为准__iter_swap(a, b, value_type(a));
}// 进行交换操作, 使用的是operator =()
template <class T>
inline void swap(T& a, T& b)
{T tmp = a;a = b;b = tmp;
}#ifndef __BORLANDC__#undef min
#undef max// max和min非常简单了, 由于返回的是引用, 因此可以嵌套使用
template <class T>
inline const T& min(const T& a, const T& b)
{return b < a ? b : a;
}template <class T>
inline const T& max(const T& a, const T& b)
{return  a < b ? b : a;
}#endif /* __BORLANDC__ */template <class T, class Compare>
inline const T& min(const T& a, const T& b, Compare comp)
{return comp(b, a) ? b : a;
}template <class T, class Compare>
inline const T& max(const T& a, const T& b, Compare comp)
{return comp(a, b) ? b : a;
}// 这是不支持随机访问的情况
template <class InputIterator, class OutputIterator>
inline OutputIterator __copy(InputIterator first, InputIterator last,OutputIterator result, input_iterator_tag)
{// first != last导致要进行迭代器的比较, 效率低for ( ; first != last; ++result, ++first)*result = *first;return result;
}template <class RandomAccessIterator, class OutputIterator, class Distance>
inline OutputIterator
__copy_d(RandomAccessIterator first, RandomAccessIterator last,OutputIterator result, Distance*)
{// 不进行迭代器间的比较, 直接指定循环次数, 高效for (Distance n = last - first; n > 0; --n, ++result, ++first)*result = *first;return result;
}// 这是支持随机访问的情况
template <class RandomAccessIterator, class OutputIterator>
inline OutputIterator
__copy(RandomAccessIterator first, RandomAccessIterator last,OutputIterator result, random_access_iterator_tag)
{return __copy_d(first, last, result, distance_type(first));
}template <class InputIterator, class OutputIterator>
struct __copy_dispatch
{// 这里是一个仿函数. 再次派发OutputIterator operator()(InputIterator first, InputIterator last,OutputIterator result) {return __copy(first, last, result, iterator_category(first));}
};// 提供兼容
#ifdef __STL_CLASS_PARTIAL_SPECIALIZATION// 可以直接移动, 不需要额外操作
template <class T>
inline T* __copy_t(const T* first, const T* last, T* result, __true_type)
{memmove(result, first, sizeof(T) * (last - first));return result + (last - first);
}// 需要进行一些处理, 保证对象复制的正确性
template <class T>
inline T* __copy_t(const T* first, const T* last, T* result, __false_type)
{return __copy_d(first, last, result, (ptrdiff_t*) 0);
}// 对指针提供特化
template <class T>
struct __copy_dispatch<T*, T*>
{T* operator()(T* first, T* last, T* result){// 判断其内部是否具有trivial_assignment_operator, 以进行派发typedef typename __type_traits<T>::has_trivial_assignment_operator t;return __copy_t(first, last, result, t());}
};template <class T>
struct __copy_dispatch<const T*, T*>
{T* operator()(const T* first, const T* last, T* result) {typedef typename __type_traits<T>::has_trivial_assignment_operator t;return __copy_t(first, last, result, t());}
};#endif /* __STL_CLASS_PARTIAL_SPECIALIZATION */// 将[first, last)拷贝到result处
template <class InputIterator, class OutputIterator>
inline OutputIterator copy(InputIterator first, InputIterator last,OutputIterator result)
{// 此处进行函数派发操作return __copy_dispatch<InputIterator,OutputIterator>()(first, last, result);
}// 针对char字符串的特化, 效率至上, C++的设计理念
inline char* copy(const char* first, const char* last, char* result)
{memmove(result, first, last - first);return result + (last - first);
}// 针对wchar_t字符串的特化, 效率至上, C++的设计理念
inline wchar_t* copy(const wchar_t* first, const wchar_t* last,wchar_t* result) {memmove(result, first, sizeof(wchar_t) * (last - first));return result + (last - first);
}template <class BidirectionalIterator1, class BidirectionalIterator2>
inline BidirectionalIterator2 __copy_backward(BidirectionalIterator1 first,BidirectionalIterator1 last,BidirectionalIterator2 result)
{while (first != last) *--result = *--last;return result;
}template <class BidirectionalIterator1, class BidirectionalIterator2>
struct __copy_backward_dispatch
{BidirectionalIterator2 operator()(BidirectionalIterator1 first,BidirectionalIterator1 last,BidirectionalIterator2 result){return __copy_backward(first, last, result);}
};#ifdef __STL_CLASS_PARTIAL_SPECIALIZATIONtemplate <class T>
inline T* __copy_backward_t(const T* first, const T* last, T* result,__true_type)
{const ptrdiff_t N = last - first;memmove(result - N, first, sizeof(T) * N);return result - N;
}template <class T>
inline T* __copy_backward_t(const T* first, const T* last, T* result,__false_type)
{return __copy_backward(first, last, result);
}template <class T>
struct __copy_backward_dispatch<T*, T*>
{T* operator()(T* first, T* last, T* result){typedef typename __type_traits<T>::has_trivial_assignment_operator t;return __copy_backward_t(first, last, result, t());}
};template <class T>
struct __copy_backward_dispatch<const T*, T*>
{T* operator()(const T* first, const T* last, T* result){typedef typename __type_traits<T>::has_trivial_assignment_operator t;return __copy_backward_t(first, last, result, t());}
};#endif /* __STL_CLASS_PARTIAL_SPECIALIZATION */// 将[first, last)的元素反向拷贝到(..., last)处, 其机制和copy非常接近, 不做说明
template <class BidirectionalIterator1, class BidirectionalIterator2>
inline BidirectionalIterator2 copy_backward(BidirectionalIterator1 first,BidirectionalIterator1 last,BidirectionalIterator2 result)
{return __copy_backward_dispatch<BidirectionalIterator1,BidirectionalIterator2>()(first, last,result);
}template <class InputIterator, class Size, class OutputIterator>
pair<InputIterator, OutputIterator> __copy_n(InputIterator first, Size count,OutputIterator result,input_iterator_tag)
{for ( ; count > 0; --count, ++first, ++result)*result = *first;return pair<InputIterator, OutputIterator>(first, result);
}template <class RandomAccessIterator, class Size, class OutputIterator>
inline pair<RandomAccessIterator, OutputIterator>
__copy_n(RandomAccessIterator first, Size count,OutputIterator result,random_access_iterator_tag)
{// 使用copy()以选择最高效的拷贝算法RandomAccessIterator last = first + count;return pair<RandomAccessIterator, OutputIterator>(last,copy(first, last, result));
}// 从first拷贝n个值到result处
template <class InputIterator, class Size, class OutputIterator>
inline pair<InputIterator, OutputIterator>
copy_n(InputIterator first, Size count,OutputIterator result)
{// 进行函数派发, 选咋高效版本return __copy_n(first, count, result, iterator_category(first));
}// 使用value填充[first, last)区间
template <class ForwardIterator, class T>
void fill(ForwardIterator first, ForwardIterator last, const T& value)
{for ( ; first != last; ++first)*first = value;     // 调用的是operator =(), 这个要特别注意
}// 用value填充[first, first + n)的区间
// 为了防止越界, 可以使用下面实例的技巧
// vector<int> vec();
// for (int i = 0; i < 10; ++i)
//      vec.push_back(i);
// fill_n(inserter(iv, iv.begin()), 100, 10);   // 这就可以使容器动态扩展
template <class OutputIterator, class Size, class T>
OutputIterator fill_n(OutputIterator first, Size n, const T& value)
{for ( ; n > 0; --n, ++first)*first = value;return first;
}// 找到两个序列第一个失配的地方, 结果以pair返回
template <class InputIterator1, class InputIterator2>
pair<InputIterator1, InputIterator2> mismatch(InputIterator1 first1,InputIterator1 last1,InputIterator2 first2)
{// 遍历区间, 寻找失配点while (first1 != last1 && *first1 == *first2) {++first1;++first2;}return pair<InputIterator1, InputIterator2>(first1, first2);
}// 提供用户自定义的二元判别式, 其余同上
template <class InputIterator1, class InputIterator2, class BinaryPredicate>
pair<InputIterator1, InputIterator2> mismatch(InputIterator1 first1,InputIterator1 last1,InputIterator2 first2,BinaryPredicate binary_pred)
{while (first1 != last1 && binary_pred(*first1, *first2)) {++first1;++first2;}return pair<InputIterator1, InputIterator2>(first1, first2);
}// 如果序列在[first, last)内相等, 则返回true, 如果第二个序列有多余的元素,
// 则不进行比较, 直接忽略. 如果第二个序列元素不足, 会导致未定义行为
template <class InputIterator1, class InputIterator2>
inline bool equal(InputIterator1 first1, InputIterator1 last1,InputIterator2 first2)
{for ( ; first1 != last1; ++first1, ++first2)if (*first1 != *first2)     // 只要有一个不相等就判定为falsereturn false;return true;
}// 进行比较的操作改为用户指定的二元判别式, 其余同上
template <class InputIterator1, class InputIterator2, class BinaryPredicate>
inline bool equal(InputIterator1 first1, InputIterator1 last1,InputIterator2 first2, BinaryPredicate binary_pred)
{for ( ; first1 != last1; ++first1, ++first2)if (!binary_pred(*first1, *first2))return false;return true;
}// 字典序比较, 非常类似字符串的比较
// 具体比较方式参见STL文档, 另外strcmp()也可以参考
template <class InputIterator1, class InputIterator2>
bool lexicographical_compare(InputIterator1 first1, InputIterator1 last1,InputIterator2 first2, InputIterator2 last2)
{for ( ; first1 != last1 && first2 != last2; ++first1, ++first2) {if (*first1 < *first2)return true;if (*first2 < *first1)return false;}return first1 == last1 && first2 != last2;
}// 二元判别式自己指定, 其余同上
template <class InputIterator1, class InputIterator2, class Compare>
bool lexicographical_compare(InputIterator1 first1, InputIterator1 last1,InputIterator2 first2, InputIterator2 last2,Compare comp) {for ( ; first1 != last1 && first2 != last2; ++first1, ++first2){if (comp(*first1, *first2))return true;if (comp(*first2, *first1))return false;}return first1 == last1 && first2 != last2;
}// 针对字符串的特化, 效率至上
inline bool
lexicographical_compare(const unsigned char* first1,const unsigned char* last1,const unsigned char* first2,const unsigned char* last2)
{const size_t len1 = last1 - first1;const size_t len2 = last2 - first2;const int result = memcmp(first1, first2, min(len1, len2));return result != 0 ? result < 0 : len1 < len2;
}// 针对字符串的特化, 效率至上
inline bool lexicographical_compare(const char* first1, const char* last1,const char* first2, const char* last2)
{
#if CHAR_MAX == SCHAR_MAXreturn lexicographical_compare((const signed char*) first1,(const signed char*) last1,(const signed char*) first2,(const signed char*) last2);
#elsereturn lexicographical_compare((const unsigned char*) first1,(const unsigned char*) last1,(const unsigned char*) first2,(const unsigned char*) last2);
#endif
}// 一句话概括, 这个是strcmp()的泛化版本
template <class InputIterator1, class InputIterator2>
int lexicographical_compare_3way(InputIterator1 first1, InputIterator1 last1,InputIterator2 first2, InputIterator2 last2)
{while (first1 != last1 && first2 != last2) {if (*first1 < *first2) return -1;if (*first2 < *first1) return 1;++first1; ++first2;}if (first2 == last2) {return !(first1 == last1);} else {return -1;}
}// 特换版本, 效率决定一切
inline int
lexicographical_compare_3way(const unsigned char* first1,const unsigned char* last1,const unsigned char* first2,const unsigned char* last2)
{const ptrdiff_t len1 = last1 - first1;const ptrdiff_t len2 = last2 - first2;const int result = memcmp(first1, first2, min(len1, len2));return result != 0 ? result : (len1 == len2 ? 0 : (len1 < len2 ? -1 : 1));
}inline int lexicographical_compare_3way(const char* first1, const char* last1,const char* first2, const char* last2)
{
#if CHAR_MAX == SCHAR_MAXreturn lexicographical_compare_3way((const signed char*) first1,(const signed char*) last1,(const signed char*) first2,(const signed char*) last2);
#elsereturn lexicographical_compare_3way((const unsigned char*) first1,(const unsigned char*) last1,(const unsigned char*) first2,(const unsigned char*) last2);
#endif
}__STL_END_NAMESPACE#endif /* __SGI_STL_INTERNAL_ALGOBASE_H */// Local Variables:
// mode:C++
// End:

实例:

// swap algorithm example (C++11)
#include <iostream>     // std::cout
#include <utility>      // std::swapint main () {int x=10, y=20;                  // x:10 y:20std::swap(x,y);                  // x:20 y:10int foo[4];                      // foo: ?  ?  ?  ?int bar[] = {10,20,30,40};       // foo: ?  ?  ?  ?    bar: 10 20 30 40std::swap(foo,bar);              // foo: 10 20 30 40   bar: ?  ?  ?  ?std::cout << "foo contains:";for (int i: foo) std::cout << ' ' << i;std::cout << '\n';return 0;
}

Output:

foo contains: 10 20 30 40
// fill algorithm example
#include <iostream>     // std::cout
#include <algorithm>    // std::fill
#include <vector>       // std::vectorint main () {std::vector<int> myvector (8);                       // myvector: 0 0 0 0 0 0 0 0std::fill (myvector.begin(),myvector.begin()+4,5);   // myvector: 5 5 5 5 0 0 0 0std::fill (myvector.begin()+3,myvector.end()-2,8);   // myvector: 5 5 5 8 8 8 0 0std::cout << "myvector contains:";for (std::vector<int>::iterator it=myvector.begin(); it!=myvector.end(); ++it)std::cout << ' ' << *it;std::cout << '\n';return 0;
}

Output:

myvector contains: 5 5 5 8 8 8 0 0
// fill_n example
#include <iostream>     // std::cout
#include <algorithm>    // std::fill_n
#include <vector>       // std::vectorint main () {std::vector<int> myvector (8,10);        // myvector: 10 10 10 10 10 10 10 10std::fill_n (myvector.begin(),4,20);     // myvector: 20 20 20 20 10 10 10 10std::fill_n (myvector.begin()+3,3,33);   // myvector: 20 20 20 33 33 33 10 10std::cout << "myvector contains:";for (std::vector<int>::iterator it=myvector.begin(); it!=myvector.end(); ++it)std::cout << ' ' << *it;std::cout << '\n';return 0;
}

Output:

myvector contains: 20 20 20 33 33 33 10 10
// equal algorithm example
#include <iostream>     // std::cout
#include <algorithm>    // std::equal
#include <vector>       // std::vectorbool mypredicate (int i, int j) {return (i==j);
}int main () {int myints[] = {20,40,60,80,100};               //   myints: 20 40 60 80 100std::vector<int>myvector (myints,myints+5);     // myvector: 20 40 60 80 100// using default comparison:if ( std::equal (myvector.begin(), myvector.end(), myints) )std::cout << "The contents of both sequences are equal.\n";elsestd::cout << "The contents of both sequences differ.\n";myvector[3]=81;                                 // myvector: 20 40 60 81 100// using predicate comparison:if ( std::equal (myvector.begin(), myvector.end(), myints, mypredicate) )std::cout << "The contents of both sequences are equal.\n";elsestd::cout << "The contents of both sequences differ.\n";return 0;
}

Output:

The contents of both sequences are equal.
The contents of both sequence differ.

STL源码剖析——stl_algobase.h相关推荐

  1. 《STL源码剖析》学习-- 1.9-- 可能令你困惑的C++语法1

    最近在看侯捷的<STL源码剖析>,虽然感觉自己c++看得比较深一点,还是感觉还多东西不是那么明白,这里将一些细小的东西或者概念记录一下. 有些东西是根据<C++编程思想>理解的 ...

  2. STL源码剖析——P142关于list::sort函数

    在list容器中,由于容器自身组织数据的特殊性,所以list提供了自己的排序函数list::sort, 并且实现得相当巧妙,不过<STL源码剖析>的原文中,我有些许疑问,对于该排序算法,侯 ...

  3. STL源码剖析 stack 栈 概述->(使用deque双端队列 / list链表)作为stack的底层容器

    Stack是一种先进后出的数据结构,他只有一个出口 stack允许 新增元素.移除元素.取得最顶端的元素,但是无法获得stack的内部数据,因此satck没有遍历行为 Stack定义的完整列表 (双端 ...

  4. STL源码剖析 __type_traits

    traits编程 弥补了C++本身的不足 STL只对迭代器进行规范制定出了iterator_traits,SGI在此基础上进一步扩展,产生了__type_traits 双下划线的含义是这个是SGI内部 ...

  5. STL源码剖析 空间配置器 查漏补缺

    ptrdiff_t含义 减去两个指针的结果的带符号整数类型 ptrdiff_t (Type support) - C 中文开发手册 - 开发者手册 - 云+社区 - 腾讯云 std::set_new_ ...

  6. 《STL源码剖析》笔记——allocator

    六大组件间关系 部分STL文件包含关系 allocator包含于中: 实际实现于三个文件 : 1.stl_construct.h :对象的构造和析构 2.stl_alloc.h空间配置和释放 3.st ...

  7. STL源码剖析之配接器

    adapter(配接器)在STL组件的灵活组合运用上,扮演者转换器的角色.adapter来源于一种适配器模式,其功能是:将一个class接口转换为另一个class的接口,使得原本因接口不兼容而不能合作 ...

  8. 【STL源码剖析】list模拟实现 | 适配器实现反向迭代器【超详细的底层算法解释】

    今天博主继续带来STL源码剖析专栏的第三篇博客了! 今天带来list的模拟实现! 话不多说,直接进入我们今天的内容! 前言 那么这里博主先安利一下一些干货满满的专栏啦! 手撕数据结构https://b ...

  9. STL(C++标准库,体系结构及其内核分析)(STL源码剖析)(更新完毕)

    文章目录 介绍 Level 0:使用C++标准库 0 STL六大部件 0.1 六大部件之间的关系 0.2 复杂度 0.3 容器是前闭后开(左闭右开)区间 1 容器的结构与分类 1.1 使用容器Arra ...

  10. STL源码剖析学习七:stack和queue

    STL源码剖析学习七:stack和queue stack是一种先进后出的数据结构,只有一个出口. 允许新增.删除.获取最顶端的元素,没有任何办法可以存取其他元素,不允许有遍历行为. 缺省情况下用deq ...

最新文章

  1. 【CodeForces - 151D】Quantity of Strings (字符串问题,思维推导,有坑)
  2. 1 SAP DEBUG调试改表操作手册
  3. matlab 小波启发式阈值滤波,小波阈值去噪
  4. 重磅!容器存储解决方案蓝皮书发布
  5. 电影购票c语言程序,C语言电影购票系统小样
  6. PCB LAYOUT 设计民睿科技有限公司承接PCBLAYOUT项目
  7. struts2拦截器实现登录控制
  8. C语言变量inv,编写一个函数inv,将数组a中n个整数按相反顺序存放,用指针变量作为调用该函数时的实参...
  9. 分布式文件存储系统MinIO入门教程
  10. 在百度地图上展示dwg/dxf
  11. Sinon 入门,看这篇文章就够了
  12. 打开SAP物料帐期和财务账期
  13. HTTPS(ssl)
  14. 2012年桂城街道小学毕业生升初中
  15. 计算机网络按照延伸距离划分为,计算机应用基础填空题
  16. 拓视角丨稳固金融领域领军地位,强化数字化服务和体验
  17. 百度api翻译html,百度在线翻译API (使用实例)
  18. python 传智播客_使用Python自动播客发布
  19. DeepDream:使用深度学习再造毕加索抽象风格艺术画
  20. js-md5前后端使用

热门文章

  1. 深度学习:知识回收(神经网络模型:BPNN原理)
  2. ESXI7.0主机安装群晖DS3617xs
  3. Matlab符号计算与数值计算对比
  4. html5+自动更换背景图片,html 随机切换背景图片
  5. linux修改系统语言为中文
  6. 基于java的百货中心供应链管理系统
  7. SDRAM控制器设计(8)SDRAM控制器仿真验证
  8. esp ghost引导_ghost做uefi+gpt 修复引导(转自百度知道huanglenzhi)
  9. amr 转mp3 android,安卓批量amr转mp3 微信amr批量转换
  10. 锐捷交换机堆叠配置举例