`
jgsj
  • 浏览: 1001480 次
文章分类
社区版块
存档分类
最新评论

Android 广播接收器注册与注销源码分析

 
阅读更多

注册广播接收器

Android系统的广播机制是一种消息订阅/发布机制,因此,使用这种消息驱动模型的第一步便是订阅消息;而对Android应用程序来说,订阅消息其实就是注册广播接收器。在Android的广播机制中,ActivityManagerService扮演着广播中心的角色,负责系统中所有广播的注册和发布操作,因此,Android应用程序注册广播接收器的过程就把是广播接收器注册到ActivityManagerService的过程。Android应用程序是通过调用ContextWrapper类的registerReceiver函数来把广播接收器BroadcastReceiver注册到ActivityManagerService中去的,而ContextWrapper类本身又借助ContextImpl类来注册广播接收器。在Android应用程序框架中,Activity和Service类都继承了ContextWrapper类,因此,我们可以在Activity或者Service的子类中调用registerReceiver函数来注册广播接收器。Activity、Service、ContextWrapper和ContextImpl这四个类的关系:

registerReceiver函数播接收器注册过程的时序图如下:

接下来对registerReceiver注册广播接收器的源码进行详细分析,由于Activity继承于Context类,Context类是一个抽象类,定义了registerReceiver接口函数,因此在Activity的子类中,可以直接调用registerregisterReceiver函数,但是Activity及其父类都没有实现registerReceiver接口函数,在Activity调用registerReceiver来注册广播接收器时,根据Activity类继承关系,会依次调用一遍其父类的registerReceiver函数,父类ContextWrapper对registerReceiver函数实现如下:

@Override
public Intent registerReceiver(
	BroadcastReceiver receiver, IntentFilter filter) {
	return mBase.registerReceiver(receiver, filter);
}

@Override
public Intent registerReceiver(
	BroadcastReceiver receiver, IntentFilter filter,
	String broadcastPermission, Handler scheduler) {
	return mBase.registerReceiver(receiver, filter, broadcastPermission,
			scheduler);
}
这里定义了两个registerReceiver重载函数,以上两个函数的实现都是间接调用mBase对象的registerReceiver函数,mBase定义为Context类型,在Context的子类中,ContextImpl类实现了registerReceiver函数。

@Override
public Intent registerReceiver(BroadcastReceiver receiver, IntentFilter filter) {
	return registerReceiver(receiver, filter, null, null);
}

@Override
public Intent registerReceiver(BroadcastReceiver receiver, IntentFilter filter,
		String broadcastPermission, Handler scheduler) {
	return registerReceiverInternal(receiver, filter, broadcastPermission,
			scheduler, getOuterContext());
}
在ContextImpl类中仍然实现了两个registerReceiver函数的重载,但他们都并没有真正实现该函数,而是调用函数registerReceiverInternal函数来完成,registerReceiverInternal函数的定义如下:

private Intent registerReceiverInternal(BroadcastReceiver receiver,
		IntentFilter filter, String broadcastPermission,
		Handler scheduler, Context context) {
	/* broadcastPermission = null
	   scheduler = null */
	IIntentReceiver rd = null;
	if (receiver != null) {
		if (mPackageInfo != null && context != null) {
			if (scheduler == null) {
				//获取应用程序主线程的handle对象
				scheduler = mMainThread.getHandler();
			}
			//获取IIntentReceiver广播接收分发器
			rd = mPackageInfo.getReceiverDispatcher(
				receiver, context, scheduler,
				mMainThread.getInstrumentation(), true);
		} else {
			if (scheduler == null) {
				scheduler = mMainThread.getHandler();
			}
			rd = new LoadedApk.ReceiverDispatcher(
					receiver, context, scheduler, null, true).getIIntentReceiver();
		}
	}
	try {
		/* ActivityManagerNative.getDefault()得到ActivityManagerService在客户进程的代理类对象
		ActivityManagerProxy,通过Binder进程间通信,远程调用ActivityManagerService的registerReceiver函数 */
		return ActivityManagerNative.getDefault().registerReceiver(
				mMainThread.getApplicationThread(), mBasePackageName,
				rd, filter, broadcastPermission);
	} catch (RemoteException e) {
		return null;
	}
}

获取Activity的上下文

final Context getOuterContext() {
	return mOuterContext;
}

ContextImpl() {
 mOuterContext = this;
}
由于ContextImpl类为Activity的父类,因此getOuterContext()函数得到的是注册广播接收器的Activity的上下文。

获取应用程序主线程的消息分发器handle

mMainThread.getHandler()

frameworks\base\core\java\android\app\ActivityThread.java

final Handler getHandler() {
	return mH;
}

final H mH = new H();

private class H extends Handler {
	public static final int LAUNCH_ACTIVITY         = 100;
	public void handleMessage(Message msg) {
		if (DEBUG_MESSAGES) Slog.v(TAG, ">>> handling: " + codeToString(msg.what));
		switch (msg.what) {
			case LAUNCH_ACTIVITY: {
				Trace.traceBegin(Trace.TRACE_TAG_ACTIVITY_MANAGER, "activityStart");
				ActivityClientRecord r = (ActivityClientRecord)msg.obj;

				r.packageInfo = getPackageInfoNoCheck(
						r.activityInfo.applicationInfo, r.compatInfo);
				handleLaunchActivity(r, null);
				Trace.traceEnd(Trace.TRACE_TAG_ACTIVITY_MANAGER);
			} break;
		}
	}
}
该handle是在ActivityThread类中定义,用于处理应用程序主线程的消息分发处理。

获取广播接收分发器

frameworks\base\core\java\android\app\LoadedApk.java

public IIntentReceiver getReceiverDispatcher(BroadcastReceiver r,
		Context context, Handler handler,
		Instrumentation instrumentation, boolean registered) {
	synchronized (mReceivers) {
		LoadedApk.ReceiverDispatcher rd = null;
		HashMap<BroadcastReceiver, LoadedApk.ReceiverDispatcher> map = null;
		//注册广播接收器
		if (registered) {
			//以注册广播接收器的Activity为key从mReceivers的Hashmap表中取出该上下文对应的广播接收器表
			map = mReceivers.get(context);
			//根据BroadcastReceiver从广播接收器表中取出对应的LoadedApk.ReceiverDispatcher
			if (map != null) {
				rd = map.get(r);
			}
		}
		//如果表中不存在,直接创建ReceiverDispatcher对象
		if (rd == null) {
			rd = new ReceiverDispatcher(r, context, handler,
					instrumentation, registered);
			//并且以<BroadcastReceiver,LoadedApk.ReceiverDispatcher>为键值对的形式保存
			if (registered) {
				if (map == null) {
					map = new HashMap<BroadcastReceiver, LoadedApk.ReceiverDispatcher>();
					mReceivers.put(context, map);
				}
				map.put(r, rd);
			}
		} else {
			rd.validate(context, handler);
		}
		rd.mForgotten = false;
		//取得ReceiverDispatcher对象的成员变量mIIntentReceiver
		return rd.getIIntentReceiver();
	}
}
注册的广播接收器的存储方式如下:

LoadedApk.mReceivers


该广播接收器表保存在LoadedApk对象的成员变量mReceivers中。

IIntentReceiver getIIntentReceiver() {
	return mIIntentReceiver;
}

ReceiverDispatcher(BroadcastReceiver receiver, Context context,
		Handler activityThread, Instrumentation instrumentation,
		boolean registered) {
	if (activityThread == null) {
		throw new NullPointerException("Handler must not be null");
	}
	mIIntentReceiver = new InnerReceiver(this, !registered);
	mReceiver = receiver;
	mContext = context;
	mActivityThread = activityThread;
	mInstrumentation = instrumentation;
	mRegistered = registered;
	mLocation = new IntentReceiverLeaked(null);
	mLocation.fillInStackTrace();
}

InnerReceiver(LoadedApk.ReceiverDispatcher rd, boolean strong) {
 mDispatcher = new WeakReference<LoadedApk.ReceiverDispatcher>(rd);
 mStrongRef = strong ? rd : null;
}

ActivityManagerProxy注册广播接收器

public Intent registerReceiver(IApplicationThread caller, String packageName,
		IIntentReceiver receiver,
		IntentFilter filter, String perm) throws RemoteException
{
	Parcel data = Parcel.obtain();
	Parcel reply = Parcel.obtain();
	data.writeInterfaceToken(IActivityManager.descriptor);
	data.writeStrongBinder(caller != null ? caller.asBinder() : null);
	data.writeString(packageName);
	data.writeStrongBinder(receiver != null ? receiver.asBinder() : null);
	filter.writeToParcel(data, 0);
	data.writeString(perm);
	mRemote.transact(REGISTER_RECEIVER_TRANSACTION, data, reply, 0);
	reply.readException();
	Intent intent = null;
	int haveIntent = reply.readInt();
	if (haveIntent != 0) {
		intent = Intent.CREATOR.createFromParcel(reply);
	}
	reply.recycle();
	data.recycle();
	return intent;
}

ActivityManagerProxy通过Binder驱动程序远程调用服务进程的ActivityManagerService的registerReceiver函数来注册广播接收器。

ActivityManagerService注册广播接收器

关于广播接收器的数据结构图:

public Intent registerReceiver(IApplicationThread caller, String callerPackage,
		IIntentReceiver receiver, IntentFilter filter, String permission) {
	enforceNotIsolatedCaller("registerReceiver");
	synchronized(this) {
		ProcessRecord callerApp = null;
		if (caller != null) {
			//caller = mMainThread.getApplicationThread()
			//获取应用程序的ProcessRecord
			callerApp = getRecordForAppLocked(caller);
			if (callerApp == null) {
				throw new SecurityException(
						"Unable to find app for caller " + caller
						+ " (pid=" + Binder.getCallingPid()
						+ ") when registering receiver " + receiver);
			}
			if (callerApp.info.uid != Process.SYSTEM_UID &&
					!callerApp.pkgList.contains(callerPackage)) {
				throw new SecurityException("Given caller package " + callerPackage
						+ " is not running in process " + callerApp);
			}
		} else {
			callerPackage = null;
		}

		List allSticky = null;

		// Look for any matching sticky broadcasts...
		Iterator actions = filter.actionsIterator();
		if (actions != null) {
			while (actions.hasNext()) {
				String action = (String)actions.next();
				allSticky = getStickiesLocked(action, filter, allSticky);
			}
		} else {
			allSticky = getStickiesLocked(null, filter, allSticky);
		}

		// The first sticky in the list is returned directly back to
		// the client.
		Intent sticky = allSticky != null ? (Intent)allSticky.get(0) : null;

		if (receiver == null) {
			return sticky;
		}
        //从mRegisteredReceivers表中取出receiver对应的ReceiverList,ReceiverList列表用于保存BroadcastFilter
		ReceiverList rl = (ReceiverList)mRegisteredReceivers.get(receiver.asBinder());
		if (rl == null) {
			//创建ReceiverList对象
			rl = new ReceiverList(this, callerApp,
					Binder.getCallingPid(),
					Binder.getCallingUid(), receiver);
			if (rl.app != null) {
				//保存到ProcessRecord的receivers成员变量中
				rl.app.receivers.add(rl);
			} else {
				try {
					receiver.asBinder().linkToDeath(rl, 0);
				} catch (RemoteException e) {
					return sticky;
				}
				rl.linkedToDeath = true;
			}
			//注册到mRegisteredReceivers HashMap表中
			mRegisteredReceivers.put(receiver.asBinder(), rl);
		}
		//根据IntentFilter创建广播接收过滤器
		BroadcastFilter bf = new BroadcastFilter(filter, rl, callerPackage, permission);
		//为当前广播接收器设置Action过滤器
		rl.add(bf);
		if (!bf.debugCheck()) {
			Slog.w(TAG, "==> For Dynamic broadast");
		}
		mReceiverResolver.addFilter(bf);

		// Enqueue broadcasts for all existing stickies that match
		// this filter.
		if (allSticky != null) {
			ArrayList receivers = new ArrayList();
			receivers.add(bf);

			int N = allSticky.size();
			for (int i=0; i<N; i++) {
				Intent intent = (Intent)allSticky.get(i);
				BroadcastQueue queue = broadcastQueueForIntent(intent);
				BroadcastRecord r = new BroadcastRecord(queue, intent, null,
						null, -1, -1, null, receivers, null, 0, null, null,
						false, true, true);
				queue.enqueueParallelBroadcastLocked(r);
				queue.scheduleBroadcastsLocked();
			}
		}

		return sticky;
	}
}

在ActivityManagerService中,用一个进程记录块来表示这个应用程序进程,它里面有一个列表receivers,专门用来保存这个进程注册的广播接收器。接着,又把这个ReceiverList列表以receiver为Key值保存在ActivityManagerService的成员变量mRegisteredReceivers中。创建BroadcastFilter来把广播接收器列表rl和filter关联起来,然后保存在ActivityManagerService中的成员变量mReceiverResolver中去。

注消广播接收器

frameworks\base\core\java\android\content\ContextWrapper.java

public void unregisterReceiver(BroadcastReceiver receiver) {
	mBase.unregisterReceiver(receiver);
}
frameworks\base\core\java\android\app\ContextImpl.java

public void unregisterReceiver(BroadcastReceiver receiver) {
	if (mPackageInfo != null) {
		IIntentReceiver rd = mPackageInfo.forgetReceiverDispatcher(
				getOuterContext(), receiver);
		try {
			ActivityManagerNative.getDefault().unregisterReceiver(rd);
		} catch (RemoteException e) {
		}
	} else {
		throw new RuntimeException("Not supported in system context");
	}
}
函数首先从表中查找指定广播接收器的IIntentReceiver

public IIntentReceiver forgetReceiverDispatcher(Context context,
		BroadcastReceiver r) {
	synchronized (mReceivers) {
		//从mReceivers表中取出对应的广播接收器表
		HashMap<BroadcastReceiver, LoadedApk.ReceiverDispatcher> map = mReceivers.get(context);
		LoadedApk.ReceiverDispatcher rd = null;
		if (map != null) {
			//从广播接收器表中取出指定的广播接收器的分发器
			rd = map.get(r);
			if (rd != null) {
				map.remove(r);
				if (map.size() == 0) {
					mReceivers.remove(context);
				}
				if (r.getDebugUnregister()) {
					HashMap<BroadcastReceiver, LoadedApk.ReceiverDispatcher> holder
							= mUnregisteredReceivers.get(context);
					if (holder == null) {
						holder = new HashMap<BroadcastReceiver, LoadedApk.ReceiverDispatcher>();
						mUnregisteredReceivers.put(context, holder);
					}
					RuntimeException ex = new IllegalArgumentException(
							"Originally unregistered here:");
					ex.fillInStackTrace();
					rd.setUnregisterLocation(ex);
					holder.put(r, rd);
				}
				//返回ReceiverDispatcher的IntentReceiver
				rd.mForgotten = true;
				return rd.getIIntentReceiver();
			}
		}
		HashMap<BroadcastReceiver, LoadedApk.ReceiverDispatcher> holder
				= mUnregisteredReceivers.get(context);
		if (holder != null) {
			rd = holder.get(r);
			if (rd != null) {
				RuntimeException ex = rd.getUnregisterLocation();
				throw new IllegalArgumentException(
						"Unregistering Receiver " + r
						+ " that was already unregistered", ex);
			}
		}
		if (context == null) {
			throw new IllegalStateException("Unbinding Receiver " + r
					+ " from Context that is no longer in use: " + context);
		} else {
			throw new IllegalArgumentException("Receiver not registered: " + r);
		}

	}
}
通过ActivityManagerProxy代理向ActivityManagerService发送注销广播接收器的请求

public void unregisterReceiver(IIntentReceiver receiver) throws RemoteException
{
	Parcel data = Parcel.obtain();
	Parcel reply = Parcel.obtain();
	data.writeInterfaceToken(IActivityManager.descriptor);
	data.writeStrongBinder(receiver.asBinder());
	mRemote.transact(UNREGISTER_RECEIVER_TRANSACTION, data, reply, 0);
	reply.readException();
	data.recycle();
	reply.recycle();
} 
ActivityManagerService负责广播接收器的注销工作:

public void unregisterReceiver(IIntentReceiver receiver) {
	if (DEBUG_BROADCAST) Slog.v(TAG, "Unregister receiver: " + receiver);

	final long origId = Binder.clearCallingIdentity();
	try {
		boolean doTrim = false;
		synchronized(this) {
			//从mRegisteredReceivers表中查找到receiver对应的ReceiverList
			ReceiverList rl = (ReceiverList)mRegisteredReceivers.get(receiver.asBinder());
			if (rl != null) {
				if (rl.curBroadcast != null) {
					BroadcastRecord r = rl.curBroadcast;
					final boolean doNext = finishReceiverLocked(
							receiver.asBinder(), r.resultCode, r.resultData,
							r.resultExtras, r.resultAbort, true);
					if (doNext) {
						doTrim = true;
						r.queue.processNextBroadcast(false);
					}
				}

				if (rl.app != null) {
					//从应用进程的receivers表中移除当前广播接收器对应的ReceiverList
					rl.app.receivers.remove(rl);
				}
				//从mRegisteredReceivers表中移除receiver,同时从mReceiverResolver的过滤Action
				removeReceiverLocked(rl);
				if (rl.linkedToDeath) {
					rl.linkedToDeath = false;
					rl.receiver.asBinder().unlinkToDeath(rl, 0);
				}
			}
		}

		// If we actually concluded any broadcasts, we might now be able
		// to trim the recipients' apps from our working set
		if (doTrim) {
			trimApplications();
			return;
		}

	} finally {
		Binder.restoreCallingIdentity(origId);
	}
}
至此广播接收器的注册与注销就介绍完了,注册的本质其实就是将广播接收器添加到ActivityManagerService的相应成员变量中存储,从而在分发广播的时候,可以根据表中注册的接收器来分发;而广播接收器的注销工作就是从相应的存储表中移除。

分享到:
评论

相关推荐

    1300张图片训练效果

    1300张图片训练效果

    springboot116基于java的教学辅助平台.zip

    教学辅助平台的出现,是为了更好地服务于教育工作者和学生,提高教学效果和学习效率。该平台集成了多个功能模块,旨在为用户提供全面、便捷的教学辅助服务。 平台首页作为导航入口,提供了清晰的界面布局和便捷的导航功能,方便用户快速找到所需功能。需要注意的是,“首页”这一选项在导航菜单中出现了多次,可能是设计上的冗余,需要进一步优化。 “个人中心”模块允许用户查看和管理自己的个人信息,包括修改密码等账户安全设置,确保用户信息的准确性和安全性。 在教育教学方面,“学生管理”和“教师管理”模块分别用于管理学生和教师的信息,包括学生档案、教师资料等,方便教育工作者进行学生管理和教学安排。同时,“课程信息管理”、“科目分类管理”和“班级分类管理”模块提供了课程信息的发布、科目和班级的分类管理等功能,有助于教育工作者更好地组织和管理教学内容。 此外,“课程作业管理”模块支持教师布置和批改作业,学生可以查看和提交作业,实现了作业管理的线上化,提高了教学效率。而“交流论坛”模块则为学生和教师提供了一个交流和讨论的平台,有助于促进师生互动和学术交流。 最后,“系统管理”模块为平台管理员提供了系统配置.

    yolo算法-火灾探测数据集-3466张图像带标签-火灾fire_detect-oqlpv.zip

    yolo系列算法目标检测数据集,包含标签,可以直接训练模型和验证测试,数据集已经划分好,适用yolov5,yolov8,yolov9,yolov7,yolov10,yolo11算法; 包含两种标签格:yolo格式(txt文件)和voc格式(xml文件),分别保存在两个文件夹中; yolo格式:<class> <x_center> <y_center> <width> <height>, 其中: <class> 是目标的类别索引(从0开始)。 <x_center> 和 <y_center> 是目标框中心点的x和y坐标,这些坐标是相对于图像宽度和高度的比例值,范围在0到1之间。 <width> 和 <height> 是目标框的宽度和高度,也是相对于图像宽度和高度的比例值

    基于go语言的参数解析校验器项目资源.zip

    基于go语言的参数解析校验器项目资源

    matlab主成分分析代码

    matlab主成分分析代码

    华南农业大学在四川2020-2024各专业最低录取分数及位次表.pdf

    那些年,与你同分同位次的同学都去了哪里?全国各大学在四川2020-2024年各专业最低录取分数及录取位次数据,高考志愿必备参考数据

    Spire.XLS是一个基于.NET的组件

    Spire.XLS是一个基于.NET的组件,使用它我们可以创建Excel文件,编辑已有的Excel并且可以转换Excel文件.dll

    基于爬虫技术的股票分析系统.doc

    现如今,随着互联网的发展,人们获取信息的方式也各有不同。以前的传统方式的信息流与电视,报纸,书籍,信件,等等,因为互联网的使用,现在的互联网媒体已经成为人们获取信息的最重要来源。更新互联网,让人们得到最新、最完整的信息变得越来越容易。 现在企业已经越来越重视互联网所能带来的利益,借助互联网来对自己的企业进行营销推广已经获得绝大部分企业的认可。本文我们主要进行的是股票分析系统网站的设计。何为股票分析,就是指股票投资人之间的根据市场价格对已发行上市的股票进行的买卖。而国内股票市场的迅速发展让这次开发设计显得十分必要。通过该股票分析系统网站,我们可以随时随地通过该股票分析网站,了解股票行业最新信息;根据股票行业分析来进行相关交易。本网站采用的是Springboot技术和mongodb数据库,运用 stock、 vue2、echarts、bootstrap等技术,使用eclipse开发工具完成股票数据的爬取分析。

    厨房食品佐料检测数据集VOC+YOLO格式602张18类别.zip

    文件太大放服务器了,请先到资源详情查看然后下载 样本图参考:blog.csdn.net/2403_88102872/article/details/143395913 数据集格式:Pascal VOC格式+YOLO格式(不包含分割路径的txt文件,仅仅包含jpg图片以及对应的VOC格式xml文件和yolo格式txt文件) 图片数量(jpg文件个数):602 标注数量(xml文件个数):602 标注数量(txt文件个数):602 标注类别数:18 标注类别名称:["apple","chocolate","cloth","cononut_water","detergent","fanta","gelatin","kuat","mustard","nescau","peanut","pear","sauce","shoyo","sponge","tangerine","tea","treloso"] 18种常见的厨房食品和佐料,包括苹果、巧克力、椰子水、洗涤剂、饮料、明胶、芥末、花生、酱油等

    基于卷积神经网络参数优化的情感分析论文code_cnn-text-classification.zip

    基于卷积神经网络参数优化的情感分析论文code_cnn-text-classification

    河北传媒学院在四川2020-2024各专业最低录取分数及位次表.pdf

    那些年,与你同分同位次的同学都去了哪里?全国各大学在四川2020-2024年各专业最低录取分数及录取位次数据,高考志愿必备参考数据

    Python实现的人脸识别系统及其应用

    内容概要:本文档详细描述了一个基于 Python 的人脸识别系统的构建过程,涵盖了从系统设计理念到具体功能实现的各个方面。首先介绍了系统总体设计流程,包括摄像头图像捕获、人脸检测、特征值计算、特征均值处理以及分类识别。接着深入探讨了 Dlib、NumPy、OpenCV 等关键技术库的应用,特别是 Dlib 人脸检测器接口、人脸预测器接口和人脸识别模型的具体使用方法。最后,本文档介绍了如何通过 Euclidean 距离进行人脸特征比对,实现人脸的成功识别与身份确认。此外,还讨论了人脸识别在实际生活中的多种应用场景和重要意义。 适用人群:具有一定编程基础的软件开发者和技术爱好者,尤其是从事机器学习、图像处理和计算机视觉领域的专业技术人员。 使用场景及目标:①开发人脸识别系统,实现实时图像处理和人脸特征提取;②掌握 Dlib、NumPy、OpenCV 等技术库的实际应用技巧;③深入了解人脸识别技术在安全监控、身份认证、智慧社区等领域的应用。 其他说明:本文档提供了丰富的理论背景和技术实现细节,帮助读者更好地理解和应用人脸识别技术。此外,还包括了一些实用的编码技巧和最佳实践,有助于提高开发效率和代码质量。

    轻量级高性能GO语言开发框架。支持MVC、依赖注入、动态返回.zip

    轻量级高性能GO语言开发框架。支持MVC、依赖注入、动态返回

    stm32的串口hex文件发送与文本文件发送

    stm32的串口hex文件发送与文本文件发送

    广西医科大学在四川2020-2024各专业最低录取分数及位次表.pdf

    那些年,与你同分同位次的同学都去了哪里?全国各大学在四川2020-2024年各专业最低录取分数及录取位次数据,高考志愿必备参考数据

    macOS_Sonoma_14.1.1.rdr.split.003

    macOS_Sonoma_14.1.1.rdr.split.003

    天津财经大学珠江学院在四川2020-2024各专业最低录取分数及位次表.pdf

    那些年,与你同分同位次的同学都去了哪里?全国各大学在四川2020-2024年各专业最低录取分数及录取位次数据,高考志愿必备参考数据

    400699526844862小爱同学.apk

    400699526844862小爱同学.apk

    微信小程序在校内服务平台的应用及开发

    内容概要:本文介绍了基于微信小程序的校园一体化服务平台的设计与开发。该平台利用微信小程序的便捷性和广泛的用户基础,结合JSP技术和MySQL数据库,实现了个人中心、用户管理、寻物启事管理、物品分类管理、失物招领管理、表白广场管理、吐槽大会管理、二手交易管理、交易类型管理、拼车出行管理和系统管理等多项功能。整个系统具有操作简单、易维护、灵活实用等特点。 适合人群:具有一定编程基础的学生和教师,以及希望深入了解微信小程序开发的技术人员。 使用场景及目标:主要用于高校内的信息管理,如失物招领、物品分类、二手交易等,提升校园生活的便捷性和效率,改善用户体验。 其他说明:系统开发过程中,重点考虑了技术可行性、经济可行性和操作可行性,并进行了详细的系统测试,确保系统的稳定性和可靠性。

    Java课程设计之销售管理系统

    (1)课程设计项目简单描述 鉴于当今超市产品种类繁多,光靠人手动的登记已经不能满足一般商家的需求。我们编辑该程序帮助商家完成产品、商家信息的管理,包括产品、客户、供应商等相关信息的添加、修改、删除等功能。 (2)需求分析(或是任务分析) 1)产品类别信息管理:对客户的基本信息进行添加、修改和删除。 2)产品信息管理:对产品的基本信息进行添加、修改和删除。 3)供应商信息管理: 对供应商的基本信息进行添加、修改和删除。 4)订单信息管理:对订单的基本信 息进行添加、修改和删除。 5)统计报表:按选择日期期间,并按产品类别分组统 计订单金额,使用表格显示统计结果

Global site tag (gtag.js) - Google Analytics