首页 > 其他分享 >深度好题

深度好题

时间:2024-11-25 11:11:32浏览次数:8  
标签:int 深度 mid 好题 100001 lca now id

突然想起来自己还不会回滚莫队,于是刷了几道,顺便昨天刚看的换根也练了几道题,这里挑了几个典型的说一下

窝太菜廖,可能觉得是深度好题的原因仅仅是我会打而已

不过起码没贺题解

思路显然是看了的,我不认为我有独立切掉紫题的能力

当然,不是紫题的题显然是没看题解

P1494 小 Z 的袜子

维护每种颜色在 \([L,R]\) 中出现的次数 \(c_i\),答案即为 \(\frac{\sum\limits C^{2}_{c_i}}{C_{r-l+1}^2}\)

莫队,单点增删的时候只需要维护 \(C^2_{c_{a_{i}}}\) 的改变量即可

#include<bits/stdc++.h>
using namespace std;
#define int long long
int n,m;
int c[50001];
int cnt[50001];
int len;
int locate[50001];
int ans=0;
struct ques{
    int l,r;
    int id;
    bool operator<(const ques&A)const{
        if(locate[l]==locate[A.l]) return r<A.r;
        return locate[l]<locate[A.l];
    }
}q[50001];
void prework(){
    len=sqrt(n);
    for(int i=1;i<=n;++i){
        locate[i]=i/len;
    }
}
void change(int pos,bool isadd){
    // cout<<"change "<<pos<<" "<<isadd<<endl;
    ans-=max(0ll,cnt[c[pos]]*(cnt[c[pos]]-1));
    cnt[c[pos]]+=(isadd?1:-1);
    ans+=max(0ll,cnt[c[pos]]*(cnt[c[pos]]-1));
    // cout<<"newans: "<<ans<<endl;
}
pair<int,int> answer[50001];
void printanswer(int l,int r,int id){
    int z=ans,m=(r-l+1)*(r-l);
    if(z==0){
        answer[id]={0,1};
        return;
    }
    int g=__gcd(z,m);
    answer[id]={z/g,m/g};
}
signed main(){
    ios::sync_with_stdio(false);
    cin>>n>>m;
    for(int i=1;i<=n;++i){
        cin>>c[i];
    }
    for(int i=1;i<=m;++i){
        cin>>q[i].l>>q[i].r;
        q[i].id=i;
    }
    prework();
    sort(q+1,q+m+1);
    int l=1,r=0;
    for(int i=1;i<=m;++i){
        // cout<<"goal "<<q[i].l<<" "<<q[i].r<<endl;
        while(r<q[i].r) change(++r,true);
        while(l>q[i].l) change(--l,true);
        while(l<q[i].l) change(l++,false);
        while(r>q[i].r) change(r--,false);
        printanswer(q[i].l,q[i].r,q[i].id);
    }
    for(int i=1;i<=m;++i){
        cout<<answer[i].first<<"/"<<answer[i].second<<'\n';
    }
}

JOI sc2014_c 歴史の研究

维护每个值出现的次数,可以发现在增加元素的时候,最大值是很方便更新的,唯一的瓶颈在于删除的时候无法快速找到一个新的最大值

因此回滚莫队

其实这好像是我第一次写回滚莫队,之前甚至不知道怎么回滚

现在会了,钦定一个端点在块内,另一个点保持单调性(不回滚)就行

一开始学的就是 stack 写法(相对于 memcpy 写法而言),感觉应该会快很多

#include<bits/stdc++.h>
using namespace std;
#define int long long
int n,Q,len;
int a[100001];
int locate[100001],rb[100001];
struct ques{
    int l,r;
    int id;
    bool operator<(const ques&A)const{
        if(locate[l]==locate[A.l]) return r<A.r;
        return locate[l]<locate[A.l];
    }
}q[100001];
int ans[100001];
inline void prework(){
    len=sqrt(n);
    for(int i=1;i<=n;++i){
        locate[i]=i/len;
        rb[locate[i]]=i;
    }
}
int b[100001];
int cnt[100001];
int vis2[100001];
int cnt2[100001];
int maxn;
int maxn2;
inline void add(int pos){
    // cout<<"add "<<a[pos]<<endl;
    cnt[a[pos]]++;
    maxn=max(maxn,cnt[a[pos]]*b[a[pos]]);
    // for(int i=1;i<=n;++i){
    //     cout<<cnt[i]<<' ';
    // }
    // cout<<endl;
}
int memoryans;
stack<int>st;
signed main(){
    ios::sync_with_stdio(false);
    cin>>n>>Q;
    for(int i=1;i<=n;++i){
        cin>>a[i];
        b[i]=a[i];
    }
    sort(b+1,b+n+1);
    int tmp=unique(b+1,b+n+1)-b-1;
    for(int i=1;i<=n;++i){
        a[i]=lower_bound(b+1,b+tmp+1,a[i])-b;
    }
    for(int i=1;i<=Q;++i){
        cin>>q[i].l>>q[i].r;
        q[i].id=i;
    }
    prework();
    sort(q+1,q+Q+1);
    int lastloc=-1;
    int l,r;
    int backl,backr;
    for(int i=1;i<=Q;++i){
        if(locate[q[i].l]==locate[q[i].r]){
            maxn2=0;
            for(int j=q[i].l;j<=q[i].r;++j){
                if(vis2[a[j]]!=i){
                    vis2[a[j]]=i;
                    cnt2[a[j]]=0;
                }
                cnt2[a[j]]++;
                maxn2=max(maxn2,cnt2[a[j]]*b[a[j]]);
            }
            ans[q[i].id]=maxn2;
        }
        else{
            if(locate[q[i].l]!=lastloc){
                // cout<<"nre locate"<<endl;
                memset(cnt,0,sizeof cnt);
                maxn=0;
                lastloc=locate[q[i].l];
                backl=rb[locate[q[i].l]]+1;
                backr=rb[locate[q[i].l]];
                l=backl;r=backr;
            }
            while(r<q[i].r) add(++r);
            memoryans=maxn;
            while(l>q[i].l){
                add(--l);
                st.push(l);
            }
            ans[q[i].id]=maxn;
            maxn=memoryans;
            l=backl;
            while(!st.empty()){
                cnt[a[st.top()]]--;
                st.pop();
            }
        }
    }
    for(int i=1;i<=Q;++i){
        cout<<ans[i]<<'\n';
    }
}

P8078 秃子酋长

暴力思路是直接上莫队,对值域维护一个 set,插入或删除一个元素的贡献都直接在 set 里查前驱和后继,然后统计它们的距离差

暴力
#include<bits/stdc++.h>
using namespace std;
int n,m,len;
int a[500001];
int locate[500001],l[500001],r[500001];
struct ques{
    int l,r;
    int id;
    bool operator<(const ques&A)const{
        if(locate[l]==locate[A.l]) return r<A.r;
        return l<A.l;
    }
}q[500001];
int pos[500001];
int ans[500001];
set<int>s;
int anss;
void prework(){
    len=sqrt(n);
    memset(l,-1,sizeof l);
    for(int i=1;i<=n;++i){
        locate[i]=i/len;
        if(l[locate[i]]==-1) l[locate[i]]=i;
        r[locate[i]]=i;
    }
}
inline void modify(int __pos,bool isadd){
    if(isadd){
        s.insert(a[__pos]);
        auto iter=s.lower_bound(a[__pos]);
        if(iter!=s.begin() and next(iter)!=s.end()){
            anss+=abs(pos[*prev(iter)]-__pos);
            anss+=abs(pos[*next(iter)]-__pos);
            anss-=abs(pos[*prev(iter)]-pos[*next(iter)]);
        }
        else if(iter!=s.begin()){
            anss+=abs(pos[*prev(iter)]-__pos);
        }
        else if(next(iter)!=s.end()){
            anss+=abs(pos[*next(iter)]-__pos);
        }
    }
    else{
        auto iter=s.lower_bound(a[__pos]);
        if(iter!=s.begin() and next(iter)!=s.end()){
            anss-=abs(pos[*prev(iter)]-__pos);
            anss-=abs(pos[*next(iter)]-__pos);
            anss+=abs(pos[*prev(iter)]-pos[*next(iter)]);
        }
        else if(iter!=s.begin()){
            anss-=abs(pos[*prev(iter)]-__pos);
        }
        else if(next(iter)!=s.end()){
            anss-=abs(pos[*next(iter)]-__pos);
        }
        s.erase(a[__pos]);
    }
}
int main(){
    ios::sync_with_stdio(false);
    cin>>n>>m;
    for(int i=1;i<=n;++i){
        cin>>a[i];
        pos[a[i]]=i;
    }
    for(int i=1;i<=m;++i){
        cin>>q[i].l>>q[i].r;
        q[i].id=i;
    }
    prework();
    sort(q+1,q+m+1);
    int l=1,r=0;
    for(int i=1;i<=m;++i){
        while(r<q[i].r) modify(++r,true);
        while(l>q[i].l) modify(--l,true);
        while(r>q[i].r) modify(r--,false);
        while(l<q[i].l) modify(l++,false);
        // for(auto i:s) cout<<i<<' ';
        // cout<<endl;
        ans[q[i].id]=anss;
    }
    for(int i=1;i<=m;++i){
        cout<<ans[i]<<'\n';
    }
}

由于这个 set 复杂度太高,注意到我们用这个 set 主要是为了在值域上查前驱和后继,然而对值域开一个双向链表也能做到这一点

由于链表不支持随机访问,插入元素可能会十分麻烦,复杂度比 set 还高,但是可以很方便地删除,因此想到做只删除的回滚莫队

具体来说,先开一个链表,维护出全局的链表状态和答案,然后跑回滚的时候,左端点固定到块的左端点,右端点跑递减,回滚的时候(这时候你不得不在链表上做增加操作了,否则复杂度又降回 nq 了,但是这里回滚的增加操作比较方便做,因为你不需要统计答案,只需要还原链表的状态,答案可以直接暴力赋值成回滚前记录的答案)

可能有点卡常这个题,不过作为 lxl 出的题还是太松了

#include<bits/stdc++.h>
using namespace std;
template<typename T>
inline void read(T &x){
    int ans=0;char ch=getchar();
    while(!isdigit(ch)) ch=getchar();
    while(isdigit(ch)) ans=ans*10+ch-'0',ch=getchar();
    x=ans;
}
#define int long long
int n,m,len;
int a[500001];
int locate[500001],l[500001],r[500001];
int pos[500001];
int ans[500001];
struct ques{
    int l,r;
    int id;
    inline bool operator<(const ques&A)const{
        if(locate[l]==locate[A.l]) return r>A.r;
        return l<A.l;
    }
}q[500001];
struct hdklist{
    int act_l,act_r;
    int head,tail;
    struct list_t{
        int l,r;
    }ls[500001];
    int ans=0;
    inline void del(int __pos){
        if(__pos!=tail) ans-=abs(pos[ls[__pos].r]-pos[__pos]);
        if(__pos!=head) ans-=abs(pos[ls[__pos].l]-pos[__pos]);
        if(__pos!=tail and __pos!=head) ans+=abs(pos[ls[__pos].l]-pos[ls[__pos].r]);
        if(__pos==tail) tail=ls[__pos].l;
        if(__pos==head) head=ls[__pos].r;
        ls[ls[__pos].l].r=ls[__pos].r;
        ls[ls[__pos].r].l=ls[__pos].l;
    }
    inline void add(int __pos){ //这个东西是个废物函数
        ls[ls[__pos].l].r=__pos;//啥也不管,得从外面记录值然后修改
        ls[ls[__pos].r].l=__pos;//因为这么干复杂度很低所以这么干了
    }
    inline void operator=(const hdklist&A){
        for(int i=1;i<=n;++i){
            ls[i].l=A.ls[i].l;
            ls[i].r=A.ls[i].r;
        }
        head=A.head;
        tail=A.tail;
        act_l=A.act_l;
        act_r=A.act_r;
        ans=A.ans;
    }
};
hdklist t1,t2;
inline void prework(){
    len=sqrt(2*n);
    memset(l,-1,sizeof l);
    for(int i=1;i<=n;++i){
        locate[i]=i/len;
        if(l[locate[i]]==-1) l[locate[i]]=i;
        r[locate[i]]=i;
    }
}
stack<int>st;
signed main(){
    read(n);read(m);
    for(int i=1;i<=n;++i){
        read(a[i]);
        pos[a[i]]=i;
    }
    for(int i=1;i<=m;++i){
        read(q[i].l);read(q[i].r);
        q[i].id=i;
    }
    prework();
    sort(q+1,q+m+1);
    for(int i=1;i<=n;++i){
        if(i!=1) t1.ls[i].l=i-1;
        if(i!=n){
            t1.ls[i].r=i+1;
            t1.ans+=abs(pos[i]-pos[i+1]);
        }
    }
    t1.head=1,t1.tail=n;
    t1.act_l=1,t1.act_r=n;
    int lastloc=-1;
    for(int i=1;i<=m;++i){
        if(locate[q[i].l]!=lastloc){
            lastloc=locate[q[i].l];
            while(t1.act_l<l[locate[q[i].l]]) t1.del(a[t1.act_l++]);
            t2=t1;
        }
        while(t2.act_r>q[i].r) t2.del(a[t2.act_r--]);
        int act_l=t2.act_l,act_r=t2.act_r,anss=t2.ans,head=t2.head,tail=t2.tail;
        while(t2.act_l<q[i].l){
            st.push(a[t2.act_l]);
            t2.del(a[t2.act_l++]);
        }
        ans[q[i].id]=t2.ans;
        while(!st.empty()){
            t2.add(st.top());
            st.pop();
        }
        t2.act_l=act_l;
        t2.act_r=act_r;
        t2.ans=anss;
        t2.head=head;
        t2.tail=tail;
    }
    for(int i=1;i<=m;++i){
        printf("%lld\n",ans[i]);
    }
}

CF916E Jamie and Tree

小杂烩题

迎面而来的第一个问题是换根 LCA

尝试分讨,设当前根为 \(r\)

  • 如果 \(x,y\) 均在 \(r\) 的子树内,答案为 \(\text{lca}(x,y)\)
  • 如果 \(x,y\) 中只有一个在 \(r\) 的子树内,答案为 \(r\)
  • 如果 \(x,y\) 均不在 \(r\) 的子树内
    • 当 \(\text{lca}(x,r)=\text{lca}(y,r)\) 时,\(x,y\) 与 \(r\) 不在当前根节点的同一颗子树内,此时 \(x,y\) 所在子树形态不变,答案仍为 \(\text{lca}(x,y)\)
    • 当 \(\text{lca}(x,r)\neq\text{lca}(y,r)\) 时,\(r\) 一定与某一点位于同一子树内,此时换根后另一点必经过该子树根节点才能到达 \(r\),因此答案为 \(\text{lca}(x,r),\text{lca}(y,r)\) 中较深者

可以发现,分类讨论中出现的公共祖先只有 \(\text{lca}(x,y),\text{lca}(x,r),\text{lca}(y,r)\),因此直接贪心地取深度最大者即可(这三个一定是新树上的合法公共祖先,只是有可能不再是最近的)

迎面而来的第二个问题是换根子树内修改

仍然分讨,设当前根为 \(r\),要修改的子树为 \(x\)

  • 当 \(r=x\),直接修改整颗树(注意这种情况必须单独拉出来考虑,放在下面直接按深度差减一算出负数了)
  • 当 \(r\) 不在 \(x\) 的子树内,换根后形态不变,仍然修改 \(x\) 的子树
  • 当 \(r\) 在 \(x\) 的子树内,此时需要修改的节点是整棵树除去从 \(r\) 到 \(x\) 路径上最靠近 \(x\) 的节点(可以是 \(r\))的子树之后的所有节点,这么说太绕了,但是确实不好表述,建议自行对这种情况画图理解

对于前两种情况,直接树剖后建线段树就行了

对于第三种情况,这个点就相当于是 \((x,r)\) 路径上 \(x\) 点的直属子节点,可以通过类似快速幂的倍增来实现(维护一个 \(fa_{i,x}\)),相当于从 \(r\) 点向上跳 \(deep_r-deep_x-1\)(这就是第一种情况不能分讨进来的原因),情况里的子树减法可以直接转化成贡献相减,修改操作直接对需要减掉的子树做一个负修改就好了

写到这里就做完了,主要是思维难度,码力要求实际上不是很高,没调多久就过了

#include<bits/stdc++.h>
using namespace std;
#define int long long
int n,q;
vector<int>e[100001];
int w[100001],wnew[100001];
int fa[20][100001],deep[100001],siz[100001];
int maxson[100001];
int id[100001],idcnt;
int top[100001];
void dfs1(int now,int last){
    // cout<<"dfs1 "<<now<<" "<<last<<endl;
    fa[0][now]=last;
    deep[now]=deep[last]+1;
    siz[now]=1;
    int maxsonsize=0;
    for(int i:e[now]){
        if(i!=last){
            dfs1(i,now);
            siz[now]+=siz[i];
            if(siz[i]>maxsonsize){
                maxsonsize=siz[i];
                maxson[now]=i;
            }
        }
    }
}
void dfs2(int now,int nowtop){
    id[now]=++idcnt;
    wnew[id[now]]=w[now];
    top[now]=nowtop;
    if(!maxson[now]) return;
    dfs2(maxson[now],nowtop);
    for(int i:e[now]){
        if(i!=fa[0][now] and i!=maxson[now]){
            dfs2(i,i);
        }
    }
}
inline int kth_ancestor(int x,int k){
    int base=0;
    // cout<<x<<"- "<<k<<endl;
    while(k){
        if(k&1){
            assert(base<=19);
            x=fa[base][x];
        }
        base++;
        k>>=1;
    }
    return x;
}
inline int lca(int x,int y){
    while(top[x]!=top[y]){
        if(deep[top[x]]<deep[top[y]]) swap(x,y);
        x=fa[0][top[x]];
    }
    return deep[x]<deep[y]?x:y;
}
inline int deep_maxn(int x,int y){
    return deep[x]>deep[y]?x:y;
}
namespace stree{
    struct tree{
        int sum;
        int lazy;
    }t[100001*4];
    #define tol (id*2)
    #define tor (id*2+1)
    #define mid(l,r) mid=((l)+(r))/2
    void build(int id,int l,int r){
        // cout<<"build "<<id<<" "<<l<<" "<<r<<endl;
        assert(l<=r);
        if(l==r){
            t[id].sum=wnew[l];
            return;
        }
        int mid(l,r);
        build(tol,l,mid);
        build(tor,mid+1,r);
        t[id].sum=t[tol].sum+t[tor].sum;
    }
    void pushdown(int id,int l,int r){
        if(t[id].lazy){
            int mid(l,r);
            t[tol].sum+=t[id].lazy*(mid-l+1);
            t[tol].lazy+=t[id].lazy;
            t[tor].sum+=t[id].lazy*(r-(mid+1)+1);
            t[tor].lazy+=t[id].lazy;
            t[id].lazy=0;
        }
    }
    void change(int id,int l,int r,int L,int R,int val){
        if(L<=l and r<=R){
            t[id].sum+=val*(r-l+1);
            t[id].lazy+=val;
            return;
        }
        int mid(l,r);
        pushdown(id,l,r);
        if(R<=mid) change(tol,l,mid,L,R,val);
        else if(L>=mid+1) change(tor,mid+1,r,L,R,val);
        else{
            change(tol,l,mid,L,mid,val);
            change(tor,mid+1,r,mid+1,R,val);
        }
        t[id].sum=t[tol].sum+t[tor].sum;
    }
    int ask(int id,int l,int r,int L,int R){
        // cout<<"ask "<<id<<" "<<l<<" "<<r<<" "<<L<<" "<<R<<endl;
        if(L<=l and r<=R){
            return t[id].sum;
        }
        int mid(l,r);
        pushdown(id,l,r);
        if(R<=mid) return ask(tol,l,mid,L,R);
        else if(L>=mid+1) return ask(tor,mid+1,r,L,R);
        return ask(tol,l,mid,L,mid)+ask(tor,mid+1,r,mid+1,R);
    }
}
inline void change_subtree(int x,int val){
    stree::change(1,1,n,id[x],id[x]+siz[x]-1,val);
}
inline int ask_subtree(int x){
    // cout<<"asksubtree "<<x<<" "<<siz[x]<<endl;
    return stree::ask(1,1,n,id[x],id[x]+siz[x]-1);
}
inline bool isin_subtree(int x,int subroot){
    if(deep[x]<deep[subroot]) return false;
    if(kth_ancestor(x,deep[x]-deep[subroot])==subroot) return true;
    return false;
}
int root;
signed main(){
    // freopen("in.in","r",stdin);
    ios::sync_with_stdio(false);
    cin>>n>>q;
    for(int i=1;i<=n;++i){
        cin>>w[i];
    }
    for(int i=1;i<=n-1;++i){
        int x,y;
        cin>>x>>y;
        e[x].push_back(y);
        e[y].push_back(x);
    }
    dfs1(1,0);
    dfs2(1,1);
    // cout<<"siz: ";
    // for(int i=1;i<=n;++i) cout<<siz[i]<<' ';
    // cout<<endl;
    root=1;
    for(int i=1;i<=19;++i){
        for(int j=1;j<=n;++j){
            fa[i][j]=fa[i-1][fa[i-1][j]];
        }
    }
    stree::build(1,1,n);
    while(q--){
        int opt;cin>>opt;
        if(opt==1){
            cin>>root;
        }
        else if(opt==2){
            int x,y,v;
            cin>>x>>y>>v;
            int __lca=deep_maxn(deep_maxn(lca(x,y),lca(x,root)),lca(y,root));
            if(__lca==root){
                change_subtree(1,v);
            }
            else if(!isin_subtree(root,__lca)){
                change_subtree(__lca,v);
            }
            else{
                change_subtree(1,v);
                change_subtree(kth_ancestor(root,deep[root]-deep[__lca]-1),-v);
            }
        }
        else{
            int x;cin>>x;
            if(x==root){
                cout<<ask_subtree(1)<<'\n';
            }
            else if(!isin_subtree(root,x)){
                cout<<ask_subtree(x)<<'\n';
            }
            else{
                cout<<ask_subtree(1)-ask_subtree(kth_ancestor(root,deep[root]-deep[x]-1))<<'\n';
            }
        }
    }
}

标签:int,深度,mid,好题,100001,lca,now,id
From: https://www.cnblogs.com/HaneDaCafe/p/18567069

相关文章

  • 【深度学习】铁路轨道缺陷检测
    一、概述铁路轨道在生产、热处理、机械加工等过程中,可能会出现凹陷、未熔合、气孔、裂纹等缺陷影响其性能和寿命,如其不良品流入市场,铁路运输安全将无法得到保障。二、传统算法检测缺点传统算法依赖于手动选择特征,当图像一致性较差的时候需要人工反复调整参数适应检测,在......
  • 深度学习——Transformer
    本文详细介绍面试过程中可能遇到的Transformer知识点。文章目录初识Transformer1.编码器-解码器架构解码器的额外结构2.自注意力机制(Self-AttentionMechanism)解码器中的注意力机制的2点特殊3.位置编码(PositionalEncoding)4.前馈神经网络(Feed-ForwardNeural......
  • 深度学习入门- 梯度(Gradient)(一)
    目录一.梯度的数学基础1.复合函数2.链式法则3.驻点,极值点,鞍点4.偏导数5.梯度6. 梯度法一.梯度的数学基础1.复合函数    由多个函数构成的函数,比如z=(x+y)**2,由函数1: z=t**2和函数2: t=x+y构成。2.链式法则    如果某个函数由复合函数表示,则该复......
  • Java同步机制深度解析: synchronized vs ReentrantLock
    目录1.引言2.synchronized关键字2.1基本用法2.2实现原理2.2.1Java对象头2.2.2Monitor对象2.3锁升级过程2.4锁消除和锁粗化2.5自旋锁与自适应自旋锁2.6synchronized的特性3.ReentrantLock3.1基本用法3.2实现原理3.2.1AQS的核心思想3.2.2ReentrantLock的......
  • 深度测评腾讯云 HAI 智算服务:高性能 AI 计算的新标杆
    本文一、引言二、产品功能深度解析2.1多样化的GPU配置选择2.2预配置开发环境示例:2.3实时性能监控三、核心技术特点与优势3.1云端弹性扩展3.2高性能计算架构四、实际测试与代码案例4.1NLP案例:使用BERT进行情感分类数据集:IMDb评论情感分析环境配置数据加载......
  • 读书的广度与深度
    读书的广度与深度升学以来,出于学习需要,接触到了大量的文字资料,于是自己也有了一些读书方面的感想,今整理下来,以作备忘用。“深度”与“广度”我们常说,读书既要“深”,也要“广”。“深”说的是需要专精某一领域,“广”说的是不能局限在单个领域里,还要广泛接触不同领域的知识。”......
  • Python3.9.13与深度学习框架TensorFlow的完整详细安装教程
    一、Python与TensorFlow版本的关系        TensorFlow的不同版本适用不同的Python版本,这是因为TensorFlow需要与Python的特性保持一致,以便最优化性能和功能。以下是一些主要版本之间的对应关系:TensorFlow版本支持的Python版本2.10.x3.7,3.8,3.92.9.x3.7,......
  • 项目实战:基于深度学习的人脸表情识别系统设计与实现
    大家好,人脸表情识别是计算机视觉领域中的一个重要研究方向,它涉及到对人类情感状态的理解和分析。随着深度学习技术的发展,基于深度学习的人脸表情识别系统因其高精度和强大的特征学习能力而受到广泛关注。本文旨在探讨基于深度学习的人脸表情识别系统的设计与实现,从数据处理、模......
  • PyTorch框架——基于深度学习YOLOv3神经网络目标检测x光安检违禁品检查系统
    基于深度学习神经网络YOLOv3目标检测的x光安检违禁品检查系统,其能识别的违禁品有5种(刀、枪、扳手、钳子、剪刀),见如下:第一步:YOLOv3介绍yoloV3以V1,V2为基础进行的改进,主要有:利用多尺度特征进行目标检测;先验框更丰富;调整了网络结构;对象分类使用logistic代替了softmax,更适用于......
  • 《贪婪算法实战:寻找最短无序连续子数组的深度解析与实现》
    ......